首页 > 开发 > php > 正文

php的RETURN_STRINGL为什么使用静态字符串会出现Segmentation fault

2017-09-06 15:07:20  来源:网友分享

我的c基础不是很好,还请知道的能详解一下,谢谢

如果我直接使用类似如下的代码,就会出segfault错误

char* ret = "hello world";RETURN_STRINGL(ret, strlen(ret), 0);

无论ret是直接写字符串,还是先初始化成char[100]这样,都不行

但是只要将程序稍加改进使用动态分配内存就没事:

char* hello = "hello world";int   len = strlen(hello);char* ret = (char*)emalloc(len);memcpy(ret, hello, len);RETURN_STRINGL(ret, len, 0);

补充一句:后来发现RETURN_STRINGL的第三个参数改成1也不会有越界访问错误了

解决方案

@皮皮鲁 回答的完全错误,纯属于误导,没想到还被采纳了。"hello world"这个字符串是一个静态字符串,并不在栈里面。char* hello只是指向它而已。

这个问题的实质是:
php本身是类型安全的脚本语言,对于RETURN_STRINGL或是RETURN_STRING返回的字符串,php会在适当的时候free掉,所以程序员要保证返回的字符串在堆里,能够free掉,这就是为什么动态分配就没事的原因。而:

char* ret = "hello world";
RETURN_STRINGL(ret, strlen(ret), 0);

这是直接返回了一个静态字符串,导致php在free这个字符串的时候出错。

RETURN_STRINGL和RETURN_STRING最后一个参数,如果是1,表示对第一个参数中的字符串在堆里复制一份返回。这就是为什么最后一个参数等于1的时候,程序正常的原因。