这是一个简化的C脚本:
#include <stdlib.h> void cprogram(double *wts,double *res,int *kks,int n,double *ex) { int m; m=n+1; res[0]=1.0; kks[0]=1.0;}
这是我简化的Python代码:
from ctypes import * import sys libc = CDLL("src/program.so") class CONTEXT(Structure): _fields_ = [ ("wts",POINTER(c_double)),//tried just to see if it would work ("res",c_double),("kks",c_int),("n",("ex",c_double)] def runner(): kk = (1,2,3) n = 3 mm = n + 1 wts = (c_double * n)(1,1,1) res = (c_double * mm)(0) kks = (c_int * len(kk))(*kk) n = c_int(n) ex = c_double(0) libc.cprogram.restype = POINTER(CONTEXT) tmp = libc.cprogram(wts,res,kks,n,ex) runner()
我尝试过像print tmp [1] .wts [1]和print tmp [2]这样的命令,但是这只打印内存地址而不是值(或者像2.15880221124e-314那样不可思议的小值).我希望能够返回wts值的列表.
解决方法
最重要的是,即使它确实通过引用返回了一个CONTEXT对象,tmp [1]也会尝试在该对象之后对内存进行deref,因此它仍然是垃圾.
当你试图将随机内存解释为double时,如果你运气好的话,你会得到像2.15880221124e-314这样的段错误或值 – 如果你运气不好,你会得到正确的但仍然是随机的值,如0.0 .
同时,由于你的C函数修改了它的参数,你不需要在这里做任何花哨的事情.只需使用您传入的变量即可.
所以:
def runner(): kk = (1,1) res = (c_double * mm)(0) kks = (c_int * len(kk))(*kk) n = c_int(n) ex = c_double(0) libc.cprogram.restype = None libc.cprogram(wts,ex) print wts[1]
这样可以打印出1.0.
如果您的C函数确实返回了与您的ctypes声明匹配的CONTEXT结构数组,那么这一切都可以正常工作.例如:
#include <stdlib.h> typedef struct { double *wts; double res; int kks; int n; double ex; } CONTEXT; CONTEXT *cprogram(double *wts,double *ex) { int m; m=n+1; res[0]=1.0; kks[0]=1.0; CONTEXT *contexts = malloc(sizeof(CONTEXT) * 4); for (int i=0; i!=4; ++i) { double *wtsses = malloc(sizeof(double) * 5); for (int j=0; j!=4; ++j) { wtsses[j] = i + j; } CONTEXT context = { wtsses,*res,*kks,m,*ex }; contexts[i] = context; } return contexts; }
编译它,使用添加的打印tmp [1] .wts [1]运行现有的Python脚本,它将打印出2.0.
最后,对于你的后续工作,首先让我们改变C代码以获取int * n,其中* n是输入:
void cprogram(double *wts,int *n,double *ex) { int m; m=*n+1; res[0]=1.0; kks[0]=1.0;}
现在,要从Python调用它,您必须创建一个c_int并传递指向它的指针.因为你已经在前半部分做了(之前没有必要 – 只需设置argtypes ……但现在这是必要的,这很方便),它只是一个单行改变:
libc.cprogram(wts,pointer(n),ex)
如果n是输入输出参数,这甚至可以工作.
但实际上,您根本不需要从Python中看到指针对象;你正在做的唯一事情就是创建它以传递给函数,然后让它被收集.要在不创建ctypes指针对象的情况下传递C指针(即使n是输入输出参数,它也会起作用),请使用byref:
libc.cprogram(wts,byref(n),ex)
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。