gpt4 book ai didi

php - 在 JsonSerializable 中编码克隆 $this

转载 作者:可可西里 更新时间:2023-10-31 23:37:18 25 4
gpt4 key购买 nike

这个简化的案例会导致 PHP 段错误(退出 127):

class Datum implements \JsonSerializable{
public function jsonSerialize(){
return clone $this;
}
}
echo json_encode(new Datum);

最后一行代码导致 exit(127)。我无法在当前环境中检索任何堆栈。

与此同时,删除 clone token 有效。

是否有任何可能的解释为什么会发生这种情况?

最佳答案

此代码导致无限递归。

看来 PHP JSON 模块以这种方式(伪代码)支持 JsonSerializable:

function json_encode($data){
if($data instanceof JsonSerializable) return json_encode($data->jsonSerialize());
else real_json_encode($data); // handling primitive data or arrays or pure data objects
}

如果您返回另一个 JsonSerializable 实例,json_encode 将尝试再次序列化它,从而导致无限递归。

这适用于 return $this;,然而,可能是由于 json_encode 的实现有意变通,当返回的对象相同时,它直接进入真正的 json_encode,即当 $this 被返回。然而,这不会发生在克隆对象上,因为 $a !== clone $a

引用资料

这个答案可以通过 php-src 的引用来支持。

// in php_json_encode_zval
if (instanceof_function(Z_OBJCE_P(val), php_json_serializable_ce)) {
return php_json_encode_serializable_object(buf, val, options, encoder);
}

// in php_json_encode_serializable_object
if ((Z_TYPE(retval) == IS_OBJECT) &&
(Z_OBJ(retval) == Z_OBJ_P(val))) {
/* Handle the case where jsonSerialize does: return $this; by going straight to encode array */
PHP_JSON_HASH_APPLY_PROTECTION_DEC(myht);
return_code = php_json_encode_array(buf, &retval, options, encoder);
} else {
/* All other types, encode as normal */
return_code = php_json_encode_zval(buf, &retval, options, encoder);
PHP_JSON_HASH_APPLY_PROTECTION_DEC(myht);
}

这些片段证明 PHP 会将 return $this; 编码为数组(或不可序列化对象),同时返回任何其他内容使得 Z_OBJ(retval) == Z_OBJ_P( val) false,转到 else block ,再次递归调用 php_json_encode_zval

TL;DR,简单的解决方案:return (array) $this; 而不是 clone $this;

关于php - 在 JsonSerializable 中编码克隆 $this,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40870614/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com