Py_INCREF / DECREF:何时
说明以下内容是否正确:
-
如果在C函数中创建了Python对象,但该函数未返回该对象,
INCREF
则不需要,而是一个DECREF
。 -
[false]如果函数确实返回了它,则
INCREF
在接收返回值的函数中确实需要执行[/ false]。 -
当分配了C类型的变量作为属性,如
double
,int
等等,给Python对象,没有INCREF
或DECREF
需要。 -
使用以下物品安全吗?将Python对象作为属性分配给其他Python对象的过程如下:
PyObject *foo;
foo = bar; // A Python object
tmp = self->foo;
Py_INCREF(foo);
self->foo = foo;
Py_XDECREF(tmp);
// taken from the manual, but it is unclear if this works in every situation -
DECREF
对于具有属性的所有其他Python对象,都需要对Python对象进行释放,而对于C类型的属性则不需要。
编辑:
关于“ C类型作为属性”,我的意思是bar和baz:
typedef struct {
PyObject_HEAD
PyObject *foo;
int bar;
double baz;
} FooBarBaz;
-
首先,请仔细阅读,尤其是最后一段,http://docs.python.org/extending/extending.html#ownership-
rules考虑它的简单方法是考虑参考计数。
-
您的第一句话是正确的。如果您创建一个新的Python对象(例如
PyLong
),则它的引用计数已经为1。如果您要返回它,那么很好,但是如果您不打算返回它,那么它需要由Python进行垃圾收集并且仅将refcount = 0标记为GC,因此如果不打算返回,则需要DECREF。 -
第二个陈述是错误的。如果您需要返回它并创建它,则只需将其返回即可。归还所有权。如果要在返回前使用INCREF,则是要告诉Python 您 也保留了一个副本。同样,如果创建它,则refcount = 1。如果然后执行INCREF,则refcount = 2。但这 不是 您想要的,您想以refcount = 1返回。
-
我 不太 确定我是否明白这一点,但这更多是与C相关的问题。如何将
int
或添加double
到Python对象? -
您能举个例子说明这种方法行不通吗?
-
同样,我不确定C类型何时是Python对象的属性。每
int
,double
,long
等是通过以某种方式或另一种Python对象缠绕。
上面的链接中概述了这些答案的注意事项。阅读完之后,您甚至根本不需要我的拙劣解释。我希望我澄清了,不要让更多的困惑。
-