故事背景是这样的,有一套项目,服务器端是用C#写的,为了完成某种事情,它需要使用到一个组件,这个组件很小但很重要,很不巧的是,这个这个组件是用PHP语言写的,如果为了使用这个组件而专门搭建一个PHP的环境显得有点高射炮打蚊子(况且还有其他不可预见的阻力)。或许有读者会提出“*”:不是PHP写的么,直接看源码翻译出一份C#版的不就行了?然而事实并不如想象的美好,总之就是短期内无法这么做了。
如今的C#已经非常强大了,它除了可以做我们普通的站点开发、桌面开发和原生的Windows Phone、应用商店开发,还可以做其他诸如IOS、安卓开发;也通过用CLR来托起一个JVM(这里指IKVM.NET)来跑Java应用程序,当然也可以通过把PHP编译成IL来跑PHP的网站程序了。
本篇中,我们就如何进行PHP与C#混合编程作如下讨论:
(1)、PHP与C#的胶水:“Phalanger”
(2)、牛刀小试跑跑PHP
(3)、如何添加PHP类库
(4)、C#与PHP混合互调
(5)、美丽如玫瑰,采摘须谨慎
本文中的示例代码请点击这里进行下载。
1、什么是Phalanger
什么事Phalanger,最简单的概括就是,它能够把PHP编译成DLL供我们C#调用。
Phalanger的官网是:http://www.php-compiler.net/
我们要做混合编程,首先得先从官网那里下载一个安装包。
下载下来之后,打开压缩包,双击Setup开始安装。
把这些都给安装了,前面两个必装,最后一个是模板文件,不想装就不装,已经安装的话它会写着“Already installed”。由图中,我们可以得知一个信息,那就是它只支持Framework 4.0。
把上面的东西安装好之后,我们需要的“胶水”就得到了。
2、在.NET里面跑一个php
嗯,phpinfo(),我没有猜出,这个函数应该是各位读者们第一个敲的,也是搭建完PHP环境之后必敲写的一个函数。我们也牛刀小试一番,试试执行这个函数。
我们先创建一个空的WebApplication
然后添加一个PHP文件,由于我们没有直接的PHP文件添加,所以我们随便添加了一个内容少一点的文件,然后改它的后缀为php并删光里面的代码。
写上我们的代码(还带有智能提示,先进!!!):
然后按下“F5”
然后就奇迹般的出现了我们想要的页面。
3、添加PHP的类库
事实上,即使我们是写原生的PHP,我们都需要使用到大量的PHP库,譬如MySQL库、GD库、CURL库等。在这里当然也不例外,我们也需要使用那些库。不过在这里,我们所使用的库并不是PHP/ext中的那些库,而是Phalanger给我们准备好的库,它们随着Phalanger的安装一同安装到我们的电脑当中,有兴趣的读者可以翻开GAC目录,里面会多了很多php打头的文件夹,那些就是与Phalanger相关的库了。
需要使用哪个库,就自行的在WebConfig的phpNet节点下添加,譬如我需要用MySQL的库,则在WebConfig这样配置
<?xml version="1.0" encoding="utf-8"?> <!-- 有关如何配置 ASP.NET 应用程序的详细消息,请访问 http://go.microsoft.com/fwlink/?LinkId=169433 --> <configuration> <system.web> <compilation debug="true" targetFramework="4.0" /> </system.web> <phpNet> <classLibrary> <add assembly="PhpNetMySql, Version=3.0.0.0, Culture=neutral, PublicKeyToken=2771987119c16a03" section="mysql"/> </classLibrary> </phpNet> </configuration>
这里注意一点,MYSQL的扩展库请使用CodePlex上的,而非自带的,自带的版本链接高板MYSQL可能会遇到问题(地址如下:http://phalangermysql.codeplex.com/releases/view/103022)
这里还有另外一点需要注意,CodePlex上的扩展下载下来之后是这样的。
MySql的库命名有问题,正确的应该是“MySql.Data.dll”,各位读者请留意。
然后就是写下我们读取数据库信息的代码:
<?php /***** 判断库是否被加载 *****/ $extensionName="mysql"; if(!extension_loaded($extensionName)){ echo $extensionName.‘没有被加载进来‘; exit; } /**** 操作mysql ****/ $host="192.168.70.128"; $name="root"; $pwd="root"; $conn=mysql_connect($host,$name,$pwd) or die("mysql数据库连接失败"); mysql_select_db("phalangerdb",$conn)or die("无法选择数据库"); mysql_query("set names utf8"); $sqlstr="select * from person"; $result = mysql_query($sqlstr); echo ‘<pre>‘; while($row=mysql_fetch_row($result)){ print_r($row); } echo ‘</pre>‘; mysql_free_result($result); mysql_close($conn);
然后运行并查看结果:
它成功的读取到我们数据库的东西并输出到页面中。
4、C#与PHP互调
既然是混合编程,如果没有两种语言之间的相互调用那又如何能够称得上混合编程呢?本节中,我们主要分两个部分,其一就是PHP调C#函数,其二就是C#掉PHP函数。
PHP调用C#,Phalanger官网中主要的范例均是此,这里就不作讲解了,有需要的读者可移步到Phalanger官网中看里面的Blog。本文中讲解的是使用概率更高同时也是官网中比较缺乏资料的C#调用PHP的函数。
首先我们先创建一个PHP的函数:
<?php function Sum($a,$b){ return $a+$b; } function SayHello(){ return ‘Hello,我是小蝶惊鸿‘; }
然后新建一个WebForm页面程序,并在CodeBehind中调用它(这里是简单加法例子):
namespace PhalangerDemo.demo3 { using System; public partial class WebForm1 : System.Web.UI.Page { private PHP.Core.ScriptContext phpContext; public WebForm1() { phpContext = PHP.Core.ScriptContext.CurrentContext; phpContext.Include("Fun.php", true); } protected void Page_Load(object sender, EventArgs e) { var context = phpContext.Call("SayHello", new object[] { }).ToString(); Response.Write(context.StartsWith("&") ? context.Substring(1) : context); } protected void btnAdd_Click(object sender, EventArgs e) { var num1 = txtNum1.Text as object; var num2 = txtNum2.Text as object; var result = phpContext.Call("Sum", new object[] { num1, num2 }).ToString(); txtRes.Text = result.StartsWith("&") ? result.Substring(1) : result; } } }
然后再页面中运行看看,Cool:
这里,我要对CodeBehind(C#)部分的代码进行下讲解,它的原理大概如下:
(1)、先获取PHP的上下文对象PHPContext
(2)、然后往PHPContext中require_once我们写的“Fun.php”
(3)、通过Call方法,第一个参数传入方法名,第二个参数传入Object数组类型的参数。调用PHP中的函数
(4)、在被调用的PHP函数执行完成之后,将它的返回接收回来。并过滤开头的“&”字符。
至于更深入的,譬如如何New一个PHP的Class之类的,我没有进行深入的研究,所以这里就不作描述,有兴趣的读者可以自行深入研究,同时也欢迎有此经验的读者进行分享。
5、这美丽又丑陋的Phalanger
在前面的几章中,Phalanger的表现是如此的美丽优秀,它好比一朵玫瑰,看起来是那么的鲜艳,闻起来是那么的幽香,但是当你想采摘它的时候,手中拿着的确是幽艳玫瑰之下的荆棘。也只有你划破伤口流着血的时候,你才感觉到原来采摘这朵玫瑰是那么的痛。
我把开篇时的故事继续说完,到了这一步,读者们大概也可以猜到是个什么情况了,没错,我的尝试失败了,这个PHP的组件无法正常的运行。
下面,我给各位读者分享两个这个组件所遇到致命性伤痕:
(1)、if中对byte的真假识别不一致。
同样是读取一个文件,通过if(byte[])来判断这个文件是否为空,原生的PHP中可以根据传入的byte[]是否为null来决定true/false,而Phalanger则无论如何一直返回false。
(2)、同样的PHP内置函数,执行的效果却不一致。
这里的所说的函数运行效果不一致并不是说效果完全的不一致,普通的使用还是没有问题的(正如我刚才读取mysql那样),而是有极少数的函数在特殊的条件下运行后得出的结果在原生PHP与Phalanger中是不一样的。
当然了,一百个读者有一百个海姆莱特,萝卜青菜各有所爱,Phalanger到底值不值得使用,应该怎么使用,还是全凭各位读者自己博弈了。