作者sbrhsieh (sbr)
看板Python
标题Re: [问题] 让Python 用C运算 之 传变数的问题
时间Mon Jun 8 16:35:08 2009
※ 引述《KSJ (阿真)》之铭言:
: //下面这行本来是看看传进下面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值) 是没问题的
我後来仔细看看你这一段描述,也对照过你提供的完整 source code,确定你在
描述时没有笔误。我想我应该找到问题点,但不确定这是否就是你欠缺的观念。
首先 native function WMSE 一开始是如下 parse 传进来的参数:
if (!PyArg_ParseTuple(args, "(OOOOI)",
&pylagS,&pylagT,&pyCOV,&pyFitVar,&func_num)) return NULL;
这表示 WMSE export 到 Python module 後被使用时预期 caller pass 单一个
tuple object,而此 tuple 依序装着 4 个 object 与 1 个 int。
# in Python
SEKSGUI.WMSE((a, b, c, e, 0))
但是在 native function: PSO 中,程式准备好一个 tuple object(referenced
by 'pyFUNC_VARIABLEseqTuple'),把这个 tuple 输出到 standard output 是
([1, 1, 2, 2], [1, 2, 1, 2], [5, 4, 6, 3], (8.72, 5.52), 0)
这个 tuple 也是依序持有 4 个 object 与 1 个 int object。
然而你不能把这个 tuple(address) 直接 pass 给 WMSE function,而必须要
把这个 tuple 装进另一个 tuple(成为唯一的 element)後,再把 enclosing
tuple pass 给 WMSE function,也就是说要 pass 如下的 object:
(([1, 1, 2, 2], [1, 2, 1, 2], [5, 4, 6, 3], (8.72, 5.52), 0))
以目前你的程式里的做法:
PyObject *value = (*func)(NULL,pyFUNC_VARIABLEseqTuple);
这相当於在 Python 端执行这样的 statement:
SEKSGUI.WMSE([1, 1, 2, 2], [1, 2, 1, 2], [5, 4, 6, 3], (8.72, 5.52), 0)
pass 了五个参数而非 pass 一个 tuple 当作参数。
-------------------------------------------------------------------------
以 C 语言来实作 Python function 时,native function 有以下形式的 prototype:
static PyObject *
function_name(PyObject *self, PyObject *args);
static PyObject *
function_name(PyObject *self, PyObject *args, PyObject *kw);
是分别对应到 Python 中的:
def function_name(*args): ...
def function_name(*args, **kw): ...
也就是说不管在 Python 端 pass 几个参数给 native function,native function
都是收到一个 tuple(此 tuple 依序装着 pass 过来的参数),反过来说 native
function 的 formal parameter 'args' 收到一个 tuple 不代表 Python 端明白
pass 一个 tuple 当作 actua argument。
我认为当初你本来是打算把 WMSE 设计成如下使用:
SEKSGUI.WMSE(aObj, bObj, cObj, dObj, aInt)
那麽 WMSE 在 parse 参数时使用 "OOOOI" 当作 format 即可。native function
WMSE 的 formal parameter 'args' 会是收到一个 tuple:
(aObj, bObj, cObj, dObj, aInt)
--
※ 发信站: 批踢踢实业坊(ptt.cc)
◆ From: 218.173.129.26
※ 编辑: sbrhsieh 来自: 218.173.129.26 (06/08 16:45)
※ 编辑: sbrhsieh 来自: 218.173.129.26 (06/08 16:48)