2024-09-29 17:13:27 发布
网友
我已经搜索了good和Stack Overflow,但找不到我所要查找的内容的答案。无论如何,是否可以从C++内部调用Python函数调用?捕获函数调用和it参数?在
用示例编辑:
定义a(pOne): //做一些事情
定义b(): a(“a”)
谢谢
最好的方法是“扩展”Python。在
我最喜欢的方法是Boost.Python
您也可以使用原始的C-API,但我不建议这样做。在
一些嵌入和扩展的示例Boost.Python你可以看看我一直在做的this项目。在
如果您只想编写一些C++代码并使它可以从Python调用,那么您可以通过{a1}来实现。(实际上,只是延伸;忽略另一半。)
基本上,如果您编写一个新的模块foo,任何人都可以import foo并调用foo.a("a"),而模块foo是作为Python文件foo.py实现的,还是从foo.cpp编译到动态库foo.so中也没有关系。在
foo
import foo
foo.a("a")
foo.py
foo.cpp
foo.so
对于这一点,正如凯尔C所建议的那样,有许多更高级的方法可以做到这一点,因此您不需要直接处理C API。(除了他的建议Boost.Python我也建议看^{},它让你用几乎可以直接与C++对话的Python语言编写代码,同时也可以直接把事情直接暴露给Python。
然而,这不是你想要的。您希望能够做的是获取一些在Python代码中定义的函数,并将其钩住以执行不同的操作。为此,您确实需要阅读上面链接的扩展和嵌入文档。您必须编写一个嵌入式解释器,重现一些行为(具体多少取决于您到底想在哪里钩住它),并确保无论在哪里调用PyObject_Call或类似的函数,它首先检查该对象是否是a函数,如果是,则调用钩子代码。在
PyObject_Call
a
这将是相当困难的。如果您还没有编写一个嵌入式Python解释器,那么在考虑如何钩住它之前就开始编写。在
值得注意的是,从Python内部进行挂接可能比从解释器外部进行钩稽要容易得多。(如果你从来没有听说过“MangKePog”,就去谷歌。)你可以总是从你在C++中构建的模块中创建你的Python钩子调用代码,或者甚至通过^ {CD11}}直接调用一个编译的.so文件。在
最后,如果您想在运行时钩住一些正在运行的解释器实例,那就更困难了。显然,您需要能够执行某种类型的调试/跟踪/etc.attach和代码插入,这些操作的细节完全依赖于您的平台。然后,您将希望执行与以前的硬版本相同的操作,只是您必须先拦截对以下对象的调用,例如PyObject_Call(from libpython.so/Python.dll/Python.framework/无论什么)的调用,以先通过钩子。如果您还不知道如何在平台上钩住SO调用,那么在考虑从外部钩住Python代码之前,您需要先了解这一点。在
libpython.so
Python.dll
Python.framework
你有理由不修改python代码吗?如果不是,为什么不编写一个扩展模块或python ctypes调用to your dll,并用调用钩子的修饰符包装{}?在
import my_cpp_module def hook_a_to_call_my_cpp_module(orig_a): def replacement_a(pOne): try: my_cpp_module.my_cpp_function(pOne) finally: return orig_a(pOne) # return replacement_a @hook_a_to_call_my_cpp_module def a(pOne): pass
您甚至可以在不接触源文件的情况下执行此操作,该文件包含a,如下所示:
如果你有一些理由根本不改变你的py代码,你能在评论中说明吗?在
最好的方法是“扩展”Python。在
我最喜欢的方法是Boost.Python
您也可以使用原始的C-API,但我不建议这样做。在
一些嵌入和扩展的示例Boost.Python你可以看看我一直在做的this项目。在
如果您只想编写一些C++代码并使它可以从Python调用,那么您可以通过{a1}来实现。(实际上,只是延伸;忽略另一半。)
基本上,如果您编写一个新的模块
foo
,任何人都可以import foo
并调用foo.a("a")
,而模块foo
是作为Python文件foo.py
实现的,还是从foo.cpp
编译到动态库foo.so
中也没有关系。在对于这一点,正如凯尔C所建议的那样,有许多更高级的方法可以做到这一点,因此您不需要直接处理C API。(除了他的建议Boost.Python我也建议看^{} ,它让你用几乎可以直接与C++对话的Python语言编写代码,同时也可以直接把事情直接暴露给Python。
然而,这不是你想要的。您希望能够做的是获取一些在Python代码中定义的函数,并将其钩住以执行不同的操作。为此,您确实需要阅读上面链接的扩展和嵌入文档。您必须编写一个嵌入式解释器,重现一些行为(具体多少取决于您到底想在哪里钩住它),并确保无论在哪里调用
PyObject_Call
或类似的函数,它首先检查该对象是否是a
函数,如果是,则调用钩子代码。在这将是相当困难的。如果您还没有编写一个嵌入式Python解释器,那么在考虑如何钩住它之前就开始编写。在
值得注意的是,从Python内部进行挂接可能比从解释器外部进行钩稽要容易得多。(如果你从来没有听说过“MangKePog”,就去谷歌。)你可以总是从你在C++中构建的模块中创建你的Python钩子调用代码,或者甚至通过^ {CD11}}直接调用一个编译的.so文件。在
最后,如果您想在运行时钩住一些正在运行的解释器实例,那就更困难了。显然,您需要能够执行某种类型的调试/跟踪/etc.attach和代码插入,这些操作的细节完全依赖于您的平台。然后,您将希望执行与以前的硬版本相同的操作,只是您必须先拦截对以下对象的调用,例如
PyObject_Call
(fromlibpython.so
/Python.dll
/Python.framework
/无论什么)的调用,以先通过钩子。如果您还不知道如何在平台上钩住SO调用,那么在考虑从外部钩住Python代码之前,您需要先了解这一点。在你有理由不修改python代码吗?如果不是,为什么不编写一个扩展模块或python ctypes调用to your dll,并用调用钩子的修饰符包装{}?在
您甚至可以在不接触源文件的情况下执行此操作,该文件包含
^{pr2}$a
,如下所示:如果你有一些理由根本不改变你的py代码,你能在评论中说明吗?在
相关问题 更多 >
编程相关推荐