首页 > 解决方案 > 来自 python 中的 c 函数的错误返回值

问题描述

这很奇怪,我试图包装一个我用 c 编写的函数以便在 python 中使用它,为了理解如何做到这一点,我做了一个例子:

cfunctions.c

double dotproduct(int dim, double a[dim], double b[dim]){
    double res = 0;
    for(int i = 0; i < dim; i++){
        res = res + a[i]*b[i];
    }
    return res;
}

py_cfunctions.py

from ctypes import *
import numpy as np

# so_file genreated with:
# cc -fPIC -shared -o cfunctions.so cfunctions.c
so_file = 'MY_PATH/cfunctions.so'
py_cfunctions = CDLL(so_file)
c_double_p = POINTER(c_double)
# Preparing example

def dotproduct(n,a,b):
    # Convert np.array to ctype doubles
    a_data = a.astype(np.double)
    a_data_p = a_data.ctypes.data_as(c_double_p)
    b_data = b.astype(np.double)
    b_data_p = b_data.ctypes.data_as(c_double_p)
    # Compute result...
    c_return = py_cfunctions.dotproduct(n,a_data_p,b_data_p)
    if (c_return != -1):
        return c_return
    else:
        return "C Function failed, check inputs"
 
py_cfunctions.dotproduct.argtypes= [c_int, c_double_p, c_double_p] 

有趣的是,代码似乎可以工作,但返回的值总是等于n. 如果我在返回之前尝试打印res,我可以看到该值是正确的,但实际返回的值始终是n. 我错过了什么?

>>> import py_cfunctions as pycf
>>> import numpy as np
>>> a = np.array([1,2,3,4])
>>> b = np.array([4,5,6,1])
>>> pycf.dotproduct(4,a,b)
4

标签: pythoncnumpyctypescpython

解决方案


感谢@Mark Tolonen 的回答:

“在实现级别,整数在不同于浮点值的寄存器中返回。除非另有说明,否则 ctypes 假定 c_int 作为返回值。使用 ctypes 时,'显式优于隐式'。始终正确定义 .restype 和 .argtypes,您将有更少的问题。”

我必须明确定义我的函数的返回类型,并将以下行添加到pic_functions.py

py_cfunctions.dotproduct.restype = c_double


推荐阅读