开班看PHP内核也有一段时间了,现在开班边学边总计,前日就总括一下什么样创建本身的PHP扩张。

自己的条件如下:

系统:Ubuntu 14.04

php版本:5.5.19

参照摘录:用C/C++扩展你的PHP

PHP取得成功的1个最首要缘由之一是它拥有多量的可用扩充。web开发者无论有啥种须求,那种须要最有可能在PHP发行李包裹里找到。PHP发行李包裹包涵接济各类数据库,图形文件格式,压缩,XML技术扩充在内的过多扩张。

壮大API的引入使PHP3取得了巨大的展开,增加API机制使PHP开发社区很简单的付出出几十种扩张。未来,多个本子过去了,API照旧和PHP3时的可怜相似。扩大首要的合计是:尽大概的从扩充编写者这里隐藏PHP的个中机制黄岩乱弹本引擎本身,仅仅必要开发者熟习API。

有五个理由供给自个儿编排PHP扩张。首个理由是:PHP需求匡助一项她还未帮忙的技能。那平常包蕴包裹一些现成的C函数库,以便提供PHP接口。例如,要是2个叫FooBase的数据库已生产市镇,你要求建立3个PHP扩张扶助您从PHP里调用FooBase的C函数库。那几个工作大概仅由一人完结,然后被整个PHP社区共享(假使您愿意的话)。第③个不是很普遍的理由是:你供给从品质或效益的来头考虑来编排一些商业逻辑。

如果你正在开发一个网站,须求一个把字符串重复n次的函数。上面是用PHP写的事例:

function util_str_repeat($string, $n){
    $result = "";
    for($i = 0; $i < $n; $i++){
        $result .= $string;
    }
    return $result;
}

util_str_repeat("One", 3);// returns "OneOneOne".
util_str_repeat("One", 1);// returns "One".

万一由于一些竟然的原故,你需求日常调用这一个函数,而且还要传给函数相当短的字符串和大值n。那代表在剧本里有一定巨大的字符串连接量和内部存款和储蓄器重新分配进度,以至分明地下跌脚本执行进程。借使有三个函数能够更快地分配大批量且丰裕的内部存储器来存放结果字符串,然后把$string重复n次,就不须求在历次循环迭代中分配内部存款和储蓄器。

为扩张建立函数的第三步是写几个函数定义文件,该函数定义文件定义了扩展对外提供的函数原形。该例中,定义函数只有一行函数原形util_str_repeat()
:

string util_str_repeat(string str, int n)

函数定义文件的相似格式是2个函数一行。你能够定义可选参数和动用多量的PHP类型,包罗:
bool, float, int, array等。

保留为util.def文件至PHP原代码目录树下(即与ext_skel文件放在同等目录下,作者的目录是/usr/share/php5/)。

下一场正是经过增加骨架(skeleton)构造器运行函数定义文件的火候了。该构造器脚本就是ext_skel。假诺你把函数定义保存在一个叫作util.def的公文里,而且你愿意把扩充取名为util,运营下边包车型地铁授命来确立扩张骨架:

sudo ./ext_skel --extname=util --proto=util.def

举办之后,作者那里报了如下错误:

./ext_skel: 1: cd: can't cd to /usr/lib/php5/skeleton
Creating directory util
awk: cannot open /create_stubs (No such file or directory)
Creating basic files: config.m4 config.w32 .svnignore util.c./ext_skel: 216: ./ext_skel: cannot open /skeleton.c: No such file
 php_util.h./ext_skel: 234: ./ext_skel: cannot open /php_skeleton.h: No such file
 CREDITS./ext_skel: 238: ./ext_skel: cannot open /CREDITS: No such file
 EXPERIMENTAL./ext_skel: 242: ./ext_skel: cannot open /EXPERIMENTAL: No such file
 tests/001.phpt./ext_skel: 247: ./ext_skel: cannot open /tests/001.phpt: No such file
 util.php./ext_skel: 251: ./ext_skel: cannot open /skeleton.php: No such file
rm: cannot remove ‘function_entries’: No such file or directory
rm: cannot remove ‘function_declarations’: No such file or directory
rm: cannot remove ‘function_stubs’: No such file or directory
 [done].

To use your new extension, you will have to execute the following steps:

1.  $ cd ..
2.  $ vi ext/util/config.m4
3.  $ ./buildconf
4.  $ ./configure --[with|enable]-util
5.  $ make
6.  $ ./php -f ext/util/util.php
7.  $ vi ext/util/util.c
8.  $ make

Repeat steps 3-6 until you are satisfied with ext/util/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.

很显著是/usr/lib/php5/skeleton路径的谬误,编辑ext_skel文件,将/usr/lib/php5/skeleton修改为/usr/share/php5/skeleton,然后移除掉生成的util文件夹,再度实施此前的指令,成功后提示如下:

Creating directory util
Creating basic files: config.m4 config.w32 .svnignore util.c php_util.h CREDITS EXPERIMENTAL tests/001.phpt util.php [done].

To use your new extension, you will have to execute the following steps:

1.  $ cd ..
2.  $ vi ext/util/config.m4
3.  $ ./buildconf
4.  $ ./configure --[with|enable]-util
5.  $ make
6.  $ ./php -f ext/util/util.php
7.  $ vi ext/util/util.c
8.  $ make

Repeat steps 3-6 until you are satisfied with ext/util/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.

接下来利用静态编写翻译的措施编译扩展。为了使扩充能够被编写翻译,须求修改扩充目录util/下的config.m4文件。扩充没有包装任何外部的C库,你供给加上支持–enable-util配置开关到PHP编译系统里(–with-extension
开关用于那么些需求用户钦定相关C库路径的恢宏)。找到如下内容:

dnl PHP_ARG_ENABLE(util, whether to enable util support,
dnl Make sure that the comment is aligned:
dnl [  --enable-util           Enable util support])

将近来的dnl 去掉,修改为如下结果:

PHP_ARG_ENABLE(util, whether to enable util support,
Make sure that the comment is aligned:
[  --enable-util           Enable util support])

然后修改util.c文件,找到如下代码:

PHP_FUNCTION(util_str_repeat)
{
    char *str = NULL;
    int argc = ZEND_NUM_ARGS();
    int str_len;
    long n;

    if (zend_parse_parameters(argc TSRMLS_CC, "sl", &str, &str_len, &n) == FAILURE) 
        return;

    php_error(E_WARNING, "util_str_repeat: not yet implemented");
}

将其修改为如下代码:

PHP_FUNCTION(util_str_repeat)
{
    char *str = NULL;
    int argc = ZEND_NUM_ARGS();
    int str_len;
    long n;
    char *result; /* Points to resulting string */
    char *ptr; /* Points at the next location we want to copy to */
    int result_length; /* Length of resulting string */

    if (zend_parse_parameters(argc TSRMLS_CC, "sl", &str, &str_len, &n) == FAILURE)
        return;

    /* Calculate length of result */
    result_length = (str_len * n);
    /* Allocate memory for result */
    result = (char *) emalloc(result_length + 1);
    /* Point at the beginning of the result */
    ptr = result;

    while (n--) {
        /* Copy str to the result */
        memcpy(ptr, str, str_len);
        /* Increment ptr to point at the next position we want to write to */
        ptr += str_len;
    }

    /* Null terminate the result. Always null-terminate your strings
    even if they are binary strings */
    *ptr = '\0';
    /* Return result to the scripting engine without duplicating it*/
    RETURN_STRINGL(result, result_length, 0);
}

里面包车型大巴具体内容,就不在那里说了,之后会日渐写到。

接下来正是编写翻译,安装。在util目录下,命令如下(命令或然都须求加sudo):

phpize
./configure
make
make test
make install

下一场配置生成的恢宏文件,在php5.5版本中,进入到/etc/php5/mods-available目录下,创立util.ini文件,写入如下内容:

extension=util.so

然后enable util扩展

sudo php5enmod util

最后,重启php-fpm

sudo service php5-fpm restart

创立多少个php文件,测试一下,测试文件如下:

<?php
for ($i = 1; $i <= 3; $i++) {
    print util_str_repeat("CraryPrimitiveMan ", $i);
    print "\n";
}
?>

实践结果如下:

CraryPrimitiveMan 
CraryPrimitiveMan CraryPrimitiveMan 
CraryPrimitiveMan CraryPrimitiveMan CraryPrimitiveMan

如此大家就水到渠成创办了二个包含简单的PHP函数的扩充。

盗图一张~~

XML 1

明天就先到此处~~

 

相关文章

网站地图xml地图