11.1 PHP中的面向对象(二)

为了操作一个对象,我们需要先获取这个对象的实例,而这肯定会涉及调用对象的构造方法。有关如何在扩展中调用PHP的函数与对象的方法这里不展开描述了。首先我们先了解下一个object在PHP内核中到底是如何实现的。

  1. typedef struct _zend_object_value {
  2. zend_object_handle handle;
  3. zend_object_handlers *handlers;
  4. } zend_object_value;
  5. //此外再回顾一下zval的值value的结构。
  6. typedef union _zvalue_value {
  7. long lval; /* long value */
  8. double dval; /* double value */
  9. struct {
  10. char *val;
  11. int len;
  12. } str;
  13. HashTable *ht; /* hash table value */
  14. zend_object_value obj;
  15. } zvalue_value;

如果我们有一个zval *tmp,那么tmp->value.obj来访问到最终保存对象实例的zend_object_value结构体,它包含两个成员:

  • zend_object_handle handle:最终实现是一个unsigned int值,Zend会把每个对象放进数组里,这个handle就是此实例的索引。所以我们在把对象当作参数传递时,只不过是传递的handle罢了,这样对性能有利,同时也是对象的引用机制的原理。
  • zend_object_handlers *handlers:这个里面是一组函数指针,我们可以通过它来对对象进行一些操作,比如:添加引用、获取属性等。此结构体在Zend/zend_object_handlers.h里定义。
下面我给出这个类的PHP语言实现,让我们在扩展中实现它,并生成它。

  1. <?php
  2. class baby
  3. {
  4. public function __construct()
  5. {
  6. echo "a new baby!\n";
  7. }
  8. public function hello()
  9. {
  10. echo "hello world!\n";
  11. }
  12. }
  13. function test_call()
  14. {
  15. $obj = new baby();
  16. $obj->hello();
  17. }

下面我们在扩展中实现以上test_call函数。

Note:关于下面walu_call_user_function()的说明:内核中并不存在该函数,可以在这里找到该函数的定义,直接拷贝过来用即可。

  1. zend_class_entry *baby_ce;
  2. ZEND_FUNCTION(test_call)
  3. {
  4. zval *obj;
  5. MAKE_STD_ZVAL(obj);
  6. object_init_ex(obj, baby_ce);
  7. //如果确认此类没有构造函数就不用调用了。
  8. walu_call_user_function(NULL, obj, "__construct", "");
  9. walu_call_user_function(NULL, obj, "hello", "");
  10. zval_ptr_dtor(&obj);
  11. return;
  12. }
  13. ZEND_METHOD(baby, __construct)
  14. {
  15. printf("a new baby!\n");
  16. }
  17. ZEND_METHOD(baby, hello)
  18. {
  19. printf("hello world!!!!!\n");
  20. }
  21. static zend_function_entry baby_method[]={
  22. ZEND_ME(baby, __construct, NULL, ZEND_ACC_PUBLIC|ZEND_ACC_CTOR)
  23. ZEND_ME(baby, hello, NULL, ZEND_ACC_PUBLIC)
  24. {NULL, NULL, NULL}
  25. };
  26. ZEND_MINIT_FUNCTION(test)
  27. {
  28. zend_class_entry ce;
  29. INIT_CLASS_ENTRY(ce, "baby", baby_method);
  30. baby_ce = zend_register_internal_class(&ce TSRMLS_CC);
  31. return SUCCESS;
  32. }

重新编译,执行命令查看是否生效~

  1. php -r "test_call();"