日期:2011-10-31 浏览次数:20643 次
当我们打算创建一个.net程序时(包括桌面程序或者Web应用程序),如果能使用其他语言来扩展这个.net程序的功能的话那肯定会相当有实用价值。
比如某些用户可以写一个简单脚本来设置这个程序的一些设定,或者在程序中修改数据是如何持久化保存的,或者为这个.net程序写一个简单的插件。在这篇文章,我们来看看如何让php作为.net程序的脚本语言
显然这样做有很多的好处:
1,很多程序员都会写一些基本的PHP代码,甚至一个初级程序员都能为你的应用写一个简单的PHP脚本代码
2, PHP是非常容易使用的,网络上已经有了一大堆现成的php代码片段可以拿来复制后直接使用
3,归功于Phalanger库( http://phalanger.codeplex.com/), PHP代码能够很容易地获取任何.net库以及调用几乎所有.net程序提供的服务
上面描述的场景仅仅只是使用Phalanger from C#(或者其他编程语言)在运行时生成PHP代码的一小部分案例,打个比方,你能想象一下一个web网络架构使用C#来写域名模块然后使用PHP去搭建用户接口会是什么样子. 所以本文将展示如何在C#的程序中运行PHP代码,与怎么使用全局变量作为参数传递到PHP代码,以及如何读取标准.net流。
Phalanger 是一个将PHP脚本编译成.net字节码的编译器,它本身就被设计用来允许无缝地让.net与其他语言进行双向的互操作性。
这就意味着你能在php代码中调用.net方法以及使用.net的类(http://wiki.phpcompiler.net/.NET_interoperability),同时你也能在C#或者F#中调用php的方法以及使用php的类.( http://wiki.phpcompiler.net/Code_Samples/Standard_mode_interoperability)
同时本文展示了另外一种使用Phalanger的方式:通过.net程序来运行php代码.尤其当被运行的代码是动态获取的或者无法被预编译为程序集时(例如当代码是后来被用户所写的这种情况).当运行的的php代码没有任何改变时,一般你应该使用预编译的脚本库( http://wiki.phpcompiler.net/Code_Samples/Standard_mode_interoperability),这样能够得到更高的效率因为在运行时它们不会参与编译。
配置
在ASP.NET 4.0 C#的网站程序中我已经测试过这个技术了,当然,在.net控制台程序或者winforms这样的桌面应用程序中也是可行的。但要记住你的.net程序必须是使用.net 4.0(full profile)作为目标.net框架,以及必须引用至少一个Phalanger的程序集:“PhpNetCore, Version=2.1.0.0, Culture=neutral, PublicKeyToken=0A8E8C4C76728C71". Phalanger必须在你的应用程序中正确配置。虽然它一样可以被手动配置(http://www.php-compiler.net/blog/2011/installation-free-phalanger-web),但最简单的方式就是使用安装器了。
源码
不可思议的是运行PHP代码的核心就是PHP.Core.DynamicCode.Eval这个方法, 它在PhpNetCore.dll程序集中,唯一有些麻烦的可能就是方法所需的大量参数了。首先我们需要一个可用的PHP.Core.ScriptContext实例, 这就是Phalanger的运行php代码的执行实例。你能从当前线程上获取一个这样的实例.特别注意PHP不是多线程的,所以ScriptContext只是仅仅与一个线程紧密关联
1var context = PHP.Core.ScriptContext.CurrentContext;
然后我们将设置ScriptContext的输出方式,这样PHP脚本才能转换出我们所需要的流。这里我们将设置两个输出方式 - 字节流以及文本流。注意在最后你必须销毁这些流,以至于所有的数据将会被正确的刷新
1context.OutputStream = output;
2using (context.Output = new System.IO.StreamWriter(output)) {
我们也能在ScriptContext中设置全局变量,这样我们也能很方便的传输一些参数到运行的PHP代码中。
1Operators.SetVariable(context, null, "X", "Hello World!");
最终我们将使用的Eval方法来运行PHP代码. 而这个方法实际上被Phalanger内部用来处理PHP的 eval() 表达式.所以这就是为什么这个方法有如此多参数的原因。
01// evaluate our code:
02return DynamicCode.Eval(
03 code,
04 false,/*phalanger internal stuff*/
05 context,
06 null,/*local variables*/
07 null,/*reference to "$this"*/
08 null,/*current class context*/
09 "Default.aspx.cs",/*file name, used for debug and cache key*/
10 1,1,/*position in the file used for debug and cache key*/
11 -1,/*something internal*/
12 null/*current namespace, used in CLR mode*/