Py_INCREF / DECREF:何时

发布于 2021-01-29 17:11:42

说明以下内容是否正确:

  1. 如果在C函数中创建了Python对象,但该函数未返回该对象,INCREF则不需要,而是一个DECREF

  2. [false]如果函数确实返回了它,则INCREF在接收返回值的函数中确实需要执行[/ false]。

  3. 当分配了C类型的变量作为属性,如doubleint等等,给Python对象,没有INCREFDECREF需要。

  4. 使用以下物品安全吗?将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

  5. DECREF对于具有属性的所有其他Python对象,都需要对Python对象进行释放,而对于C类型的属性则不需要。


编辑:

关于“ C类型作为属性”,我的意思是bar和baz:

typedef struct {
    PyObject_HEAD
    PyObject *foo;
    int bar;
    double baz;
} FooBarBaz;
关注者
0
被浏览
34
1 个回答
  • 面试哥
    面试哥 2021-01-29
    为面试而生,有面试问题,就找面试哥。

    首先,请仔细阅读,尤其是最后一段,http://docs.python.org/extending/extending.html#ownership-
    rules

    考虑它的简单方法是考虑参考计数。

    1. 您的第一句话是正确的。如果您创建一个新的Python对象(例如PyLong),则它的引用计数已经为1。如果您要返回它,那么很好,但是如果您不打算返回它,那么它需要由Python进行垃圾收集并且仅将refcount = 0标记为GC,因此如果不打算返回,则需要DECREF。

    2. 第二个陈述是错误的。如果您需要返回它并创建它,则只需将其返回即可。归还所有权。如果要在返回前使用INCREF,则是要告诉Python 也保留了一个副本。同样,如果创建它,则refcount = 1。如果然后执行INCREF,则refcount = 2。但这 不是 您想要的,您想以refcount = 1返回。

    3. 不太 确定我是否明白这一点,但这更多是与C相关的问题。如何将int或添加double到Python对象?

    4. 您能举个例子说明这种方法行不通吗?

    5. 同样,我不确定C类型何时是Python对象的属性。每intdoublelong等是通过以某种方式或另一种Python对象缠绕。

    上面的链接中概述了这些答案的注意事项。阅读完之后,您甚至根本不需要我的拙劣解释。我希望我澄清了,不要让更多的困惑。



知识点
面圈网VIP题库

面圈网VIP题库全新上线,海量真题题库资源。 90大类考试,超10万份考试真题开放下载啦

去下载看看