Re: [問題] 讓Python 用C運算 之 傳變數的問題

看板Python作者 (阿真)時間16年前 (2009/06/07 00:11), 編輯推噓1(102)
留言3則, 2人參與, 最新討論串3/5 (看更多)
※ 引述《sbrhsieh (sbr)》之銘言: : ※ 引述《KSJ (阿真)》之銘言: : : 但是 我認為這個"長的程式" 應該也可以在C中使用吧??(這部份想請教) : : 也就是說: : : 在C寫一個py函式 : : static PyObject *funcA (PyObject *self, PyObject *args) : : { : : (把args做一些修改變成args2)//一段程式碼 : /* 在這裡插入對 standard output 輸出一些訊息的碼 */ : : PyObject *value = funcB(NULL,args2); //另一個寫好的funcB : : return Py_BuildValue("N",value); : : } : : 不知道這樣行不行得通 : : 裡面的funcB就是我的"長的程式" : : 單單使用funcB是沒問題的 : : 而要使用funcA傳變數給funcB使用 就會有"誤" : : "誤"是指 程式爆掉 不是例外 是XP不回報的那種 : 感覺上很有可能在你沒有揭露的"把args做一些修改變成args2"這一段碼就錯了, : 搞到程式死掉。先弄清楚到底是死在哪一部份。 首先想先確認 因為沒有懷疑"所寫的函式 可以被呼叫"這件事 那這樣的動作應該是可行了(我也希望是這樣) 我來仔細說明我的做法 首先 python的函式 只包含2個變數要傳入 一個是self,另一個是args self我讀到認為是只用在method 用function時self是一個指向NULL的指標 args則想成是所有需要的變數包裝 我常用一個list或tuple把需要的變數放在裡面 所以說 funcA 與 funcB 他們接受的變數 可以想成同一類 list or tuple 傳入後 再去考慮要怎麼接list裡的一個個變數 我的funcB 我是用以下的方式去接的(變數當成已宣告): if (!PyArg_ParseTuple(args, "(OOOOI)", \ &pylagS,&pylagT,&pyCOV,&pyFitVar,&func_num)) return NULL; 事實上我傳進去的是一個長得像這樣 ([1,1,2,2],[1,2,1,2],[5,4,6,3],(4,3),0) 的一個tuple 在只使用funcB (也就是以上是在python的idle用手打的)時 是可以跑完的 所以我認定 只要在c中 創造出這樣一個tuple 傳進funcB中 應該也是可行 創造tuple的工作 就必須在funcA中完成 也就是先前所提的 把args轉為args2 //我必須先把funcA的變數轉成List 才能使用內建的insert()方法 PyObject *pyFUNC_VARIABLEseqList = PySequence_List(pyFUNC_VARIABLEseq) //經過轉換 輸進funcA的args應該長 //[[1,1,2,2],[1,2,1,2],[5,4,6,3], 0] //本來的[4,3] 必須由funcA中新開的變數來做設定 //開一個tuple ( 也就是(4,3) 只是裡面的數字會變) 準備讓args insert 它 PyObject *pyVarTuple = PyTuple_New(Nseqlength); //放變數 想成 Nseqlength 是2就好 (測試也是用2 雖然我是寫n的) for (j=0;j<Nseqlength;j++) { //index的轉換 好像32位元的沒差 但64的會有 我就先轉了 PyObject *pyj = PyInt_FromLong(j); Py_ssize_t pyj_index = PyInt_AsSsize_t(pyj); Py_DECREF(pyj); PyTuple_SET_ITEM \ (pyVarTuple, pyj_index, PyFloat_FromDouble(Location[i*Nseqlength+j])); } Py_ssize_t index = 3; //insert的位置 這個可以確定是3 //也就是(4,3)的位置 //做insert的動作 PyList_Insert(pyFUNC_VARIABLEseqList,index,pyVarTuple); //將insert後的list (也就是args2) 轉成tuple //主要是運用會產生new reference的特性 產生所謂真正的args2 PyObject *pyFUNC_VARIABLEseqTuple =\ PySequence_Tuple(pyFUNC_VARIABLEseqList); //要的東西(args2)有了 剩的(args、插入用的tuple)就DECREF掉 Py_DECREF(pyFUNC_VARIABLEseqList); Py_DECREF(pyVarTuple); //下面這行本來是看看傳進下面funcB之前 這東西倒底有沒有問題的 //return Py_BuildValue("N",pyFUNC_VARIABLEseqTuple); //結果是 //([1, 1, 2, 2], [1, 2, 1, 2], [5, 4, 6, 3], (8.72, 5.52), 0) //所以我認定應該是沒問題 才繼續下面這一行 PyObject *value = (*func)(NULL,pyFUNC_VARIABLEseqTuple); 於是在這裡的最後一行 程式有誤 想必是進入funcB之後的問題?! 於是我在funcB之中加入一些中斷點 來看看是哪裡的問題 funcB只秀到有誤的部份: //宣告要接的變數 有一個長整數 四個pyobejct //長整數就是 最後的那個 0 四個pyobject就是 //[1,1,2,2,] [1,2,1,2] [5,4,6,3] (8.72,5.52) 這四個 long int func_num; PyObject *pylagS,*pylagT,*pyCOV,*pyFitVar; PySys_WriteStdout("parse OK");//中斷點1 //傳遞參數 if (!PyArg_ParseTuple(args, "(OOOOI)",\ &pylagS,&pylagT,&pyCOV,&pyFitVar,&func_num)) return NULL; PySys_WriteStdout("parse OK");//中斷點2 到中斷點1 印出parse OK 之後 程式就爆了…orz 強調一下 如果單單只叫funcB (用手key值) 是沒問題的 但由funcA 叫funcB 就爆了(如上…) 傳變數看起來也ok… 所以…請大家幫個忙 囧 感激不盡 <(_ _)> -- ※ 發信站: 批踢踢實業坊(ptt.cc) ◆ From: 140.112.63.180 ※ 編輯: KSJ 來自: 140.112.63.180 (06/07 00:22)

06/07 16:55, , 1F
看不懂 ... 整顆包起來給個下載連結, 我有看到再 de 看看
06/07 16:55, 1F

06/08 14:03, , 2F
http://file2.ws/python3 下載包 裡面有cpp跟pyd檔案
06/08 14:03, 2F

06/08 14:10, , 3F
裡面用到的是 WMSE 跟 PSO PSO呼叫WMSE時有誤
06/08 14:10, 3F
文章代碼(AID): #1AAfL88I (Python)
討論串 (同標題文章)
文章代碼(AID): #1AAfL88I (Python)