Python 源码中 PyListObject 中的两个问题请教各位大神

2018-03-06 14:10:45 +08:00
 joeHuang
问题一:在创建包含 size 个 PyObject *的 PyListObject 时,内存数量计算,溢出检查。部分代码如下
int size;
size_t nbytes = size * sizeof(PyObject *)
if( nbytes/sizeof(PyObject *) != (size_t) size){
return 溢出
}
这段代码判断溢出的思路是什么?

问题二:在插入元素时,判断是否需要申请新的内存
if (allocated >= newsize >= (allocated>>1)) 不申请新的内存
else new_allocated = (newsize >> 3) + (newsize < 9?3:6)+newsize 申请 new_allocated 个新的内存

这里 new_allocated 的计算逻辑是什么?为什么是这样计算的
2103 次点击
所在节点    Python
7 条回复
joeHuang
2018-03-06 15:03:06 +08:00
wangxn
2018-03-06 15:07:19 +08:00
能不能直接复制源码而不要用奇怪的伪码……
gnaggnoyil
2018-03-06 15:47:40 +08:00
1 中的代码的本意是认为`size * sizeof(PyObject *)`只有在不发生溢出的情况下其值才是数学意义上相乘的结果.不过我很怀疑这段代码可能不能保证 portable:一个`int`和一个`size_t`相乘的结果不保证一定是个无符号的(比如`int`的 rank 严格大于`size_t`的时候).而有符号数溢出是未定义行为.我没怎么读过 CPython 源码,如果 CPython 源码真的是这么写的话我只能说这代码质量还是有待改进……
gnaggnoyil
2018-03-06 15:51:32 +08:00
哦不对我傻了.我忘了有个东西叫做 Integer promotion. Integer promotion 的存在导致`int`的 rank 必然小于等于`size_t`进行 Integer promotion 之后的类型,所以两者相乘的结果必然是严格无符号的,所以这段代码应该是严格没问题的……
joeHuang
2018-03-14 10:03:07 +08:00
@wangxn 帖子里写的就是源码。
joeHuang
2018-03-14 13:49:10 +08:00
wangxn
2018-03-15 23:16:52 +08:00
@joeHuang 连分号都没有,能编译通过?

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://tanronggui.xyz/t/435320

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX