Discuz教程网

PHP的C扩展

[复制链接]
authicon dly 发表于 2012-4-14 09:50:52 | 显示全部楼层 |阅读模式
在php中最核心的一个数据结构就是这个:

  1. typedef union _zvalue_value { 
  2.  long lval;   /* long value */ 
  3.  double dval;  /* double value */ 
  4.  struct { 
  5.  char *val; 
  6.  int len; 
  7.  } str; 
  8.  HashTable *ht;  /* hash table value */ 
  9.  zend_object_value obj; 
  10. } zvalue_value; 
  11.  
  12.  
  13. struct _zval_struct { 
  14.  /* Variable information */ 
  15.  zvalue_value value; /* value */ 
  16.  zend_uint refcount; 
  17.  zend_uchar type; /* active type */ 
  18.  zend_uchar is_ref; 
  19. };
复制代码

这个数据结构也就是php中的每个数据类型在下层c语言中的表示。可以看到_zval_struct 结构体第一个字段是一个联合体,他来存储实际的值,这些值可以为long,double,字符串,hashtable(也就是php中的数组)和对象。而第2个参数是个引用计数,第三个参数是当前的类型。   也就是说php中的每个类型实际都是一个 struct _zval_struct类型。
  1 首先进入php的源码目录下的ext文件夹,然后运行下面的命令,这样将会生成一个my_module的文件夹。:
  1. ./ext_skel --extname=my_module
复制代码

2 然后在my_module.h里面声明你的php函数名:
  1. PHP_FUNCTION(my_function);
复制代码

2 在my_module.c文件的my_module_functions(这里的module就是你所创建的扩展模块名字)加入你所要写的php方法名:
  1. PHP_FE(my_function, NULL)
复制代码

3 接下来就实现PHP_FUNCTION(my_function)。   首先 参数的解析,当传递进来的php的类型和c的类型之间的转换:
  这里要用到的函数是:
  1. int zend_parse_parameters(int num_args TSRMLS_DC, char *type_spec, ...);
复制代码

其中的php类型和c类型的对应如下:   引用
  l - long
  d - double
  s - string (with possible null bytes) and its length
  b - boolean
  r - resource, stored in zval*
  a - array, stored in zval*
  o - object (of any class), stored in zval*
  O - object (of class specified by class entry), stored in zval*
  z - the actual zval*
  这边就看到了,这里的zval也就是一开始介绍的那个php的类型结构体。
  这边还可以使用三个符号:
  | - 这个也就是或者的意思
  / -将会提供当前参数的一个副本。
  ! - 这个符号他必须进跟在a, o, O, r, z 的后面 ,也就是说当传递进来的参数为NULL的时候,我们pass的那个指针也会被NULL。
  例子:
  1. /* 得到一个long,string和一个zval */ 
  2. long l; 
  3. char *s; 
  4. int s_len; 
  5. zval *param; 
  6. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, 
  7.              "lsz", &l, &s, &s_len, ¶m) == FAILURE) { 
  8.   return; 
  9. } 
  10.  
  11. /*传递进来的为一个对象或者一个double */ 
  12. zval *obj; 
  13. double d = 0.5; 
  14. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, 
  15.              "O|d", &obj, my_ce, &d) == FAILURE) { 
  16.   return; 
  17. } 
  18.  
  19. /* 传递进来的参数为NULL或者一个对象和一个数组 */ 
  20. zval *obj; 
  21. zval *arr; 
  22. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "O!a", &obj, &arr) == FAILURE) { 
  23.   return; 
  24. } 
  25.  
  26. /* 得到一个数组 */ 
  27. zval *arr; 
  28. if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "a/", &arr) == FAILURE) { 
  29.   return; 
  30. } 
  31.  
  32. /*得到前三个参数 */ 
  33. zval *z; 
  34. zend_bool b; 
  35. zval *r; 
  36. if (zend_parse_parameters(3, "zbr!", &z, &b, &r) == FAILURE) { 
  37.   return; 
  38. }
复制代码

当我们传递进来的为一个数组的时候我们该如何遍历这个数组呢,看下面的代码:
  1. zval *param; 
  2.     HashPosition pos; 
  3.     zval **data_value; 
  4.     if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "a", &(param)) == FAILURE) { 
  5.    RETURN_FALSE; 
  6.    } 
  7. //遍历数组. 
  8.     for (zend_hash_internal_pointer_reset_ex(Z_ARRVAL_P(param), &pos); 
  9.   zend_hash_get_current_data_ex(Z_ARRVAL_P(param), (void **)&data_value, &pos) == SUCCESS; 
  10.   zend_hash_move_forward_ex(Z_ARRVAL_P(param), &pos)) 
  11.     { 
  12.       //现在数组里面的值就存储在data_value这个指针里面,我们可以通过对他解引用来提取值,比如假设有个元素是long的,那么我们就可以这样来取(就是根据一开头介绍的那个数据结构来取): 
  13.  
  14. long temp=(*data_value)->value.lval; 
  15.     }
复制代码

接下来是返回值的问题,这里定义了好几个宏:
  1. RETURN_RESOURCE(int r) 
  2. RETURN_BOOL(int b)  
  3. RETURN_NULL( ) 
  4. RETURN_LONG(int l)  
  5. RETURN_DOUBLE(double d)  
  6. RETURN_STRING(char *s, int dup)  
  7. RETURN_STRINGL(char *s, int l, int dup)  
  8. RETURN_EMPTY_STRING( )  
  9. RETURN_FALSE  
  10. RETURN_TRUE 
复制代码

比如我要返回个字符串,可以这么写:
  1.  RETURN_STRING("banana", 1);
复制代码

这里讲下返回一个数组的问题,下面的代码是返回一个嵌套数组:
  1. zval *param: 
  2.  
  3. array_init(param); 
  4. //return_value是一个全局的变量(我是这么理解的) 
  5. array_init(return_value); 
  6. add_index_string(param, 0, "sad",1); 
  7. add_index_zval(return_value,0, param);
复制代码

PS:更详细的还是要看ext目录下面的那些扩展实例



上一篇:50点提高PHP编程效率 引入缓存提升性能
下一篇:置顶的开发资料
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

1314学习网 ( 浙ICP备10214163号 )

GMT+8, 2025-5-1 23:37

Powered by Discuz! X3.4

© 2001-2013 Comsenz Inc.

快速回复 返回顶部 返回列表