http://blog.csdn.net/super_ufo/article/details/3863731 php
php调用C代码的方法详解 html
在php程序中须要用到C代码,应该是下面两种状况: node
Boolean | b | zend_bool |
Long | l | long |
Double | d | double |
String | s | char*, int |
Resource | r | zval* |
Array | a | zval* |
Object | o | zval* |
zval | z | zval* |
Traditional | Non-Persistent | Persistent |
---|---|---|
malloc(count) calloc(count, num) |
emalloc(count) ecalloc(count, num) |
pemalloc(count, 1)* pecalloc(count, num, 1) |
strdup(str) strndup(str, len) |
estrdup(str) estrndup(str, len) |
pestrdup(str, 1) pemalloc() & memcpy() |
free(ptr) | efree(ptr) | pefree(ptr, 1) |
realloc(ptr, newsize) | erealloc(ptr, newsize) | perealloc(ptr, newsize, 1) |
malloc(count * num + extr)** | safe_emalloc(count, num, extr) | safe_pemalloc(count, num, extr) |
更好的文章:http://www.toplee.com/blog/56.html#pp1
程序员
本节没有介绍关于脚本引擎基本构造的一些知识,而是直接进入扩展的编码讲解中,所以不要担忧你没法马上得到对扩展总体把握的感受。假设你正在开发一个网站,须要一个把字符串重复n次的函数。下面是用PHP写的例子: 算法
function self_concat($string, $n) 数据库
{ vim
$result = ""; windows
for ($i = 0; $i < $n; $i++) { 数组
$result .= $string; 安全
}
return $result;
}
self_concat("One", 3) returns "OneOneOne".
self_concat("One", 1) returns "One".
假设因为一些奇怪的缘由,你须要时常调用这个函数,并且还要传给函数很长的字符串和大值n。这意味着在脚本里有至关巨大的字符串链接量和内存从新分配过程,以致显著地下降脚本执行速度。若是有一个函数可以更快地分配大量且足够的内存来存放结果字符串,而后把$string重复n次,就不须要在每次循环迭代中分配内存。
为扩展创建函数的第一步是写一个函数定义文件,该函数定义文件定义了扩展对外提供的函数原形。该例中,定义函数只有一行函数原形self_concat() :
string self_concat(string str, int n)
函数定义文件的通常格式是一个函数一行。你能够定义可选参数和使用大量的PHP类型,包括: bool, float, int, array等。
保存为myfunctions.def文件至PHP原代码目录树下。
该是经过扩展骨架(skeleton)构造器运行函数定义文件的时机了。该构造器脚本叫ext_skel,放在PHP原代码目录树的ext/目录下(PHP原码主目录下的README.EXT_SKEL提供了更多的信息)。假设你把函数定义保存在一个叫作myfunctions.def的文件里,并且你但愿把扩展取名为myfunctions,运行下面的命令来创建扩展骨架
./ext_skel --extname=myfunctions --proto=myfunctions.def
这个命令在ext/目录下创建了一个myfunctions/目录。你要作的第一件事情也许就是编译该骨架,以便编写和测试实际的C代码。编译扩展有两种方法:
☞ 做为一个可装载模块或者DSO(动态共享对象)
☞ 静态编译到PHP
由于第二种方法比较容易上手,因此本章采用静态编译。若是你对编译可装载扩展模块感兴趣,能够阅读PHP原代码根目录下的README.SELF-CONTAINED_EXTENSIONS文件。为了使扩展可以被编译,须要修改扩展目录ext/myfunctions/下的config.m4文件。扩展没有包裹任何外部的C库,你须要添加支持--enable-myfunctions配置开关到PHP编译系统里(–with-extension 开关用于那些须要用户指定相关C库路径的扩展)。能够去掉自动生成的下面两行的注释来开启这个配置。
PHP_ARG_ENABLE(myfunctions, whether to enable myfunctions support,
[ --enable-myfunctions Include myfunctions support])
如今剩下的事情就是在PHP原代码树根目录下运行./buildconf,该命令会生成一个新的配置脚本。经过查看./configure --help输出信息,能够检查新的配置选项是否被包含到配置文件中。如今,打开你喜爱的配置选项开关和--enable-myfunctions从新配置一下PHP。最后的但不是最次要的是,用make来从新编译PHP。
ext_skel应该把两个PHP函数添加到你的扩展骨架了:打算实现的self_concat()函数和用于检测myfunctions 是否编译到PHP的confirm_myfunctions_compiled()函数。完成PHP的扩展开发后,能够把后者去掉。
<?php
print confirm_myfunctions_compiled("myextension");
?>
运行这个脚本会出现相似下面的输出:
"Congratulations! You have successfully modified ext/myfunctions
config.m4. Module myfunctions is now compiled into PHP."
另外,ext_skel脚本生成一个叫myfunctions.php的脚本,你也能够利用它来验证扩展是否被成功地编译到PHP。它会列出该扩展所支持的全部函数。
如今你学会如何编译扩展了,该是真正地研究self_concat()函数的时候了。
下面就是ext_skel脚本生成的骨架结构:
/* {{{ proto string self_concat(string str, int n)
*/
PHP_FUNCTION(self_concat)
}
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, "self_concat: not yet implemented");
}
/* }}} */
自动生成的PHP函数周围包含了一些注释,这些注释用于自动生成代码文档和vi、Emacs等编辑器的代码折叠。函数自身的定义使用了宏PHP_FUNCTION(),该宏能够生成一个适合于Zend引擎的函数原型。逻辑自己分红语义各部分,取得调用函数的参数和逻辑自己。
为了得到函数传递的参数,可使用zend_parse_parameters()API函数。下面是该函数的原型:
zend_parse_parameters(int num_args TSRMLS_DC, char *type_spec, …);
第一个参数是传递给函数的参数个数。一般的作法是传给它ZEND_NUM_ARGS()。(ZEND_NUM_ARGS() 来表示对传入的参数“有多少要多少”)这是一个表示传递给函数参数总个数的宏。第二个参数是为了线程安全,老是传递TSRMLS_CC宏,后面会讲到。第三个参数是一个字符串,指定了函数指望的参数类型,后面紧跟着须要随参数值更新的变量列表。由于PHP采用松散的变量定义和动态的类型判断,这样作就使得把不一样类型的参数转化为指望的类型成为可能。例如,若是用户传递一个整数变量,可函数须要一个浮点数,那么zend_parse_parameters()就会自动地把整数转换为相应的浮点数。若是实际值没法转换成指望类型(好比整形到数组形),会触发一个警告。
下表列出了可能指定的类型。咱们从完整性考虑也列出了一些没有讨论到的类型。
类型指定符 |
对应的C类型 |
描述 |
l |
long |
符号整数 |
d |
double |
浮点数 |
s |
char *, int |
二进制字符串,长度 |
b |
zend_bool |
逻辑型(1或0) |
r |
zval * |
资源(文件指针,数据库链接等) |
a |
zval * |
联合数组 |
o |
zval * |
任何类型的对象 |
O |
zval * |
指定类型的对象。须要提供目标对象的类类型 |
z |
zval * |
无任何操做的zval |
为了容易地理解最后几个选项的含义,你须要知道zval是Zend引擎的值容器[1]。不管这个变量是布尔型,字符串型或者其余任何类型,其信息总会包含在一个zval联合体中。本章中咱们不直接存取zval,而是经过一些附加的宏来操做。下面的是或多或少在C中的zval, 以便咱们能更好地理解接下来的代码。
typedef union _zval {
long lval;
double dval;
struct {
char *val;
int len;
} str;
HashTable *ht;
zend_object_value obj;
} zval;
在咱们的例子中,咱们用基本类型调用zend_parse_parameters(),以本地C类型的方式取得函数参数的值,而不是用zval容器。
为了让zend_parse_parameters()可以改变传递给它的参数的值,并返回这个改变值,须要传递一个引用。仔细查看一下self_concat():
if (zend_parse_parameters(argc TSRMLS_CC, "sl", &str, &str_len, &n) == FAILURE)
return;
注意到自动生成的代码会检测函数的返回值FAILUER(成功即SUCCESS)来判断是否成功。若是没有成功则当即返回,而且由zend_parse_parameters()负责触发警告信息。由于函数打算接收一个字符串l和一个整数n,因此指定 ”sl” 做为其类型指示符。s须要两个参数,因此咱们传递参考char * 和 int (str 和 str_len)给zend_parse_parameters()函数。不管何时,记得老是在代码中使用字符串长度str_len来确保函数工做在二进制安全的环境中。不要使用strlen()和strcpy(),除非你不介意函数在二进制字符串下不能工做。二进制字符串是包含有nulls的字符串。二进制格式包括图象文件,压缩文件,可执行文件和更多的其余文件。”l” 只须要一个参数,因此咱们传递给它n的引用。尽管为了清晰起见,骨架脚本生成的C变量名与在函数原型定义文件中的参数名同样;这样作不是必须的,尽管在实践中鼓励这样作。
回到转换规则中来。下面三个对self_concat()函数的调用使str, str_len和n获得一样的值:
self_concat("321", 5);
self_concat(321, "5");
self_concat("321", "5");
str points to the string "321", str_len equals 3, and n equals 5.
str 指向字符串"321",str_len等于3,n等于5。
在咱们编写代码来实现链接字符串返回给PHP的函数前,还得谈谈两个重要的话题:内存管理、从PHP内部返回函数值所使用的API!!
用于从堆中分配内存的PHP API几乎和标准C API同样。在编写扩展的时候,使用下面与C对应(所以没必要再解释)的API函数:
emalloc(size_t size);
efree(void *ptr);
ecalloc(size_t nmemb, size_t size);
erealloc(void *ptr, size_t size);
estrdup(const char *s);
estrndup(const char *s, unsigned int length);
在这一点上,任何一位有经验的C程序员应该象这样思考一下:“什么?标准C没有strndup()?”是的,这是正确的,由于GNU扩展一般在Linux下可用。estrndup()只是PHP下的一个特殊函数。它的行为与estrdup()类似,可是能够指定字符串重复的次数(不须要结束空字符),同时是二进制安全的。这是推荐使用estrndup()而不是estrdup()的缘由。
在几乎全部的状况下,你应该使用这些内存分配函数。有一些状况,即扩展须要分配在请求中永久存在的内存,从而不得不使用malloc(),可是除非你知道你在作什么,你应该始终使用以上的函数。若是没有使用这些内存函数,而相反使用标准C函数分配的内存返回给脚本引擎,那么PHP会崩溃。
这些函数的优势是:任何分配的内存在偶然状况下若是没有被释放,则会在页面请求的最后被释放。所以,真正的内存泄漏不会产生。然而,不要依赖这一机制,从调试和性能两个缘由来考虑,应当确保释放应该释放的内存。剩下的优势是在多线程环境下性能的提升,调试模式下检测内存错误等。
还有一个重要的缘由,你不须要检查这些内存分配函数的返回值是否为null。当内存分配失败,它们会发出E_ERROR错误,从而决不会返回到扩展。
扩展API包含丰富的用于从函数中返回值的宏。这些宏有两种主要风格:第一种是RETVAL_type()形式,它设置了返回值但C代码继续执行。这一般使用在把控制交给脚本引擎前还但愿作的一些清理工做的时候使用,而后再使用C的返回声明 ”return” 返回到PHP;后一个宏更加广泛,其形式是RETURN_type(),他设置了返回类型,同时返回控制到PHP。下表解释了大多数存在的宏。
设置返回值而且结束函数 |
设置返回值 |
宏返回类型和参数 |
RETURN_LONG(l) |
RETVAL_LONG(l) |
整数 |
RETURN_BOOL(b) |
RETVAL_BOOL(b) |
布尔数(1或0) |
RETURN_NULL() |
RETVAL_NULL() |
NULL |
RETURN_DOUBLE(d) |
RETVAL_DOUBLE(d) |
浮点数 |
RETURN_STRING(s, dup) |
RETVAL_STRING(s, dup) |
字符串。若是dup为1,引擎会调用estrdup()重复s,使用拷贝。若是dup为0,就使用s |
RETURN_STRINGL(s, l, dup) |
RETVAL_STRINGL(s, l, dup) |
长度为l的字符串值。与上一个宏同样,但由于s的长度被指定,因此速度更快。 |
RETURN_TRUE |
RETVAL_TRUE |
返回布尔值true。注意到这个宏没有括号。 |
RETURN_FALSE |
RETVAL_FALSE |
返回布尔值false。注意到这个宏没有括号。 |
RETURN_RESOURCE(r) |
RETVAL_RESOURCE(r) |
资源句柄。 |