PyInstance_New与PyObject_CallObject有什么区别?

2013-12-10 13:46:45 +08:00
 initialdp
主程序是C/C++程序,调用Python定义的类。用这两个函数都能实例化对象,不知道有什么区别?

另外,我们原程序是采用PyObject_CallObject函数来实例化对象,运行几天后,这个函数居然返回NULL,这是怎么回事?有朋友遇到过类似的问题么?程序是windows GUI程序,没有重定向std:err,无法拿到PyErr_Print的输出信息。

谢谢。
9709 次点击
所在节点    Python
7 条回复
pynix
2013-12-11 08:41:23 +08:00
看源代码注释咯。。。。
l142857
2013-12-11 12:20:15 +08:00
想办法把sys.stderr重定向到文件吧。
l142857
2013-12-11 12:22:15 +08:00
PyInstance_New只能实例化old style class.
initialdp
2013-12-11 14:17:31 +08:00
@l142857 我看了一下2.7.6的代码,这两函数都最终调用PyEval_CallObjectWithKeywords来实例化对象。不同的是:PyInstance_New会先对类的__init__处理做一些判断。

最后,创建了一个cStringIO重定向stderr,看最终是什么结果。
l142857
2013-12-11 16:37:58 +08:00
@initialdp 不同之处应该不是这样的吧,PyObject_CallObject定义在abstract.c里边,从设计上来看应该是一个通用接口函数(执行第一个参数所代表的可执行对象),而PyInstance_New是PyClass_Type的tp_call域的值,它的职能很明确,就是用来实例化old style class的。所以如果有一个类A,则调用PyObject_CallObject(A,...)将最终导致PyInstance_New(A,...)的执行。
initialdp
2013-12-11 18:27:43 +08:00
@l142857 我不太清楚具体细化下去的流程,不过还是非常感谢您的说明啊。

重定向stderr后,发现问题与这个没关系。以前不知道哪位哥们在类初始化(__init__)时,调用了python logging,打开了一个log文件,但是在实例释放时居然没有关闭,因此系统运行一段时间后,无可用文件句柄,导致创建实例失败。SHIT!

话说回来,感觉logging设计也值得商榷啊。既然python本身是支持自动回收的,那设计log文件对象,完全可以在回收的时候自动关闭文件,这样就不会有问题了。
l142857
2013-12-11 19:01:37 +08:00
@initialdp 哈哈,各种坑啊...

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

https://www.v2ex.com/t/92546

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

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

© 2021 V2EX