• <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>

            飯中淹的避難所~~~~~

            偶爾來避難的地方~

              C++博客 :: 首頁 :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理 ::
              94 隨筆 :: 0 文章 :: 257 評論 :: 0 Trackbacks
            項目里需要使用腳本,我是框架的維護人員,就需要把腳本系統(tǒng)加入到框架里。
            因為對PYTHON比較熟悉,所以最后選擇了PYTHON作為腳本系統(tǒng)的主引擎。
            下面是在這個過程中遇到的種種問題以及最后的解決方法。

            1- 編譯出錯,提示找不到 __imp_Py_XXXX的引用
               這個問題真的很惡心,WIN版的PYTHON27.DLL用的是單線DLL的CRT,而項目如果用的是多線LIB的CRT,或者其他不兼容的CRT,就會出這個問題。
               果斷下載PYTHON2.7.2的源代碼,編譯之。然而,編譯過程也非常的惡心,工程屬性修改成MTd和MT仍舊不行。
               后看到編譯時有調(diào)用cl的命令行,找了半天在make_buildinfo這個項目里發(fā)現(xiàn)是使用代碼生成的編譯參數(shù),里面寫死的是MDd,修改之再編譯,問題解決。

            2- Py_Initialize 退出,提示無法找到site模塊。
                這個問題原本想暴力的修改源代碼里的nosite標記改之,后思考可能這個模塊是有用的,所以把運行過PYTHON.EXE的PYTHON運行路徑里的LIB里的被編譯成PYC的所有PY按路徑復(fù)制到EXE的路徑下,問題解決。

            3- 不想用BOOST.PYTHON,又想不用寫PyCFunction的形式的C模塊方法
                這個問題經(jīng)過分析,得到結(jié)論是,從任意形式的用戶函數(shù)生成一個PyObject*(*PYFUNC)(PyObject*,PyObject*)形式的函數(shù)封裝。并且對于一個獨立地址的用戶函數(shù),需要一個獨立地址的函數(shù)封裝。
                首先,我開始解決如何從一個固定函數(shù)得到另一個固定地址函數(shù)。一開始我想到用函數(shù)地址作為模版參數(shù),這樣每個獨立地址就能生成一個單獨的模版類,然后模版類里的靜態(tài)函數(shù)自然就是獨立地址的。后我使用了static的本地函數(shù)測試,未果,提示什么non-extra的錯誤。后進群詢問,得到同樣答案,并且編譯成功的結(jié)果。仔細觀察后發(fā)現(xiàn),他用的是非static修飾的函數(shù)。我去掉static修飾,竟然成了。
                接下來,我開始解決如何從PyObject*args解析出用戶函數(shù)的每個參數(shù)的問題。這個問題有幾個部分,第一部分是如何從函數(shù)里析出每個參數(shù),這部分我用了模版的某種特化,具體名字我不懂,就是如下面這種形式:
               
            template <typename TR, typename T1 = void, typename T2 = void>
            struct st_func {};
            template 
            <typename TR, typename T1>
            struct st_func<TR, T1, void> {};

                這樣就可以析出返回值TR和各個參數(shù)的類型。為了支持盡可能多的參數(shù),我寫了個程序生成了從0個參數(shù)到40個參數(shù)的模版變體。
                使用固定地址,因為函數(shù)定義需要TR,T1,所以函數(shù)指針無法直接在第一個模版參數(shù)傳遞,并且是固定參數(shù),無法放在可選參數(shù)后面,所以我選擇了先在第一個參數(shù)用void*傳遞函數(shù)指針,然后在st_func內(nèi)部用TR,T1...這些拼出一個函數(shù)指針,把首位置的指針強轉(zhuǎn)成原函數(shù)形式,再進行調(diào)用。
                因為PYTHON的解析函數(shù)需要每個參數(shù)的類型標識符,大部分類型是一個字符表示,這部分用了enum來為每個類型生成一個const且static的字符標識。比如
            template <> struct type_char<int> { enum{ typechar = 'i' };};
                然后最終PyArg_ParseTuple 需要的是一個字符串,那么我在代碼里就 這樣來生成這個字符串:
               
            char szTypes[] = { type_char<T1>::typechar, .., 0 };
                最后一個部分是解決TR是void時的函數(shù)返回值問題。后來我用了struct內(nèi)部的特化函數(shù)來解決。最終形式如下:


            template <void* FP, typename TR, typename T1>
            struct st_cppfunc_to_py<FP, TR, T1, void, void>
            {
            typedef TR (
            *TFP)(T1);
            static PyObject * func( PyObject * self, PyObject * args ) { return _func<TR>( self, args ); }
            template
            <typename ITR>
            static PyObject * _func( PyObject * self, PyObject * args ) {
            char szType[] = { type_char<T1>::typechar, 0 };
            T1 v1;
            if( PyArg_ParseTuple( args, szType, &v1 ) ) {
            TR ret
            = ((TFP)FP)( v1 );
            char szRetType[2] = { type_char<TR>::typechar, 0 };
            return Py_BuildValue( szRetType, ret );
            }
            Py_RETURN_NONE;
            }
            template
            <>
            static PyObject * _func<void>( PyObject * self, PyObject * args ) {
            char szType[] = { type_char<T1>::typechar, 0 };
            T1 v1;
            if( PyArg_ParseTuple( args, szType, &v1 ) ) {
            ((TFP)FP)( v1 );
            }
            Py_RETURN_NONE;
            }

            };

            外面包一個  template <void*FP, typename TR, typename T1> PyCFunction __cppfunc2py( TR(*RFP)(T1) ) { return st_cppfunc_to_py<FP, TR, T1>::func; }

            就可以很方便的生成嵌入PY的函數(shù)了。

            4- 包裝的scriptvalue怎么獲得PyObject*的類型呢
                用Py_TYPE(ob)就可以獲取到object的typeobject,它的tp_name就是它的名字,常用類型 int, long, float, str 都可以分辨出來。

            5- PYTHON的調(diào)試版總是報GC異常
                這個問題我是嘗試著來解決的,總結(jié)了以下幾點:
                a. 模塊的DICT是不用PY_XDECREF來釋放的   
                b. 返回值需要一個PY_XDECREF釋放。
                c. Py_BuildValue 返回值需要一個PY_XDECREF。
                d. 用戶模塊不需要 PY_XDECREF。
                e. PY文件生成的模塊需要一個 PY_XDECREF。
                f. Set Object到Tuple去調(diào)完P(guān)Y的函數(shù),PY_XDECREF(TUPLE)時,需要注意的是,Set進去的Object都會被調(diào)用一次Py_XDECREF,所以一個好的辦法是在Set進Tuple時,就INC一下他們的REF。
            6- 為何一直調(diào)不到py文件里的函數(shù)
                這個問題困擾了我?guī)追昼姡K的method獲取不到,讓我一度以為是腳本寫的問題。
                后來我打印出來腳本的搜索路徑(print sys.path),是一大堆的路徑,我放進去的路徑排在最后。于是我想,是不是有的路徑下有重名的PY文件,就給文件改了個名字,結(jié)果就OK了。
                這個問題,我后來想可以通過調(diào)整搜索優(yōu)先級來解決,不過目前還是這樣解決比較好,因為調(diào)整方法目前還不得而知。

               
            posted on 2012-02-10 21:55 飯中淹 閱讀(3264) 評論(0)  編輯 收藏 引用

            只有注冊用戶登錄后才能發(fā)表評論。
            網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


            亚洲精品无码专区久久同性男| 久久精品国产亚洲精品| 久久久久久久免费视频| 一本色综合久久| 久久久久久夜精品精品免费啦| 久久天天躁狠狠躁夜夜躁2O2O| 久久香蕉国产线看观看乱码| 欧美久久久久久午夜精品| 久久久精品2019免费观看| 久久精品国产精品亚洲艾草网美妙| 久久午夜夜伦鲁鲁片免费无码影视 | 亚洲狠狠婷婷综合久久久久| 亚洲精品国精品久久99热一| 国产农村妇女毛片精品久久| 久久WWW免费人成一看片| 亚洲一区二区三区日本久久九| 久久99久久99精品免视看动漫| 9999国产精品欧美久久久久久| 久久人妻AV中文字幕| 久久e热在这里只有国产中文精品99 | 国产亚州精品女人久久久久久 | 999久久久国产精品| 亚洲日韩中文无码久久| 深夜久久AAAAA级毛片免费看| 久久线看观看精品香蕉国产| 久久综合给合久久狠狠狠97色69| 午夜精品久久久内射近拍高清| 色偷偷888欧美精品久久久| 久久精品国产亚洲精品2020 | 久久99精品久久久久久噜噜| 1000部精品久久久久久久久| 亚洲AV日韩精品久久久久| 久久中文字幕人妻丝袜| 伊人久久大香线蕉成人| 伊人久久大香线蕉无码麻豆| 亚洲精品成人网久久久久久| 久久人人爽人人澡人人高潮AV| 国产午夜精品久久久久九九| 91麻精品国产91久久久久| 国产真实乱对白精彩久久| 国产综合免费精品久久久|