首页 > php > 利用php调用C语言 扩展PHP的功能

利用php调用C语言 扩展PHP的功能

第一步. 生成需要调用的so文件

  1. 首先做一个简单的so文件:
    /** * hello.c
  • To compile, use following commands:
    • gcc -O -c -fPIC -o hello.o hello.c
    • gcc -shared -o libhello.so hello.o
      */
      int hello_add(int a, int b)
      {
      return a + b;
      }
      然后将它编译成.so文件并放到系统中:
      $ gcc -O -c -fPIC -o hello.o hello.c // -fPIC:是指生成的动态库与位置无关
      $ gcc -shared -o libhello.so hello.o // -shared:是指明生成动态链接库
      $ su // 切换成超级用户,此时,需要输入密码。

cp libhello.so /usr/local/lib // 把生成的链接库放到指定的地址

echo /usr/local/lib > /etc/ld.so.conf.d/local.conf // 把库地址写入到配置文件中

/sbin/ldconfig // 用此命令,使刚才写的配置文件生效

  1. 写段小程序来验证其正确性:
    /**
  • hellotest.c
  • To compile, use following commands:
    • gcc -o hellotest -lhello hellotest.c
      */

include <stdio.h>

int main()
{
int a = 3, b = 4;
printf("%d + %d = %d", a, b, hello_add(a,b));
return 0;
}
编译并执行:
$ gcc -o hellotest -lhello hellotest.c // 编译测试文件,生成测试程序
$ ./hellotest // 运行测试程序

第二步. 制作PHP模块(外部模块)
请确保你已安装 PHP及APACHE服务器。
$ cd php-5.2.3/ext

  1. 然后通过下面的命令用ext_skel脚本建立一个名为 hello 的模块:
    $ ./ext_skel --extname=hello

  2. 执行该命令之后它会提示你应当用什么命令来编译模块,可惜那是将模块集成到php内部的编译方法。
    如果要编译成可动态加载的 php_hello.so,方法要更为简单。
    $ cd hello
    首先编辑 config.m4 文件,去掉第16行和第18行的注释(注释符号为 dnl 。)
    16: PHP_ARG_ENABLE(hello, whether to enable hello support,
    17: dnl Make sure that the comment is aligned:
    18: [ --enable-hello Enable hello support])

  3. 然后执行 phpize 程序,生成configure脚本:
    $ phpize
    该程序在ubuntu的php5-dev包中

  4. 打开 php_hello.h,在 PHP_FUNCTION(confirm_hello_compiled); 之下加入函数声明:
    PHP_FUNCTION(confirm_hello_compiled); /* For testing, remove later. */
    PHP_FUNCTION(hello_add);

  5. 打开 hello.c,在 PHP_FE(confirm_hello_compiled, NULL) 下方加入以下内容。
    zend_function_entry hello_functions[] = {
    PHP_FE(confirm_hello_compiled, NULL) /* For testing, remove later. /
    PHP_FE(hello_add, NULL) /
    For testing, remove later. /
    {NULL, NULL, NULL} /
    Must be the last line in hello_functions[] */};
    然后在 hello.c 的最末尾书写hello_add函数的内容:
    PHP_FUNCTION(hello_add)
    {
    long int a, b;
    long int result;
    if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "ll", &a, &b) == FAILURE) {
    return;
    }
    result = hello_add(a, b);
    RETURN_LONG(result);}保存退出,编译并安装:

  6. $ ./configure
    $ make LDFLAGS=-lhello
    $ sudo make install
    Installing shared extensions: /usr/lib/php5/20060613+lfs/$ su# cp modules/hello.so/usr/lib/php/modules
    luther@gliethttp:~$ sudo vim /etc/php5/apache2/php.ini;
    enable_dl = Off;允许dl()动态加载so扩展功能enable_dl = On
    luther@gliethttp:~$ sudo service apache2 reload 然后在 /var/www/html 下建立一个 hello.php 文件,内容如下:

然后在浏览器中打开hello.php文件,如果显示7,则说明函数调用成功了。
第三步. 制作PHP模块(内部模块)
另外可以在apache重启的时候让我们的so库直接动态编译进php5,就像linux的insmod hello.ko模块一样,不用dl加载也不用重新编译php,就可以直接使用so的函数了,步骤如下:
luther@gliethttp:~$ sudo vim /etc/php5/apache2/php.ini
enable_dl = Off
extension=hello.so
luther@gliethttp:~$ sudo service apache2 restart
不能reload而必须restart apache,这样so就像insmod hello.ko一样被融到了php5内核,然后代码就可以忽略掉dl("hello.so");了,
[注意,这种方式只适合hello.so库内所有功能代码已经全部调试ok,如果还处在调试期间,那么需要采用上面的dl强制加载的方式]
代码如下:

<?php
    echo hello_add(3, 4);
?>,
 但是该功能不太适合调试,因为每次修改hello.so中代码的话,都需要让service apacherestart重启才能让php5内核再次加载新的hello.so扩展.可以这样定义hello.so的实现,这样每次执行.php网页,都会在/var/www/下建立一个文件夹,所以php扩展实现了

一,搭建php环境
下载php 5.2.6 源码 并解压
编译安装,搭建php环境

二,创建扩展项目
进入源码目录
cd php5.2.6/ext/
./ext_skel --extname=my_ext
创建名字为my_ext的项目,最终会生成my_ext.so
三,更改配置和程序
$ vi ext/my_ext/config.m4
根据你自己的选择将
dnl PHP_ARG_WITH(my_ext, for my_ext support,
dnl Make sure that the comment is aligned:
dnl [ --with-my_ext Include my_ext support])
修改成
PHP_ARG_WITH(my_ext, for my_ext support,
Make sure that the comment is aligned:
[ --with-my_ext Include my_ext support])
或者将
dnl PHP_ARG_ENABLE(my_ext, whether to enable my_ext support,
dnl Make sure that the comment is aligned:
dnl [ --enable-my_ext Enable my_ext support])
修改成
PHP_ARG_ENABLE(my_ext, whether to enable my_ext support,
Make sure that the comment is aligned:
[ --enable-my_ext Enable my_ext support])
$ vi ext/my_ext/php_my_ext.h

PHP_FUNCTION(confirm_my_ext_compiled); /* For testing, remove later. */
更改为
PHP_FUNCTION(say_hello);

$ vi ext/my_ext/my_ext.c

zend_function_entry php5cpp_functions[] = {
PHP_FE(confirm_my_ext_compiled, NULL) /* For testing, remove later. /
{NULL, NULL, NULL} /
Must be the last line in php5cpp_functions[] /
};
更改为
zend_function_entry php5cpp_functions[] = {
PHP_FE(say_hello, NULL)
{NULL, NULL, NULL} /
Must be the last line in php5cpp_functions[] */
};
在最后添加:
PHP_FUNCTION(say_hello)
{
zend_printf("hello world\n");
}
四,编译
$ cd my_ext
$ /usr/local/php/bin/phpize
ps: 如果出现:Cannot find autoconf.……的错误信息,则需要安装 autoconf (安装过程略)
$ ./configure --with-php-config=/usr/local/php/bin/php-config
$ make
这时会编译出 my_ext/modules/my_ext.so
五,配置php.ini
将my_ext.so放入/usr/local/php/ext/目录
$ vi php.ini

修改添加如下:
extension_dir = '/usr/local/php/ext/'
extension=my_ext.so
六,测试
$ vi test.php

$ /usr/local/php/bin/php test.php
hello world.
则大功告成
下面我来讲讲如何作一个php的扩展
首先要有一个搭建好的php环境
我把php的安装在了/usr/local/php当然也通过
php的一个配置php.ini的路径但是要注意了
用这种方法安装的php扩展不能实现
我们在php安装以后的/usr/local/php/bin目录
找到这个文件phpize稍后我们将用到他
他就是个shell脚本你可以用vi phpize来查看他的内容
但是你要注意了这个脚本不是在哪里都可以应用的
[root@ns root]# phpize
Cannot find config.m4.
Make sure that you run '/usr/local/bin/phpize' in the top level source directory of the module
[root@ns root]# phpize
Cannot find config.m4.
Make sure that you run '/usr/local/bin/phpize' in the top level source directory of the module

你会看到这两种结果实际上你查看了这个脚本
很轻松的就会发现是怎么来处理的
你的模扩展的时候最好
放在/usr/local/src/php-4.3.5/ext下
来执行他你在这里也可以这样/usr/local/php/bin/phpize来执行也可以
phpize来执行
我们在/usr/local/src/php-4.3.5/ext下找到这个工具
来建立一个php扩展的一个框架
[root@ns ext]#cd /usr/local/src/php-4.3.5/ext/
[root@ns ext]# ./ext_skel --extname=jinzhesheng_module
Creating directory jinzhesheng_module
Creating basic files: config.m4 .cvsignore jinzhesheng_module.cphp_jinzhesheng_module.h CREDITS EXPERIMENTAL tests/001.phptjinzhesheng_module.php [done].
To use your new extension, you will have to execute the following steps:

  1. $ cd ..
  2. $ vi ext/jinzhesheng_module/config.m4
  3. $ ./buildconf
  4. $ ./configure --[with|enable]-jinzhesheng_module
  5. $ make
  6. $ ./php -f ext/jinzhesheng_module/jinzhesheng_module.php
  7. $ vi ext/jinzhesheng_module/jinzhesheng_module.c
  8. $ make
    执行了这个步骤以后你会看到这样的结果
    Repeat steps 3-6 until you are satisfied with ext/jinzhesheng_module/config.m4 and
    step 6 confirms that your module is compiled into PHP. Then, start writing
    code and repeat the last two steps as often as necessary.
    这样以后我们会在这个目录下生成一个目录叫jinzhesheng_module
    进入这里面我们看看
    [root@ns ext]# cd jinzhesheng_module/
    [root@ns jinzhesheng_module]# ls
    config.m4 EXPERIMENTAL jinzhesheng_module.php tests
    CREDITS jinzhesheng_module.c php_jinzhesheng_module.h
    然后我们要修改文件顺序是
    configue.m4
    jinzhesheng_module.c
    php_jinzhesheng_module.h
    使用文本编辑器打开config.m4文件,文件内容大致如下:
    dnl $Id$d
    dnl config.m4 for extension my_module
    dnl don't forget to callPHP_EXTENSION(my_module)
    dnl Comments in this file start with the string 'dnl'.
    dnl Remove where necessary. This file will not work
    dnl without editing.
    dnl If your extension references something external, use with:
    dnl PHP_ARG_WITH(my_module, for my_module support,
    dnl Make sure that the comment is aligned:
    dnl [ --with-my_module Include my_module support])
    dnl Otherwise use enable:
    dnl PHP_ARG_ENABLE(my_module, whether to enable my_module support,
    dnl Make sure that the comment is aligned:
    dnl [ --enable-my_module Enable my_module support])
    if test "$PHP_MY_MODULE" != "no"; then
    dnl If you will not be testing anything external, like existence of
    dnl headers, libraries or functions in them, just uncomment the
    dnl following line and you are ready to go.
    dnl Write more examples of tests here...
    PHP_EXTENSION(my_module, $ext_shared)
    Fi
    根据你自己的选择将
    dnl PHP_ARG_WITH(my_module, for my_module support,
    dnl Make sure that the comment is aligned:
    dnl [ --with-my_module Include my_module support])
    修改成
    PHP_ARG_WITH(my_module, for my_module support,
    Make sure that the comment is aligned:
    [ --with-my_module Include my_module support])
    或者将
    dnl PHP_ARG_ENABLE(my_module, whether to enable my_module support,
    dnl Make sure that the comment is aligned:
    dnl [ --enable-my_module Enable my_module support])
    修改成
    PHP_ARG_ENABLE(my_module, whether to enable my_module support,
    Make sure that the comment is aligned:
    [ --enable-my_module Enable my_module support])
    我这里用了后者
    然后保存退出
    然后在编辑
    Vi my_module.c
    将文件其中的下列代码进行修改
    /* Every user visible function must have an entry in my_module_functions[].
    /
    function_entry my_module_functions[] = {
    PHP_FE(say_hello, NULL) /
    ?添加着一行代码 /
    PHP_FE(confirm_my_module_compiled, NULL) /
    For testing, remove later. /
    {NULL, NULL, NULL} /
    Must be the last line in my_module_functions[] */
    };
    在文件的最后添加下列代码
    PHP_FUNCTION(say_hello)
    {
    zend_printf("hello world\n");
    }
    保存文件退出
    然后我们就可以在这个目录下使用上面的命令了
    /usr/local/php/bin/phpize
    执行以后会看到下面的
    [root@ns jinzhesheng_module]# /usr/local/php/bin/phpize
    Configuring for:
    PHP Api Version: 20020918
    Zend Module Api No: 20020429
    Zend Extension Api No: 20050606
    [root@ns jinzhesheng_module]#
    然后执行./configure --enable-jinzhesheng_module --with-apxs=/usr/local/apache/bin/apxs --with-php-config=/usr/local/php/bin/php-config
    我们在安装以后的php的bin目录下的可以找到这个文件的
    php-config 和phpize
    这一步骤一定要注意你的apache的apxs放在哪里了
    然后执行make
    你会看到出现错误了你重新定义了函数我们前面的
    这个你在回头改一下这个文件把原来的函数删除掉在生成的文件里面会有同样的函数
    你在加入你的代码
    就可以通过了
    这个时候会在当前的目录下生成一个目录叫modules他的下面就放着你要的
    jinzhesheng_module.so文件
    cp modules/jinzhesheng_module.so /usr/local/php/ext/
    这里需要你先设置你的php的扩展目录的在
    在php.ini里面
    通过extension_dir
    最后一不是你在php.ini文件中打开这个扩展
    extension=jinzhesheng_module.so
    然后
    重新起动apache
    用phpinfo来察看一下ok了

上一篇: HTTP报文实例

下一篇: 用c语言写cgi程序实现文件上传