我正在开发一个
PHP扩展,其中一个对象方法需要返回一个数组zval.
该方法如下:
ZEND_METHOD(myObject,myMethod) { zval **myArrayProperty; if (zend_hash_find(Z_OBJPROP_P(getThis()),"myArrayProperty",sizeof("myArrayProperty"),(void **) &myArrayProperty) == FAILURE) { RETURN_FALSE; } RETURN_ZVAL(*myArrayProperty,1,0); }
代码工作正常并做了预期的事情 – 它返回对象的myArrayProperty.但是,我想优化这个过程.
myArrayProperty存储一个数组,它可能非常大.并且RETURN_ZVAL()宏复制该数组以返回该值.复制过程需要花费大量时间来获取内存并复制所有数组值.同时,返回的数组通常用于只读操作.所以一个很好的优化是使用PHP的机制和引用计数,不要复制myArrayProperty内容.相反,我会增加myArrayProperty的引用计数,只返回指向它的指针.当使用PHP扩展中的变量时,这与通常使用的策略相同.
但是,似乎没有办法 – 你必须复制值才能从PHP扩展函数返回它.更改函数签名以通过引用返回值不是一个选项,因为它链接属性和返回值 – 即稍后更改返回值,也会更改属性.这不是一种可接受的行为.
function myMethod() { { return $this->myArrayProperty; }
通过引用计数机制进行优化.这就是我在StackOverflow上提出这个问题的原因,以防我错过了什么.
如果函数返回值,则只能使用RETURN_ZVAL_FAST宏从PHP 5.6(当前主数据库)开始:
RETURN_ZVAL_FAST(*myArrayProperty);
如果函数返回by-reference(arginfo中的return_reference = 1),则可以使用以下代码返回:
zval_ptr_dtor(&return_value); SEPARATE_ZVAL_TO_MAKE_IS_REF(myArrayProperty); Z_ADDREF_PP(myArrayProperty); *return_value_ptr = *myArrayProperty;
如果你的函数按值返回并且你使用的是PHP 5.5或更高版本,你仍然可以优化refcount = 1的情况:
if (Z_REFCOUNT_PP(myArrayProperty) == 1) { RETVAL_ZVAL(*myArrayProperty,1); Z_ADDREF_P(return_value); *myArrayProperty = return_value; } else { RETVAL_ZVAL(*myArrayProperty,0); }