PHP内核探索:创建对象实例
PHP内核    2019-04-25 16:29:25    10    0    0
admin   PHP内核

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

01typedef struct _zend_object_value {
02    zend_object_handle handle;
03    zend_object_handlers *handlers;
04} zend_object_value;
05 
06//此外再回顾一下zval的值value的结构。
07typedef union _zvalue_value {
08    long lval;                  /* long value */
09    double dval;                /* double value */
10    struct {
11        char *val;
12        int len;
13    } str;
14    HashTable *ht;              /* hash table value */
15    zend_object_value obj;
16} 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语言实现,让我们在扩展中实现它,并生成它。

01<?php
02class baby
03{
04    public function __construct()
05    {
06        echo "a new baby!\n";
07    }  
08     
09    public function hello()
10    {
11        echo "hello world!\n";
12    }
13}
14 
15function test_call()
16{
17    $obj new baby();
18    $obj->hello();
19}
20?>

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

01zend_class_entry *baby_ce;
02 
03ZEND_FUNCTION(test_call)
04{
05    zval *obj;
06    MAKE_STD_ZVAL(obj);
07    object_init_ex(obj, baby_ce);
08     
09    //如果确认此类没有构造函数就不用调用了。
10    walu_call_user_function(NULL, obj, "__construct""");
11     
12    walu_call_user_function(NULL, obj, "hello""");
13    zval_ptr_dtor(&obj);
14    return;
15}
16 
17ZEND_METHOD(baby, __construct)
18{
19    printf("a new baby!\n");
20}
21 
22ZEND_METHOD(baby, hello)
23{
24    printf("hello world!!!!!\n");
25}
26 
27static zend_function_entry baby_method[]={
28    ZEND_ME(baby, __construct, NULL, ZEND_ACC_PUBLIC|ZEND_ACC_CTOR)
29    ZEND_ME(baby, hello, NULL, ZEND_ACC_PUBLIC)
30    {NULL, NULL, NULL}
31};
32 
33ZEND_MINIT_FUNCTION(test)
34{
35    zend_class_entry ce;
36    INIT_CLASS_ENTRY(ce, "baby", baby_method);
37    baby_ce = zend_register_internal_class(&ce TSRMLS_CC);
38    return SUCCESS;
39}

上一篇: PHP内核探索:对象属性读写

下一篇: PHP内核探索:对象

10
登录 后评论.
没有帐号? 现在注册.
0 评论
Table of content