1. 程式人生 > >c++中嵌入python入門3

c++中嵌入python入門3

這次主要講講怎麼把python中的class嵌入到c++中去。
順便講講元組的操作和怎麼編譯python原始碼。

1. 首先講講元組的操作
由於引數是通過元組傳進去的,所以我們不能老是通過Py_BuildValue這個函式來操作元組,那樣太不方便了。
Python提供了元組相關的操作,下面這個例子演示瞭如何操作。主要是下面幾個函式:
//new一個元組,傳入size
pArgs = PyTuple_New(argc - 3); 
//set元組的直,第一個為元組,第二個為index(從0開始),第三個為value
PyTuple_SetItem(pArgs,0,Py_BuildValue("i",2000) );
PyTuple_SetItem(pArgs,1,Py_BuildValue("i",8) );

來自python doc的一個例子

#include <Python.h>
int
main(int argc, char *argv[])
{
    PyObject *pName, *pModule, *pDict, *pFunc;
    PyObject *pArgs, *pValue;
    int i;

    if (argc < 3) {
        fprintf(stderr,"Usage: call pythonfile funcname [args]\n");
        return 1;
    }

    Py_Initialize();
    pName = PyString_FromString(argv[1]);
    /* Error checking of pName left out */

    pModule = PyImport_Import(pName);
    Py_DECREF(pName);

    if (pModule != NULL) {
        pFunc = PyObject_GetAttrString(pModule, argv[2]);
        /* pFunc is a new reference */

        if (pFunc && PyCallable_Check(pFunc)) {
            pArgs = PyTuple_New(argc - 3);
            for (i = 0; i < argc - 3; ++i) {
                pValue = PyInt_FromLong(atoi(argv[i + 3]));
                if (!pValue) {
                    Py_DECREF(pArgs);
                    Py_DECREF(pModule);
                    fprintf(stderr, "Cannot convert argument\n");
                    return 1;
                }
                /* pValue reference stolen here: */
                PyTuple_SetItem(pArgs, i, pValue);
            }
            pValue = PyObject_CallObject(pFunc, pArgs);
            Py_DECREF(pArgs);
            if (pValue != NULL) {
                printf("Result of call: %ld\n", PyInt_AsLong(pValue));
                Py_DECREF(pValue);
            }
            else {
                Py_DECREF(pFunc);
                Py_DECREF(pModule);
                PyErr_Print();
                fprintf(stderr,"Call failed\n");
                return 1;
            }
        }
        else {
            if (PyErr_Occurred())
                PyErr_Print();
            fprintf(stderr, "Cannot find function \"%s\"\n", argv[2]);
        }
        Py_XDECREF(pFunc);
        Py_DECREF(pModule);
    }
    else {
        PyErr_Print();
        fprintf(stderr, "Failed to load \"%s\"\n", argv[1]);
        return 1;
    }
    Py_Finalize();
    return 0;
}


2. class操作
把下面加入到test2.py中去。定義了一個很簡單的類,有一個name成員變數,一個printName成員函式
class TestClass:
    def __init__(self,name):
        self.name = name

    def printName(self):
        print self.name

cpp檔案
#include <python.h>
int main()
{
 Py_Initialize();

 PyObject * pModule = NULL;
 PyObject * pFunc   = NULL;
 PyObject * pArg    = NULL;
 PyObject * pClass  = NULL;
 PyObject * pObject = NULL;

 pModule = PyImport_ImportModule("test2");
 pClass  = PyObject_GetAttrString(pModule, "TestClass");//得到那個類
 pArg = PyTuple_New(1);
 PyTuple_SetItem(pArg, 0, Py_BuildValue("s", "Jacky"));
 pObject = PyEval_CallObject(pClass, pArg);//生成一個物件,或者叫作例項

 pFunc = PyObject_GetAttrString(pObject, "printName");//得到該例項的成員函式
 PyEval_CallObject(pFunc, NULL);//執行該例項的成員函式

 Py_Finalize();

 return 0;
}


沒有什麼資料,就先寫到這裡了。下面介紹一下怎麼build python25的原始碼

3. 編譯python原始碼
為什麼要編譯呢?因為沒有python25_d.lib!呵呵。順便可以瞭解一下程式碼結構。
解壓縮後,有好多目錄,其中pcbuild和pcbuild8是我們要的。pcbuild對應著vc7.1的,pcbuild8對應著vc8.0的
因為在用vc7.1,也就是2003了。所以我就說說怎麼用2003來編譯吧。事實上是從一位牛人那裡學來的

http://blog.donews.com/lemur/archive/2005/12/17/660973.aspx,那位大哥大概一年半前就在解剖python了,厲害

阿。看來我只能後來居上了,娃哈哈。我按照他說的試了一下,編譯成功!

不過遇到一點小問題,用vc2003開啟那個solution的時候,發現作者沒有把source code control去掉,鬱悶!害的我

們開啟的時候一堆messagebox。不過不用管它就好了,一直確定。最後試了一下那個python25_d.lib,沒問題。不過記

得把python25_d.dll copy到一個能被找到的目錄,比如說c:\windows\system32\下面。python25.dll也在這個目錄下

面。over。恩。