51Testing软件测试论坛

 找回密码
 (注-册)加入51Testing

QQ登录

只需一步,快速开始

微信登录,快人一步

手机号码,快捷登录

查看: 2135|回复: 2
打印 上一主题 下一主题

C和Python之间的接口实现

[复制链接]
  • TA的每日心情
    无聊
    2024-9-27 10:07
  • 签到天数: 62 天

    连续签到: 1 天

    [LV.6]测试旅长

    跳转到指定楼层
    1#
    发表于 2018-2-7 17:12:26 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
          1 C/C++ 调用 Python(基础篇)
    Python 本身就是一个C库。你所看到的可执行体python只不过是个stub。真正的python实体在动态链接
    库里实现,在Windows平台上,这个文件位于 %SystemRoot%\System32\python27.dll。

    你也可以在自己的程序中调用Python,看起来非常容易:
    1. //my_python.c
    2. #include <Python.h>

    3. int main(int argc, char *argv[])
    4. {
    5.   Py_SetProgramName(argv[0]);
    6.   Py_Initialize();
    7.   PyRun_SimpleString("print 'Hello Python!'\n");
    8.   Py_Finalize();
    9.   return 0;
    10. }
    复制代码

           在Windows平台下,打开Visual Studio命令提示符,编译命令为
    cl my_python.c -IC:\Python27\include C:\Python27\libs\python27.lib
    在Linux下编译命令为
    gcc my_python.c -o my_python -I/usr/include/python2.7/ -lpython2.7
    在Mac OS X 下的编译命令同上

          产生可执行文件后,直接运行,结果为输出
    Hello Python!
    Python库函数PyRun_SimpleString可以执行字符串形式的Python代码。

          虽然非常简单,但这段代码除了能用C语言动态生成一些Python代码之外,并没有什么用处。我们
    需要的是C语言的数据结构能够和Python交互。

         下面举个例子,比如说,有一天我们用Python写了一个功能特别强大的函数:
    1. def great_function(a):
    2.     return a + 1
    复制代码
          接下来要把它包装成C语言的函数。我们期待的C语言的对应函数应该是这样的:
    1. int great_function_from_python(int a) {
    2.     int res;
    3.     // some magic
    4.     return res;
    5. }
    复制代码
           首先,复用Python模块得做‘import’,这里也不例外。所以我们把great_function放到一个module里,
    比如说,这个module名字叫 great_module.py
    接下来就要用C来调用Python了,完整的代码如下:
    1. #include <Python.h>

    2. int great_function_from_python(int a) {
    3.     int res;
    4.     PyObject *pModule,*pFunc;
    5.     PyObject *pArgs, *pValue;
    6.    
    7.     /* import */
    8.     pModule = PyImport_Import(PyString_FromString("great_module"));

    9.     /* great_module.great_function */
    10.     pFunc = PyObject_GetAttrString(pModule, "great_function");
    11.    
    12.     /* build args */
    13.     pArgs = PyTuple_New(1);
    14.     PyTuple_SetItem(pArgs,0, PyInt_FromLong(a));
    15.       
    16.     /* call */
    17.     pValue = PyObject_CallObject(pFunc, pArgs);
    18.    
    19.     res = PyInt_AsLong(pValue);
    20.     return res;
    21. }
    复制代码
         从上述代码可以窥见Python内部运行的方式:
    所有Python元素,module、function、tuple、string等等,实际上都是PyObject。C语言里操纵它们,
    一律使用PyObject *。
    Python的类型与C语言类型可以相互转换。Python类型XXX转换为C语言类型YYY要使用PyXXX_AsYYY
    函数;C类型YYY转换为Python类型XXX要使用PyXXX_FromYYY函数。
    也可以创建Python类型的变量,使用PyXXX_New可以创建类型为XXX的变量。
    若a是Tuple,则a = b对应于 PyTuple_SetItem(a,i,b),有理由相信还有一个函数PyTuple_GetItem
    完成取得某一项的值。
    不仅Python语言很优雅,Python的库函数API也非常优雅。

    现在我们得到了一个C语言的函数了,可以写一个main测试它
    1. #include <Python.h>

    2. int great_function_from_python(int a);

    3. int main(int argc, char *argv[]) {
    4.     Py_Initialize();
    5.     printf("%d",great_function_from_python(2));
    6.     Py_Finalize();
    7. }
    复制代码
           编译的方式就用本节开头使用的方法。
    在Linux/Mac OSX运行此示例之前,可能先需要设置环境变量:
    bash:
    export PYTHONPATH=.PYTHONPATH
    csh:
    setenv PYTHONPATH .PYTHONPATH

    2 Python 调用 C/C++(基础篇)
    这种做法称为Python扩展。
    比如说,我们有一个功能强大的C函数:
    int great_function(int a) {
        return a + 1;
    }
    期望在Python里这样使用:
    >>> from great_module import great_function
    >>> great_function(2)
    3
    考虑最简单的情况。我们把功能强大的函数放入C文件 great_module.c 中。
    1. #include <Python.h>

    2. int great_function(int a) {
    3.     return a + 1;
    4. }

    5. static PyObject * _great_function(PyObject *self, PyObject *args)
    6. {
    7.     int _a;
    8.     int res;

    9.     if (!PyArg_ParseTuple(args, "i", &_a))
    10.         return NULL;
    11.     res = great_function(_a);
    12.     return PyLong_FromLong(res);
    13. }

    14. static PyMethodDef GreateModuleMethods[] = {
    15.     {
    16.         "great_function",
    17.         _great_function,
    18.         METH_VARARGS,
    19.         ""
    20.     },
    21.     {NULL, NULL, 0, NULL}
    22. };

    23. PyMODINIT_FUNC initgreat_module(void) {
    24.     (void) Py_InitModule("great_module", GreateModuleMethods);
    25. }
    复制代码

           除了功能强大的函数great_function外,这个文件中还有以下部分:
    包裹函数_great_function。它负责将Python的参数转化为C的参数(PyArg_ParseTuple),调用实际
    的great_function,并处理great_function的返回值,最终返回给Python环境。
    导出表GreateModuleMethods。它负责告诉Python这个模块里有哪些函数可以被Python调用。导出表
    的名字可以随便起,每一项有4个参数:第一个参数是提供给Python环境的函数名称,第二个参数是
    _great_function,即包裹函数。第三个参数的含义是参数变长,第四个参数是一个说明性的字符串。
    导出表总是以{NULL, NULL, 0, NULL}结束。
         导出函数initgreat_module。这个的名字不是任取的,是你的module名称添加前缀init。导出函数中将
    模块名称与导出表进行连接。

    在Windows下面,在Visual Studio命令提示符下编译这个文件的命令是
    cl /LD great_module.c /o great_module.pyd -IC:\Python27\include C:\Python27\libs\python27.lib
    /LD 即生成动态链接库。编译成功后在当前目录可以得到 great_module.pyd(实际上是dll)。
    这个pyd可以在Python环境下直接当作module使用。

    在Linux下面,则用gcc编译:
    gcc -fPIC -shared great_module.c -o great_module.so -I/usr/include/python2.7/ -lpython2.7
    在当前目录下得到great_module.so,同理可以在Python中直接使用。




             3 C/C++ 调用 Python(使用Cython)

             在前面的小节中谈到,Python的数据类型和C的数据类型貌似是有某种“一一对应”的关系的,此外,
    由于Python(确切的说是CPython)本身是由C语言实现的,故Python数据类型之间的函数运算也必然
    与C语言有对应关系。那么,有没有可能“自动”的做替换,把Python代码直接变成C代码呢?答案是肯定的,
    这就是Cython主要解决的问题。

            安装Cython非常简单。Python 2.7.9以上的版本已经自带easy_install:
    easy_install -U cython
    在Windows环境下依然需要Visual Studio,由于安装的过程需要编译Cython的源代码,故上述命令需
    要在Visual Studio命令提示符下完成。一会儿使用Cython的时候,也需要在Visual Studio命令提示符下
    进行操作,这一点和第一部分的要求是一样的。

            继续以例子说明:
    #great_module.pyx
    cdef public great_function(a,index):
        return a[index]
    这其中有非Python关键字cdef和public。这些关键字属于Cython。由于我们需要在C语言中使用“编
    译好的Python代码”,所以得让great_function从外面变得可见,方法就是以“public”修饰。而cdef类
    似于Python的def,只有使用cdef才可以使用Cython的关键字public。

          这个函数中其他的部分与正常的Python代码是一样的。

          接下来编译 great_module.pyx
    cython great_module.pyx
    得到great_module.h和great_module.c。打开great_module.h可以找到这样一句声明:
    __PYX_EXTERN_C DL_IMPORT(PyObject) *great_function(PyObject *, PyObject *)
    写一个main使用great_function。注意great_function并不规定a是何种类型,它的功能只是提取a的第
    index的成员而已,故使用great_function的时候,a可以传入Python String,也可以传入tuple之类的
    其他可迭代类型。仍然使用之前提到的类型转换函数PyXXX_FromYYY和PyXXX_AsYYY。
    1. //main.c
    2. #include <Python.h>
    3. #include "great_module.h"

    4. int main(int argc, char *argv[]) {
    5.     PyObject *tuple;
    6.     Py_Initialize();
    7.     initgreat_module();
    8.     printf("%s\n",PyString_AsString(
    9.                 great_function(
    10.                     PyString_FromString("hello"),
    11.                     PyInt_FromLong(1)
    12.                 )
    13.             ));
    14.     tuple = Py_BuildValue("(iis)", 1, 2, "three");
    15.     printf("%d\n",PyInt_AsLong(
    16.                 great_function(
    17.                     tuple,
    18.                     PyInt_FromLong(1)
    19.                 )
    20.             ));
    21.     printf("%s\n",PyString_AsString(
    22.                 great_function(
    23.                     tuple,
    24.                     PyInt_FromLong(2)
    25.                 )
    26.             ));
    27.     Py_Finalize();
    28. }
    复制代码
           编译命令和第一部分相同:
           在Windows下编译命令为
    cl main.c great_module.c -IC:\Python27\include C:\Python27\libs\python27.lib
    在Linux下编译命令为
           gcc main.c great_module.c -o main -I/usr/include/python2.7/ -lpython2.7
    这个例子中我们使用了Python的动态类型特性。如果你想指定类型,可以利用Cython的静态类型关键字。
    例子如下:

           #great_module.pyx
    cdef public char great_function(const char * a,int index):
        return a[index]
    cython编译后得到的.h里,great_function的声明是这样的:
    __PYX_EXTERN_C DL_IMPORT(char) great_function(char const *, int);
    很开心对不对!
    这样的话,我们的main函数已经几乎看不到Python的痕迹了:
    1. //main.c
    2. #include <Python.h>
    3. #include "great_module.h"

    4. int main(int argc, char *argv[]) {
    5.     Py_Initialize();
    6.     initgreat_module();
    7.     printf("%c",great_function("Hello",2));
    8.     Py_Finalize();
    9. }
    复制代码
            这一部分的最后我们给一个看似实用的应用(仅限于Windows):
    还是利用刚才的great_module.pyx,准备一个dllmain.c:
    1. #include <Python.h>
    2. #include <Windows.h>
    3. #include "great_module.h"

    4. extern __declspec(dllexport) int __stdcall _great_function(const char * a, int b) {
    5.     return great_function(a,b);
    6. }

    7. BOOL WINAPI DllMain(HINSTANCE hinstDLL,DWORD fdwReason,LPVOID lpReserved) {
    8.     switch( fdwReason ) {
    9.         case DLL_PROCESS_ATTACH:
    10.             Py_Initialize();
    11.             initgreat_module();
    12.             break;
    13.         case DLL_PROCESS_DETACH:
    14.             Py_Finalize();
    15.             break;
    16.     }
    17.     return TRUE;
    18. }
    复制代码


                  在Visual Studio命令提示符下编译:
    cl /LD dllmain.c great_module.c -IC:\Python27\include C:\Python27\libs\python27.lib
    会得到一个dllmain.dll。我们在Excel里面使用它,没错,传说中的Excel与Python混合编程:
    &amp;lt;img data-rawheight=&quot;797&quot; data-rawwidth=&quot;1007&quot; src=&quot;
    https://pic2.zhimg.com/2f45c9f2f ... 181_b.png&quot; class=&quot;origin_image
    zh-lightbox-thumb&quot; width=&quot;1007&quot; data-original=&quot;https://pic2.zhimg.com/
    2f45c9f2f8407d46f51f203efc2e8181_r.png&quot;&amp;gt;




    分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
    收藏收藏
    回复

    使用道具 举报

  • TA的每日心情
    无聊
    2024-9-27 10:07
  • 签到天数: 62 天

    连续签到: 1 天

    [LV.6]测试旅长

    2#
     楼主| 发表于 2018-2-7 17:13:40 | 只看该作者
             4 Python调用C/C++(使用SWIG)
    用C/C++对脚本语言的功能扩展是非常常见的事情,Python也不例外。除了SWIG,市面上还有若干
    用于Python扩展的工具包,比较知名的还有Boost.Python、SIP等,此外,Cython由于可以直接集成
    C/C++代码,并方便的生成Python模块,故也可以完成扩展Python的任务。

    答主在这里选用SWIG的一个重要原因是,它不仅可以用于Python,也可以用于其他语言。如今SWIG
    已经支持C/C++的好基友Java,主流脚本语言Python、Perl、Ruby、PHP、JavaScript、tcl、Lua,还
    有Go、C#,以及R。SWIG是基于配置的,也就是说,原则上一套配置改变不同的编译方法就能适用各
    种语言(当然,这是理想情况了……)

    SWIG的安装方便,有Windows的预编译包,解压即用,绿色健康。主流Linux通常集成swig的包,也可
    以下载源代码自己编译,SWIG非常小巧,通常安装不会出什么问题。

    用SWIG扩展Python,你需要有一个待扩展的C/C++库。这个库有可能是你自己写的,也有可能是某
    个项目提供的。这里举一个不浮夸的例子:希望在Python中用到SSE4指令集的CRC32指令。

    首先打开指令集的文档:https://software.intel.com/en-us/node/514245
    可以看到有6个函数。分析6个函数的原型,其参数和返回值都是简单的整数。于是书写SWIG的配置
    文件(为了简化起见,未包含2个64位函数):


            /* File: mymodule.i */
    %module mymodule

    %{
    #include "nmmintrin.h"
    %}

    int _mm_popcnt_u32(unsigned int v);
    unsigned int _mm_crc32_u8 (unsigned int crc, unsigned char v);
    unsigned int _mm_crc32_u16(unsigned int crc, unsigned short v);
    unsigned int _mm_crc32_u32(unsigned int crc, unsigned int v);
    接下来使用SWIG将这个配置文件编译为所谓Python Module Wrapper

    swig -python mymodule.i

    得到一个 mymodule_wrap.c和一个mymodule.py。把它编译为Python扩展:

    Windows:
    cl /LD mymodule_wrap.c /o _mymodule.pyd -IC:\Python27\include C:\Python27\libs\python27.lib

    Linux:
    gcc -fPIC -shared mymodule_wrap.c -o _mymodule.so -I/usr/include/python2.7/ -lpython2.7
    注意输出文件名前面要加一个下划线。
    现在可以立即在Python下使用这个module了:

    >>> import mymodule
    >>> mymodule._mm_popcnt_u32(10)
    2



    回复 支持 反对

    使用道具 举报

    本版积分规则

    关闭

    站长推荐上一条 /1 下一条

    小黑屋|手机版|Archiver|51Testing软件测试网 ( 沪ICP备05003035号 关于我们

    GMT+8, 2024-11-27 13:46 , Processed in 0.064690 second(s), 23 queries .

    Powered by Discuz! X3.2

    © 2001-2024 Comsenz Inc.

    快速回复 返回顶部 返回列表