• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            S.l.e!ep.¢%

            像打了激速一樣,以四倍的速度運轉(zhuǎn),開心的工作
            簡單、開放、平等的公司文化;尊重個性、自由與個人價值;
            posts - 1098, comments - 335, trackbacks - 0, articles - 1
              C++博客 :: 首頁 :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理

            用C語言擴展Python的功能

            Posted on 2009-10-12 22:08 S.l.e!ep.¢% 閱讀(766) 評論(0)  編輯 收藏 引用 所屬分類: python

            用C語言擴展Python的功能

            如何找到、使用和參與開放源代碼項目

            developerWorks
            文檔選項
            將打印機的版面設(shè)置成橫向打印模式

            打印本頁

            將此頁作為電子郵件發(fā)送

            將此頁作為電子郵件發(fā)送

            未顯示需要 JavaScript 的文檔選項


            級別: 初級

            肖文鵬 (xiaowp@263.net), 北京理工大學(xué)計算機系碩士研究生

            2003 年 2 月 03 日

            Pyton和C分別有著各自的優(yōu)缺點,用Python開發(fā)程序速度快,可靠性高,并且有許多現(xiàn)成模塊可供使用,但執(zhí)行速度相對較慢;C語言則正好相反,其執(zhí)行速度快,但開發(fā)效率低。為了充分利用兩種語言各自的優(yōu)點,比較好的做法是用Python開發(fā)整個軟件框架,而用C語言實現(xiàn)其關(guān)鍵模塊。本文介紹如何利用C語言來擴展Python的功能,并輔以具體的實例講述如何編寫Python的擴展模塊。

            一、簡介

            Python是一門功能強大的高級腳本語言,它的強大不僅表現(xiàn)在其自身的功能上,而且還表現(xiàn)在其良好的可擴展性上,正因如此,Python已經(jīng)開始受到越來越多人的青睞,并且被屢屢成功地應(yīng)用于各類大型軟件系統(tǒng)的開發(fā)過程中。

            與其它普通腳本語言有所不同,Python程序員可以借助Python語言提供的API,使用C或者C++來對Python進行功能性擴展,從而即可以利用Python方便靈活的語法和功能,又可以獲得與C或者C++幾乎相同的執(zhí)行性能。執(zhí)行速度慢是幾乎所有腳本語言都具有的共性,也是倍受人們指責(zé)的一個重要因素,Python則通過與C語言的有機結(jié)合巧妙地解決了這一問題,從而使腳本語言的應(yīng)用范圍得到了很大擴展。

            在用Python開發(fā)實際軟件系統(tǒng)時,很多時候都需要使用C/C++來對Python進行擴展。最常見的情況是目前已經(jīng)存在一個用C編寫的庫,需要在Python語言中使用該庫的某些功能,此時就可以借助Python提供的擴展功能來實現(xiàn)。此外,由于Python從本質(zhì)上講還是一種腳本語言,某些功能用Python實現(xiàn)可能很難滿足實際軟件系統(tǒng)對執(zhí)行效率的要求,此時也可以借助Python提供的擴展功能,將這些關(guān)鍵代碼段用C或者C++實現(xiàn),從而提供程序的執(zhí)行性能。

            本文主要介紹Python提供的C語言擴展接口,以及如何使用這些接口和C/C++語言來對Python進行功能性擴展,并輔以具體的實例講述如何實現(xiàn)Python的功能擴展。





            回頁首


            二、Python的C語言接口

            Python是用C語言實現(xiàn)的一種腳本語言,本身具有優(yōu)良的開放性和可擴展性,并提供了方便靈活的應(yīng)用程序接口(API),從而使得C/C++程序員能夠在各個級別上對Python解釋器的功能進行擴展。在使用C/C++對Python進行功能擴展之前,必須首先掌握Python解釋所提供的C語言接口。

            2.1 Python對象(PyObject)

            Python是一門面向?qū)ο蟮哪_本語言,所有的對象在Python解釋器中都被表示成PyObject,PyObject結(jié)構(gòu)包含Python對象的所有成員指針,并且對Python對象的類型信息和引用計數(shù)進行維護。在進行Python的擴展編程時,一旦要在C或者C++中對Python對象進行處理,就意味著要維護一個PyObject結(jié)構(gòu)。

            在Python的C語言擴展接口中,大部分函數(shù)都有一個或者多個參數(shù)為PyObject指針類型,并且返回值也大都為PyObject指針。

            2.2 引用計數(shù)

            為了簡化內(nèi)存管理,Python通過引用計數(shù)機制實現(xiàn)了自動的垃圾回收功能,Python中的每個對象都有一個引用計數(shù),用來計數(shù)該對象在不同場所分別被引用了多少次。每當(dāng)引用一次Python對象,相應(yīng)的引用計數(shù)就增1,每當(dāng)消毀一次Python對象,則相應(yīng)的引用就減1,只有當(dāng)引用計數(shù)為零時,才真正從內(nèi)存中刪除Python對象。

            下面的例子說明了Python解釋器如何利用引用計數(shù)來對Pyhon對象進行管理:

            例1:refcount.py
            class refcount:
                # etc.
            r1 = refcount() # 引用計數(shù)為1
            r2 = r1         # 引用計數(shù)為2
            del(r1)         # 引用計數(shù)為1
            del(r2)         # 引用計數(shù)為0,刪除對象
            

            在C/C++中處理Python對象時,對引用計數(shù)進行正確的維護是一個關(guān)鍵問題,處理不好將很容易產(chǎn)生內(nèi)存泄漏。Python的C語言接口提供了一些宏來對引用計數(shù)進行維護,最常見的是用Py_INCREF()來增加使Python對象的引用計數(shù)增1,用Py_DECREF()來使Python對象的引用計數(shù)減1。

            2.3 數(shù)據(jù)類型

            Python定義了六種數(shù)據(jù)類型:整型、浮點型、字符串、元組、列表和字典,在使用C語言對Python進行功能擴展時,首先要了解如何在C和Python的數(shù)據(jù)類型間進行轉(zhuǎn)化。

            2.3.1 整型、浮點型和字符串

            在Python的C語言擴展中要用到整型、浮點型和字符串這三種數(shù)據(jù)類型時相對比較簡單,只需要知道如何生成和維護它們就可以了。下面的例子給出了如何在C語言中使用Python的這三種數(shù)據(jù)類型:

            例2:typeifs.c
            // build an integer
            PyObject* pInt = Py_BuildValue("i", 2003);
            assert(PyInt_Check(pInt));
            int i = PyInt_AsLong(pInt);
            Py_DECREF(pInt);
            // build a float
            PyObject* pFloat = Py_BuildValue("f", 3.14f);
            assert(PyFloat_Check(pFloat));
            float f = PyFloat_AsDouble(pFloat);
            Py_DECREF(pFloat);
            // build a string
            PyObject* pString = Py_BuildValue("s", "Python");
            assert(PyString_Check(pString);
            int nLen = PyString_Size(pString);
            char* s = PyString_AsString(pString);
            Py_DECREF(pString);
            

            2.3.2 元組

            Python語言中的元組是一個長度固定的數(shù)組,當(dāng)Python解釋器調(diào)用C語言擴展中的方法時,所有非關(guān)鍵字(non-keyword)參數(shù)都以元組方式進行傳遞。下面的例子示范了如何在C語言中使用Python的元組類型:

            例3:typetuple.c
            // create the tuple
            PyObject* pTuple = PyTuple_New(3);
            assert(PyTuple_Check(pTuple));
            assert(PyTuple_Size(pTuple) == 3);
            // set the item
            PyTuple_SetItem(pTuple, 0, Py_BuildValue("i", 2003));
            PyTuple_SetItem(pTuple, 1, Py_BuildValue("f", 3.14f));
            PyTuple_SetItem(pTuple, 2, Py_BuildValue("s", "Python"));
            // parse tuple items
            int i;
            float f;
            char *s;
            if (!PyArg_ParseTuple(pTuple, "ifs", &i, &f, &s))
                PyErr_SetString(PyExc_TypeError, "invalid parameter");
            // cleanup
            Py_DECREF(pTuple);
            

            2.3.3 列表

            Python語言中的列表是一個長度可變的數(shù)組,列表比元組更為靈活,使用列表可以對其存儲的Python對象進行隨機訪問。下面的例子示范了如何在C語言中使用Python的列表類型:

            例4:typelist.c
            // create the list
            PyObject* pList = PyList_New(3); // new reference
            assert(PyList_Check(pList));
            // set some initial values
            for(int i = 0; i < 3; ++i)
                PyList_SetItem(pList, i, Py_BuildValue("i", i));
            // insert an item
            PyList_Insert(pList, 2, Py_BuildValue("s", "inserted"));
            // append an item
            PyList_Append(pList, Py_BuildValue("s", "appended"));
            // sort the list
            PyList_Sort(pList);
            // reverse the list
            PyList_Reverse(pList);
            // fetch and manipulate a list slice
            PyObject* pSlice = PyList_GetSlice(pList, 2, 4); // new reference
            for(int j = 0; j < PyList_Size(pSlice); ++j) {
              PyObject *pValue = PyList_GetItem(pList, j);
              assert(pValue);
            }
            Py_DECREF(pSlice);
            // cleanup
            Py_DECREF(pList);
            

            2.3.4 字典

            Python語言中的字典是一個根據(jù)關(guān)鍵字進行訪問的數(shù)據(jù)類型。下面的例子示范了如何在C語言中使用Python的字典類型:

            例5:typedic.c
            // create the dictionary
            PyObject* pDict = PyDict_New(); // new reference
            assert(PyDict_Check(pDict));
            // add a few named values
            PyDict_SetItemString(pDict, "first", 
                                 Py_BuildValue("i", 2003));
            PyDict_SetItemString(pDict, "second", 
                                 Py_BuildValue("f", 3.14f));
            // enumerate all named values
            PyObject* pKeys = PyDict_Keys(); // new reference
            for(int i = 0; i < PyList_Size(pKeys); ++i) {
              PyObject *pKey = PyList_GetItem(pKeys, i);
              PyObject *pValue = PyDict_GetItem(pDict, pKey);
              assert(pValue);
            }
            Py_DECREF(pKeys);
            // remove a named value
            PyDict_DelItemString(pDict, "second");
            // cleanup
            Py_DECREF(pDict);
            





            回頁首


            三、Python的C語言擴展

            3.1 模塊封裝

            在了解了Python的C語言接口后,就可以利用Python解釋器提供的這些接口來編寫Python的C語言擴展,假設(shè)有如下一個C語言函數(shù):

            例6:example.c
            int fact(int n)
            {
              if (n <= 1) 
                return 1;
              else 
                return n * fact(n - 1);
            } 
            

            該函數(shù)的功能是計算某個給定自然數(shù)的階乘,如果想在Python解釋器中調(diào)用該函數(shù),則應(yīng)該首先將其實現(xiàn)為Python中的一個模塊,這需要編寫相應(yīng)的封裝接口,如下所示:

            例7: wrap.c
            #include <Python.h>
            PyObject* wrap_fact(PyObject* self, PyObject* args) 
            {
              int n, result;
              
              if (! PyArg_ParseTuple(args, "i:fact", &n))
                return NULL;
              result = fact(n);
              return Py_BuildValue("i", result);
            }
            static PyMethodDef exampleMethods[] = 
            {
              {"fact", wrap_fact, METH_VARARGS, "Caculate N!"},
              {NULL, NULL}
            };
            void initexample() 
            {
              PyObject* m;
              m = Py_InitModule("example", exampleMethods);
            }
            

            一個典型的Python擴展模塊至少應(yīng)該包含三個部分:導(dǎo)出函數(shù)、方法列表和初始化函數(shù)。

            3.2 導(dǎo)出函數(shù)

            要在Python解釋器中使用C語言中的某個函數(shù),首先要為其編寫相應(yīng)的導(dǎo)出函數(shù),上述例子中的導(dǎo)出函數(shù)為wrap_fact。在Python的C語言擴展中,所有的導(dǎo)出函數(shù)都具有相同的函數(shù)原型:

            PyObject* method(PyObject* self, PyObject* args);

            該函數(shù)是Python解釋器和C函數(shù)進行交互的接口,帶有兩個參數(shù):self和args。參數(shù)self只在C函數(shù)被實現(xiàn)為內(nèi)聯(lián)方法(built-in method)時才被用到,通常該參數(shù)的值為空(NULL)。參數(shù)args中包含了Python解釋器要傳遞給C函數(shù)的所有參數(shù),通常使用Python的C語言擴展接口提供的函數(shù)PyArg_ParseTuple()來獲得這些參數(shù)值。

            所有的導(dǎo)出函數(shù)都返回一個PyObject指針,如果對應(yīng)的C函數(shù)沒有真正的返回值(即返回值類型為void),則應(yīng)返回一個全局的None對象(Py_None),并將其引用計數(shù)增1,如下所示:

            PyObject* method(PyObject *self, PyObject *args) 
            {
              Py_INCREF(Py_None);
              return Py_None;
            }
            

            3.3 方法列表

            方法列表中給出了所有可以被Python解釋器使用的方法,上述例子對應(yīng)的方法列表為:

            static PyMethodDef exampleMethods[] = 
            {
              {"fact", wrap_fact, METH_VARARGS, "Caculate N!"},
              {NULL, NULL}
            };
            

            方法列表中的每項由四個部分組成:方法名、導(dǎo)出函數(shù)、參數(shù)傳遞方式和方法描述。方法名是從Python解釋器中調(diào)用該方法時所使用的名字。參數(shù)傳遞方式則規(guī)定了Python向C函數(shù)傳遞參數(shù)的具體形式,可選的兩種方式是METH_VARARGS和METH_KEYWORDS,其中METH_VARARGS是參數(shù)傳遞的標(biāo)準形式,它通過Python的元組在Python解釋器和C函數(shù)之間傳遞參數(shù),若采用METH_KEYWORD方式,則Python解釋器和C函數(shù)之間將通過Python的字典類型在兩者之間進行參數(shù)傳遞。

            3.4 初始化函數(shù)

            所有的Python擴展模塊都必須要有一個初始化函數(shù),以便Python解釋器能夠?qū)δK進行正確的初始化。Python解釋器規(guī)定所有的初始化函數(shù)的函數(shù)名都必須以init開頭,并加上模塊的名字。對于模塊example來說,則相應(yīng)的初始化函數(shù)為:

            void initexample() 
            {
              PyObject* m;
              m = Py_InitModule("example", exampleMethods);
            }
            

            當(dāng)Python解釋器需要導(dǎo)入該模塊時,將根據(jù)該模塊的名稱查找相應(yīng)的初始化函數(shù),一旦找到則調(diào)用該函數(shù)進行相應(yīng)的初始化工作,初始化函數(shù)則通過調(diào)用Python的C語言擴展接口所提供的函數(shù)Py_InitModule(),來向Python解釋器注冊該模塊中所有可以用到的方法。

            3.5 編譯鏈接

            要在Python解釋器中使用C語言編寫的擴展模塊,必須將其編譯成動態(tài)鏈接庫的形式。下面以RedHat Linux 8.0為例,介紹如何將C編寫的Python擴展模塊編譯成動態(tài)鏈接庫:

            [xiaowp@gary code]$ gcc -fpic -c -I/usr/include/python2.2 \
                                -I /usr/lib/python2.2/config \
                                example.c wrapper.c
            [xiaowp@gary code]$ gcc -shared -o example.so example.o wrapper.o
            

            3.6 引入Python解釋器

            當(dāng)生成Python擴展模塊的動態(tài)鏈接庫后,就可以在Python解釋器中使用該擴展模塊了,與Python自帶的模塊一樣,擴展模塊也是通過import命令引入后再使用的,如下所示:

            [xiaowp@gary code]$ python
            Python 2.2.1 (#1, Aug 30 2002, 12:15:30)
            [GCC 3.2 20020822 (Red Hat Linux Rawhide 3.2-4)] on linux2
            Type "help", "copyright", "credits" or "license" for more information.
            >>> import example
            >>> example.fact(4)
            24
            >>>
            





            回頁首


            四、結(jié)束語

            作為一門功能強大的腳本語言,Python將被更加廣泛地應(yīng)用于各個領(lǐng)域。為了克服腳本語言執(zhí)行速度慢的問題,Python提供了相應(yīng)的C語言擴展接口,通過將影響執(zhí)行性能的關(guān)鍵代碼用C語言實現(xiàn),可以很大程度上提高用Python編寫的腳本在運行時的速度,從而滿足實際需要。



            參考資料

            1. 可以從Python( http://www.python.org)網(wǎng)站著手了解所有關(guān)于Python的內(nèi)容。
            2. 可以在Python網(wǎng)站上找到正式的Python C/API文檔( http://www.python.org/doc/current/api/api.html)。
            3. 可以在Python網(wǎng)站上找到正式的編寫Python擴展模塊的文檔( http://www.python.org/doc/current/api/api.html)。


            關(guān)于作者

            肖文鵬,是北京理工大學(xué)計算機系的一名碩士研究生,主要從事操作系統(tǒng)和分布式計算環(huán)境的研究,喜愛Linux和Python。你可以通過 xiaowp@263.net與他取得聯(lián)系。

            久久性精品| 亚洲国产综合久久天堂 | 久久久这里有精品中文字幕| 久久夜色精品国产噜噜亚洲AV| 老司机午夜网站国内精品久久久久久久久| 精品久久久久久无码专区不卡| 久久99精品久久久大学生| 一级做a爰片久久毛片看看| 亚洲国产成人精品91久久久| 久久一区二区三区99| 色悠久久久久久久综合网| 久久精品国产第一区二区| 精品无码久久久久久国产| 久久99精品久久久久久9蜜桃| 精品久久久久久99人妻| 久久久久国产成人精品亚洲午夜| 9999国产精品欧美久久久久久 | 国产精品久久久久久| 国产精品九九九久久九九| 国产精品久久久久久久久鸭 | 天天久久狠狠色综合| 99久久精品免费看国产| 久久www免费人成看国产片| 久久综合视频网站| 色狠狠久久综合网| 亚洲精品无码久久久久| 国产产无码乱码精品久久鸭| 嫩草影院久久99| 久久亚洲2019中文字幕| 欧美精品乱码99久久蜜桃| 久久亚洲日韩精品一区二区三区| 久久ZYZ资源站无码中文动漫| 色噜噜狠狠先锋影音久久| 美女久久久久久| 精品久久久久久久无码| 精品久久久久久国产三级| 久久精品国产99国产精品亚洲 | 亚洲午夜精品久久久久久浪潮| 久久人人爽人人爽人人片av高请 | 精品综合久久久久久88小说 | 久久精品天天中文字幕人妻|