您好, 欢迎来到 !    登录 | 注册 | | 设为首页 | 收藏本站

Python:检查异常引发的位置

Python:检查异常引发的位置

这是一件很愚蠢的事情,大多数人会说这是不可能的(THC4k为一般的问题提供了令人信服的证据),但听起来确实很有趣,并且在许多实际用例中应该是完全可行的。

。您需要后退框架。用sys._getframe或来获得第一个inspect.currentframe(不要告诉任何人,第二个似乎是第一个的别名)。然后,您可以使用f.f_back

。每个人都会有一条f.f_lasti指令。这是框架中执行的最后一条指令。您必须保存它。现在遍历字节码中的后缀f.f_code.co_code--查找SETUP_EXCEPT带有f.f_lasti`之后 的参数的操作码。跳转点是异常处理。

。这是变得更加模糊的地方。关键是实际的比较操作将是一个COMPARE_OP以10为参数的a。在我所见过的所有情况下,都带有一个POP_JUMP_IF_FALSE。这将跳到下一个except子句或该finally子句。它将在加载代码之前将异常加载到堆栈上。如果只有一个,则它将是一个直线LOAD_GLOBAL或aLOAD_GLOBALLOAD_FAST(取决于模块是全局模块还是局部模块),然后是aLOAD_ATTR。如果有多个异常被匹配,那么将有一系列加载操作,然后是BUILD_TUPLE(惯用的)或BUILD_LIST(某些其他奇怪或非惯用的情况)。

关键是您可以按照LOAD_X说明进行操作,并将名称与要匹配的异常进行比较。请注意,您 比较名称 。如果他们已经重新分配了名称,那么您就是SOL。

。假设您找到了一个匹配项。现在您需要函数对象。我可以想到的最好方法是(我保留更新的权利):f.f_code将具有co_filename属性。您可以循环浏览sys.modules,每个都有__name__属性。您可以比较应该使用的两个注意事项__name__.endswith(co_filename)。当您找到匹配项时,可以遍历模块功能并将它们的f.func_code.co_firstlineno属性与框架进行比较f.f_lineno属性。当您找到一个匹配项时,您便具有了自己的功能。您还应该遍历模块中每个类的方法。在某些情况下,这种处理有可能发生在某些嵌套函数中,我目前无法想到要这样做。(这将是其他字节码的入侵,并且本身就是flakey)

。利润。

这应该使您大致了解如何执行此操作。在各种各样的极端情况下,您将无法做到这一点,但是在任何正常的用例中,您都应该能够做到。如果您编写依赖于此功能代码,尽管 失败。这有点像“尽我所能”。

python 2022/1/1 18:48:11 有337人围观

撰写回答


你尚未登录,登录后可以

和开发者交流问题的细节

关注并接收问题和回答的更新提醒

参与内容的编辑和改进,让解决方法与时俱进

请先登录

推荐问题


联系我
置顶