想要在c++ 中嵌入script 代碼, 除了自己寫(xiě)腳本引擎外, lua, python 都可以在c++ 中使用, 另外 MonoBind, AngelScript library 都是一些c++ script library, 可以嵌入到c++ 中使用 .
今天在c++ 中試著嵌入 python 代碼 (示例代碼在 Python-2.5.2\Demo\embed\ 下)
#include <Python.h>
int main(int argc, char *argv[])
{
// Py_NoSiteFlag = 1;
// Py_SetPythonHome("D:\\usr\\Python"); // PYTHONHOME
Py_Initialize();
PyRun_SimpleString("from time import time,ctime\n"
"print 'Today is',ctime(time())\n");
Py_Finalize();
return 0;
}
在運(yùn)行時(shí)可能會(huì)產(chǎn)生類(lèi)似 'import site' failed; use -v for traceback 的錯(cuò)誤, 原因是python 在import module 的時(shí)候的路徑問(wèn)題. 有3種方法可以解決(以前通過(guò)設(shè)置環(huán)境變量 PYTHONPATH 好像在2.5 已經(jīng)無(wú)效了).
0. 取消注釋 Py_NoSiteFlag = 1;
這個(gè)只是取消import site , 當(dāng)然如果在代碼中要import 啥的話(huà), 還是會(huì)出現(xiàn)錯(cuò)誤的.
a. 設(shè)置環(huán)境變量 PYTHONHOME = D:\usr\Python
b. 在調(diào)用 Py_Initialize 之前調(diào)用函數(shù)
Py_SetPythonHome("D:\\usr\\Python"); // 參數(shù)是python 的安裝目錄
2. 其他一些有用的資源
Python/C API Reference Manual (API 參考) , Extending and Embedding the Python Interpreter (擴(kuò)展及嵌入Python解釋器, 主要說(shuō)明了如何擴(kuò)展Python, 給Python 寫(xiě)擴(kuò)展, 其中 5. Embedding Python in Another Application 一章講述了在C++中嵌入/調(diào)用Python 代碼 )
使用C/C++擴(kuò)展Python 對(duì)文 Extending and Embedding the Python Interpreter 作了精簡(jiǎn), 很不錯(cuò)的一篇文章, 但是跳過(guò)了一些基礎(chǔ) .
Building Hybrid Systems with Boost.Python 介紹了使用boost.python 方便python 插件開(kāi)發(fā), python綁定c++程序 是其中文版本.
Embedding Python in Multi-Threaded C/C++ Applications 講了c++在多線(xiàn)程環(huán)境如何使用Python , 文 C++多線(xiàn)程中調(diào)用python api函數(shù) 提供了一個(gè)多線(xiàn)程的封裝.
SCXX - A Simple Python/C++ API
http://davidf.sjsoft.com/mirrors/mcmillan-inc/scxx.html
C++擴(kuò)展和嵌入Python應(yīng)用 (介紹了一些Python/C API 函數(shù), 以及ext 例子, 一般般)
http://hi.baidu.com/yunsweet/blog/item/20b08aeebaa2b1282cf534c7.html
3. Python 多線(xiàn)程的使用
zz http://blog.csdn.net/liguangyi/archive/2007/06/20/1659697.aspx
今天看了近一天關(guān)于多線(xiàn)程的應(yīng)用中,如何安全調(diào)用python方面的資料,開(kāi)始的時(shí)候看的簡(jiǎn)直頭大如斗,被python語(yǔ)言的全局鎖(Global Interpreter Lock)、線(xiàn)程狀態(tài)(Thread State )等都有點(diǎn)繞暈了,后來(lái)經(jīng)過(guò)各方面文章和幫助文檔的相互參考,發(fā)現(xiàn)對(duì)于2.4/2.5版本,提供了PyGILState_Ensure, PyGILState_Release,哎,這下可方便大發(fā)了。
一、首先定義一個(gè)封裝類(lèi),主要是保證PyGILState_Ensure, PyGILState_Release配對(duì)使用,而且這個(gè)類(lèi)是可以嵌套使用的。
#include <python.h>
class PyThreadStateLock
{
public:
PyThreadStateLock(void)
{
state = PyGILState_Ensure( );
}
~PyThreadStateLock(void)
{
PyGILState_Release( state );
}
private:
PyGILState_STATE state;
};
二、在主線(xiàn)程中,這樣處理
// 初始化
Py_Initialize();
// 初始化線(xiàn)程支持
PyEval_InitThreads();
// 啟動(dòng)子線(xiàn)程前執(zhí)行,為了釋放PyEval_InitThreads獲得的全局鎖,否則子線(xiàn)程可能無(wú)法獲取到全局鎖。
PyEval_ReleaseThread(PyThreadState_Get());
// 其他的處理,如啟動(dòng)子線(xiàn)程等
......
// 保證子線(xiàn)程調(diào)用都結(jié)束后
PyGILState_Ensure();
Py_Finalize();
// 之后不能再調(diào)用任何python的API
三、在主線(xiàn)程,或者子線(xiàn)程中,調(diào)用python本身函數(shù)的都采用如下處理
{
class PyThreadStateLock PyThreadLock;
// 調(diào)用python的API函數(shù)處理
......
}
呵呵,看這樣是否非常簡(jiǎn)單了。
另外還有兩個(gè)和全局鎖有關(guān)的宏,Py_BEGIN_ALLOW_THREADS 和 Py_END_ALLOW_THREADS。這兩個(gè)宏是為了在較長(zhǎng)時(shí)間的C函數(shù)調(diào)用前,臨時(shí)釋放全局鎖,完成后重新獲取全局鎖,以避免阻塞其他python的線(xiàn)程繼續(xù)運(yùn)行。這兩個(gè)宏可以這樣調(diào)用
{
class PyThreadStateLock PyThreadLock;
// 調(diào)用python的API函數(shù)處理
......
Py_BEGIN_ALLOW_THREADS
// 調(diào)用需要長(zhǎng)時(shí)間的C函數(shù)
......
Py_END_ALLOW_THREADS
// 調(diào)用python的API函數(shù)處理
......
}
4. 可能的錯(cuò)誤及解決
a. 在vs 200x 下 debug 模式出現(xiàn)鏈接問(wèn)題
extmodule.obj : error LNK2019: unresolved external symbol __imp___Py_Dealloc referenced in function _PySwigObject_format
extmodule.obj : error LNK2019: unresolved external symbol __imp___Py_NegativeRefcount referenced in function _PySwigObject_format
extmodule.obj : error LNK2001: unresolved external symbol __imp___Py_RefTotal
extmodule.obj : error LNK2019: unresolved external symbol __imp___PyObject_DebugFree referenced in function _PySwigObject_dealloc
extmodule.obj : error LNK2019: unresolved external symbol __imp___PyObject_DebugMalloc referenced in function _PySwigObject_New
extmodule.obj : error LNK2019: unresolved external symbol __imp__Py_InitModule4TraceRefs referenced in function _init_extmodule
主要是因?yàn)?Py_DEBUG/Py_TRACE_REFS 引起, 修改 Python\include 下的 pyconfig.h, object.h 兩個(gè)文件就行了 ... 詳見(jiàn) http://www.nabble.com/link-error-in-debug-mode-td3126668.html