c 语言相应内存位置读取数据,使用ctypes访问内存地址处的数据

您的C函数返回void,而不是CONTEXT *.因此,您的代码只是将随机未初始化的内存转换为CONTEXT *,然后尝试解除它.

最重要的是,即使它确实通过引用返回了一个CONTEXT对象,tmp [1]也会尝试在该对象之后对内存进行deref,因此它仍然是垃圾.

当你试图将随机内存解释为double时,如果你运气好的话,你会得到像2.15880221124e-314这样的段错误或值 – 如果你运气不好,你会得到正确的但仍然是随机的值,如0.0 .

同时,由于你的C函数修改了它的参数,你不需要在这里做任何花哨的事情.只需使用您传入的变量即可.

所以:

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 = None

libc.cprogram(wts, res, kks, n, ex)

print wts[1]

这样可以打印出1.0.

如果您的C函数确实返回了与您的ctypes声明匹配的CONTEXT结构数组,那么这一切都可以正常工作.例如:

#include

typedef struct {

double *wts;

double res;

int kks;

int n;

double ex;

} CONTEXT;

CONTEXT *cprogram(double *wts, double *res, int *kks, int n, 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, double *res, int *kks, int *n, double *ex) {

int m;

m=*n+1;

res[0]=1.0;

kks[0]=1.0;}

现在,要从Python调用它,您必须创建一个c_int并传递指向它的指针.因为你已经在前半部分做了(之前没有必要 – 只需设置argtypes ……但现在这是必要的,这很方便),它只是一个单行改变:

libc.cprogram(wts, res, kks, pointer(n), ex)

如果n是输入输出参数,这甚至可以工作.

但实际上,您根本不需要从Python中看到指针对象;你正在做的唯一事情就是创建它以传递给函数,然后让它被收集.要在不创建ctypes指针对象的情况下传递C指针(即使n是输入输出参数,它也会起作用),请使用byref:

libc.cprogram(wts, res, kks, byref(n), ex)

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值