- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我刚刚查看了一些为 Python 创建 C 扩展模块的代码,该模块不包含足够的错误检查。在大多数情况下这很容易,但当涉及到模块初始化函数时我不确定。
为了讨论方便,我们取(精简的)module-init function for itertools
(是的,CPython 提供的那个):
m = PyModule_Create(&itertoolsmodule);
if (m == NULL)
return NULL;
for (i=0 ; typelist[i] != NULL ; i++) {
if (PyType_Ready(typelist[i]) < 0)
return NULL;
name = strchr(typelist[i]->tp_name, '.');
assert (name != NULL);
Py_INCREF(typelist[i]);
PyModule_AddObject(m, name+1, (PyObject *)typelist[i]);
}
return m;
它确实检查 PyModule_Create
是否失败(这很好),然后检查 PyType_Ready
是否失败(这很好),但它不会 Py_DECREF( m)
在这种情况下(这是令人惊讶/令人困惑的),但它完全无法检查 PyModule_AddObject
是否失败。根据it's documentation它可能失败:
Add an object to module as name. This is a convenience function which can be used from the module’s initialization function. This steals a reference to value. Return -1 on error, 0 on success.
好吧,也许为了防止无法添加类型而破坏模块初始化似乎有点矫枉过正。但即使他们不想完全中止创建模块:它应该泄漏 typelist[i]
的引用,对吗?
许多内置 CPython C 模块不会在 module-init 函数中进行彻底的错误检查和处理(这可能就是我正在修复的 C 扩展也没有它们的原因),而且它们通常非常严格存在此类问题和潜在的泄漏。所以我的问题基本上是:错误检查在 module-init 函数中是否重要,特别是当涉及 PyModule_Add*
函数(如 PyModule_AddObject
)时?或者可以像 CPython 在很多地方一样省略它们吗?
最佳答案
我通常赞成在使用 Python 的 C API 时进行严格的错误检查 - 人们经常编写长的、多步骤的函数,不检查任何错误,然后当它神秘地失败时表现出困惑。在这种情况下(模块初始化),您可以证明错误检查稍微宽松是合理的:
主要原因是,这些函数只会因 C 代码中的错误而真正失败,并且它们会重复执行此操作 - 对于毫无戒心的用户来说,它们几乎不可能不可预测地失败。服用PyModule_AddObject
例如,它可能会失败,因为:
NULL
(您应该提前检查这一点)__dict__
(我不知道这是如何发生的,但我看不出它是偶然发生在您刚刚创建的模块上的)<PyDict_SetItemString
失败(很可能是由 PyUnicode_FromString
失败引起的)。正如您在评论中指出的那样,后者可能是由 MemoryError
引起的(这可能随时发生并且不可预测)。但是,当您因分配约 10 个字符串而出现 MemoryError
时,Python 解释器不太可能继续运行更长时间。
所以我认为我的结论是“如果你的模块似乎工作正常,你可能不需要这个错误检查,但如果事情出了问题,那么它对于找出问题所在是有用的”。我可能要添加的一件事是在返回模块之前对错误进行最终检查:
if (PyErr_Occurred()) return NULL;
/* or */
if (PyErr_Occurred()) {
/* print a warning? */
PyErr_Clear();
return m;
}
这样做的原因是,如果设置了错误指示符但不返回 NULL,Python 的行为可能会非常奇怪(您会在奇怪的时间引发没有意义的异常) )。因此,快速的最终检查具有一定的值(value)。
<小时/>关于模块初始化失败时的引用处理:显然,正确执行它是“最好的”,但我认为您可以证明跳过它是合理的。这是运行一次的代码(因此您不会因为反复丢失少量内存而丢失大量内存)。如果发生错误,那么最可能的选择是程序中止(因此所有内存都被恢复)。即使您不中止,泄漏的大小也可能非常小(实际上约为 100 字节)。
关于python - 如果 PyModule_Add* 函数失败,C 扩展是否应该在模块初始化中失败?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46687928/
我在使用以下代码时遇到问题: function http_file_exists($url){ $f=fopen($url,"r"); if($f){ fclose($f); retu
我已经通过 Git 部署到 Azure 几个月了,没有出现重大问题,但现在我似乎遇到了一个无法克服的错误。 我创建了一个新的 Azure 网站,为正在开发的项目创建单独的预览链接。我在新站点上设置了
我已经通过flutter创建了一个App并完成了它,我想在flutter文档中阅读时进行部署。 我收到此错误: FAILURE: Build failed with an exception. * W
我在Windows 10中使用一些简单的Powershell代码遇到了这个奇怪的问题,我认为这可能是我做错了,但我不是Powershell的天才。 我有这个: $ix = [System.Net.Dn
我正在尝试使用 RapidJSON 解析从服务器接收到的数据。以下是收到的确切字符串: [ { "Node": "9478149a08f9", "Address": "172.17
我尝试为 ios 编译 OpenCV。我总是收到这些错误。我用不同版本的opencv试了一下,结果都是一样的。 我运行这个:python 平台/ios/build_framework.py ios_o
我在一台机器上做基本的发布/订阅,我的客户端是 StackExchange-Redis 的 C# 客户端,我在同一台机器上运行基于 Windows 的 Redis 服务器(服务器版本 2.8.4) 当
我有这段代码,但无法执行,请帮我解决这个问题 连接 connect_error) { die ("connection failed: " . $terhubung->connect_erro
我在 tomcat 上运行并由 maven 编译的 Web 应用程序给出了以下警告和错误。我可以在本地存储库中看到所有 JAR,但有人可以帮忙吗。 WARNING: Failed to scan JA
我正在 Windows 8 上使用 Android Studio 开发一个 android 应用程序,我正在使用一些 native 代码。突然间我无法编译我的 C 文件。当我运行 ndk-build
下面的代码对类和结构的成员进行序列化和反序列化。序列化工作正常,但我在尝试使用 oarch >> BOOST_SERIALIZATION_NVP(outObj); 反序列化时遇到了以下错误; 代码中是
如果我运行此命令“rspec ./spec/requests/api/v1/password_reset_request_spec.rb”,此文件中的所有测试都会通过。 但是,当我运行“rspec”时
我在尝试执行测试以使用 Protractor 上传文件时出错,我的代码是这个 it('it should be possible to upload a file', function() {
System.loadLibrary("nativefaceswap"); 当我运行我的应用程序时,我在 Android Studio 中发现了此类错误。在logcat中显示: java.lang.U
我希望有人能帮助我!使用任何方法或命令行的任何 SSL/HTTPS 调用均无效。 我在 Windows 10 中使用 Ubuntu Server 18.04 作为子系统。我的问题是昨天才开始出现的,因
通过删除这两个值将日期字段从 null=True 和 Blank=True 更改为 required 时,使用 db.alter 命令时遇到问题。 当以下行被注释掉时,迁移运行不会出现问题。
我第一次使用 Heroku 尝试创建应用程序(使用 SendGrid 的 Inbound Parse Webhook"和 Twilio SMS 通过电子邮件发送和接收 SMS 消息)。通过 Virtu
我正在将我的 swift 项目更新到 Xcode 7 上的 Swift 2.0。xcode 在构建项目时报告了以下错误: 命令/Applications/Xcode.app/Contents/Deve
在我的代码中,SSL 库函数 SSL_library_init() 没有按预期返回 1。我如何才能看到它返回了什么错误? 我在 SSL_library_init() 之后调用了 SSL_load_er
我正在尝试运行在以下链接中找到的答案: Asynchronously Load the Contents of a Div 但是当我这样做时,我会遇到我不太理解的错误。 我的代码: $(documen
我是一名优秀的程序员,十分优秀!