Re: [問題] 讓Python 用C運算 之 傳變數的問題
※ 引述《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
06/07 16:55, 1F
→
06/08 14:03, , 2F
06/08 14:03, 2F
→
06/08 14:10, , 3F
06/08 14:10, 3F
討論串 (同標題文章)
Python 近期熱門文章
PTT數位生活區 即時熱門文章