如何从PHP扩展返回数组,而不将其复制到内存中?

我正在开发一个 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扩展函数返回它.更改函数签名以通过引用返回值不是一个选项,因为它链接属性和返回值 – 即稍后更改返回值,也会更改属性.这不是一种可接受的行为.

无法参与引用计数看起来很奇怪,因为PHP中的代码相同:

function myMethod() {
{
    return $this->myArrayProperty;
}

通过引用计数机制进行优化.这就是我在StackOverflow上提出这个问题的原因,以防我错过了什么.

那么,有没有办法从PHP扩展中的函数返回一个数组,而不是在内存中复制数组?

如果函数返回值,则只能使用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);
}

相关文章

统一支付是JSAPI/NATIVE/APP各种支付场景下生成支付订单,返...
统一支付是JSAPI/NATIVE/APP各种支付场景下生成支付订单,返...
前言 之前做了微信登录,所以总结一下微信授权登录并获取用户...
FastAdmin是我第一个接触的后台管理系统框架。FastAdmin是一...
之前公司需要一个内部的通讯软件,就叫我做一个。通讯软件嘛...
统一支付是JSAPI/NATIVE/APP各种支付场景下生成支付订单,返...