這次主要講講怎么把python中的class嵌入到c++中去。
順便講講元組的操作和怎么編譯python源代碼。
1. 首先講講元組的操作
由于參數(shù)是通過元組傳進(jìn)去的,所以我們不能老是通過Py_BuildValue這個(gè)函數(shù)來(lái)操作元組,那樣太不方便了。
Python提供了元組相關(guān)的操作,下面這個(gè)例子演示了如何操作。主要是下面幾個(gè)函數(shù):
//new一個(gè)元組,傳入size
pArgs = PyTuple_New(argc - 3);
//set元組的直,第一個(gè)為元組,第二個(gè)為index(從0開始),第三個(gè)為value
PyTuple_SetItem(pArgs,0,Py_BuildValue("i",2000) );
PyTuple_SetItem(pArgs,1,Py_BuildValue("i",8) );
來(lái)自python doc的一個(gè)例子
#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中去。定義了一個(gè)很簡(jiǎn)單的類,有一個(gè)name成員變量,一個(gè)printName成員函數(shù)
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");//得到那個(gè)類
pArg = PyTuple_New(1);
PyTuple_SetItem(pArg, 0, Py_BuildValue("s", "Jacky"));
pObject = PyEval_CallObject(pClass, pArg);//生成一個(gè)對(duì)象,或者叫作實(shí)例
pFunc = PyObject_GetAttrString(pObject, "printName");//得到該實(shí)例的成員函數(shù)
PyEval_CallObject(pFunc, NULL);//執(zhí)行該實(shí)例的成員函數(shù)
Py_Finalize();
return 0;
}
沒有什么資料,就先寫到這里了。下面介紹一下怎么build python25的源代碼
3. 編譯python源代碼
為什么要編譯呢?因?yàn)闆]有python25_d.lib!呵呵。順便可以了解一下代碼結(jié)構(gòu)。
解壓縮后,有好多目錄,其中pcbuild和pcbuild8是我們要的。pcbuild對(duì)應(yīng)著vc7.1的,pcbuild8對(duì)應(yīng)著vc8.0的
因?yàn)樵谟胿c7.1,也就是2003了。所以我就說(shuō)說(shuō)怎么用2003來(lái)編譯吧。事實(shí)上是從一位牛人那里學(xué)來(lái)的
http://blog.donews.com/lemur/archive/2005/12/17/660973.aspx,那位大哥大概一年半前就在解剖python了,厲害
阿。看來(lái)我只能后來(lái)居上了,娃哈哈。我按照他說(shuō)的試了一下,編譯成功!
不過遇到一點(diǎn)小問題,用vc2003打開那個(gè)solution的時(shí)候,發(fā)現(xiàn)作者沒有把source code control去掉,郁悶!害的我
們打開的時(shí)候一堆messagebox。不過不用管它就好了,一直確定。最后試了一下那個(gè)python25_d.lib,沒問題。不過記
得把python25_d.dll copy到一個(gè)能被找到的目錄,比如說(shuō)c:\windows\system32\下面。python25.dll也在這個(gè)目錄下
面。over。恩。