快速开发一个PHP扩展图文教程
时间:2011-02-11 来源:xiaokaizi
    作者:heiyeluren
    时间:2008-12-5
    博客:http://blog.csdn.net/heiyeshuwu
  
本文通过非常快速的方式讲解了如何制作一个PHP 5.2 环境的扩展(PHP Extension),希望能够在图文的方式下让想快速学习的朋友了解一下制作过程。
    需求:比如开发一个叫做 heiyeluren 的扩展,扩展里就一个函数 heiyeluren_test(),输入一个字符串,函数返回:Your input string: xxxxx。
    要求:了解C/C++编程,熟悉PHP编程
    环境:下载一份php对应版本的源码,我这里是 php-5.2.6,先正常安装php,假设我们的php安装在 /usr/local/php 目录,源码在 /root/soft/php/php-5.2.6/,现在开始!
    步骤一:生成扩展框架
  
    
    cd /root/soft/php/php-5.2.6/ext
    ./ext_skel --extname=heiyeluren
    cd /root/soft/php/php-5.2.6/ext/heiyeluren
    vi config.m4
    打开文件后去掉 dnl ,获得下面的信息:
    PHP_ARG_ENABLE(heiyeluren, whether to enable heiyeluren support,
    [ --enable-heiyeluren Enable heiyeluren support])
    保存退出.
    (图01)
    
  
    第二步:编写代码
    
    vi php_heiyeluren.h
    找到:PHP_FUNCTION(confirm_heiyeluren_compiled); ,新增一行:
    PHP_FUNCTION(heiyeluren_test);
    保存退出。
    (图02)
    
  
    
    vi heiyeluren.c
    数组里增加我们的函数,找到 zend_function_entry heiyeluren_functions[],增加:
    PHP_FE(heiyeluren, NULL)
    (图03)
  
    
  
    再到 heiyeluren.c 文件最后面增加如下代码:
    PHP_FUNCTION(heiyeluren_test)
    {
    char *arg = NULL;
    int arg_len, len;
    char *strg;
  
    if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "s", &arg, &arg_len) == FAILURE) {
    return;
    }
  
    len = spprintf(&strg, 0, "Your input string: %s\n", arg);
    RETURN_STRINGL(strg, len, 0);
    }
    保存退出。
    (图04)
  
    
  
    第三步:编译安装
    
    cd /root/soft/php/php-5.2.6/ext/heiyeluren
    /usr/local/php/bin/phpize
    ./configure --with-php-config=/usr/local/php/bin/php-config
    make
    make test
    make install
  
    现在看看是不是有个 /usr/local/php/lib/php/extensions/no-debug-non-zts-20060613/heiyeluren.so
    编辑php.ini,把扩展加入进去:
    vi /usr/local/php/lib/php.ini
    在[PHP]模块下增加:
    extension = heiyeluren.so
    保存退出。
    (图05)
  
    注意:如果你不存在扩展文件目录,或者安装报错,那么可以自行建立这个目录,然后把扩展拷贝到目录下,然后记得把 php.ini 文件中的 extension_dir 修改为该目录:
    extension_dir = "/usr/local/php/lib/php/extensions/no-debug-non-zts-20060613/"
    (图06)
  
    第四步:检查安装结果
    现在看看模块加载了没有:
    /usr/local/php/bin/php -m,应该会打印出:
    [PHP Modules]
    ...
    heiyeluren
    ...
    [Zend Modules]
  
    
    然后重启apache,输出 phpinfo() ,应该能够看到:
    heiyeluren
    heiyeluren support enabled
    (图07)
  
    
  
    看看函数是否存在并且调用,在web目录下建立:heiyeluren.php
    <?php
    echo "<pre>";
    print_r(get_loaded_extensions());
    print_r(get_extension_funcs('heiyeluren'));
    echo heiyeluren_test('My first php extension');
    echo "</pre>";
    ?>
    访问apache,应该能够看到:
    Array
    (
    ...
    [33] => heiyeluren
    )
    Array
    (
    [0] => confirm_heiyeluren_compiled
    [1] => heiyeluren_test
    )
    Your input string: heiyeluren
    (图08)
  
    
  
    
    扩展制作成功!
  
补充:作者:blankyao
关于如何着手开发一个PHP扩展,大家可以去看下黑夜路人同学写的快速开发一个PHP扩展一文,这篇文章深入浅出的介绍了开发一个PHP扩展的基本流程,今天我就来总结下PHP扩展开发里面几个基本的宏和函数。
    一、PHP_FUNCTION
    用户空间和扩展之间的连接通过PHP_FUNCTION来完成,
  
    PHP_FUNCTION(sample_hello_world)
    {
    php_printf(”Hello World!\n”);
    }
  
    PHP_FUNCTION()宏和普通的c函数一样,它的定义为
    #define PHP_FUNCTION(name)   void zif_##name(INTERNAL_FUNCTION_PARAMETERS)
    在这里,可以写为下面的代码
  
void zif_sample_hello_world(zval *return_value, char return_value_used, zval *this_ptr TSRMLS_C)
    二、function_entry
    简单的声明函数是不够的,zend引擎需要知道函数的地址以及如何将函数名字导入用户空间
  
    static function_entry php_sample_functions[] = {
    PHP_FE(sample_hello_world,        NULL)
    { NULL, NULL, NULL }
    };
  
    php_sample_functions向量要随着你向扩展中添加函数而增长,每一个函数都会是这个向量中的一部分。
    考虑PHP_FE()宏,PHP_FE(sample_hello_world,        NULL)展开后会是{ “sample_hello_world”, zif_sample_hello_world, NULL},
    这样既提供了函数的名字,同时提供了指向它的实现函数的指针,地三个参数暗示一些信息比如某个特定的参数要以引用的方式传递。
  
    三、zend_module_entry
    现在已经有了对外的函数,但是还没有和zend引擎联系起来。你只要将function_entry类型的结构加入zend_module_entry这个类型的结构就可以了。
  
只 会这些是不够的,你还需要了解一些PHP内核的机制,以及Zend Engine的机制。另外很重要的一点是,你需要有扎实的C语言基础,网上关于PHP扩展以及内核机制的内容不太多,大多数都是英文的,推荐大家阅读下 Sams.Extending.and.Embedding.PHP.一书,也是英文的。
今天就先介绍这点,不足之处,欢迎大家指教!










