将PyArrayObject数据类型转换为C数组
我想在一个C扩展中使用我的Numpy数组。很多例子都用到了PyArrayObject这个结构。
array->data , array->strides[0] , array->strides[1] , ...
这些指针用来访问数据。如果我想用一种更熟悉(或者更整洁)的方式来访问我的数组,比如用下标
array[i][j]
我该怎么做呢?我是不是应该把(bool *)array->data转成我创建的C数组来操作?(我的元素是布尔值)
我现在的函数声明是(当然还没完成)
static PyObject *
xor_masking(PyObject *self, PyObject *args)
{
PyObject *input;
PyObject *mask;
PyObject *adjacency;
PyObject *state;
PyArrayObject *arr_mask;
PyArrayObject *arr_adjacency;
PyArrayObject *arr_state;
PyArrayObject *arr_next_state;
double sum;
int counter_node, n_nodes;
/* PyArg_ParseTuple
* checks if from args, the pointers of type "O" can be extracted, and extracts them
*/
if (!PyArg_ParseTuple(args, "OOO:xor_masking_C", &mask, &adjacency, &state))
return NULL;
/*
* The pointer returned by PyArray_ContiguousFromObject is typecasted to
* a PyArrayObject Pointer and array is pointed to the same address.
*/
arr_mask = (PyArrayObject *)
PyArray_ContiguousFromObject(mask, PyArray_BOOL, 2, 2);
arr_adjacency = (PyArrayObject *)
PyArray_ContiguousFromObject(adjacency, PyArray_BOOL, 2, 2);
arr_state = (PyArrayObject *)
PyArray_ContiguousFromObject(state, PyArray_BOOL, 2, 2);
if (array == NULL)
return NULL;
int n_mask_0 = mask->dimensions[0];
int n_mask_1 = mask->dimensions[1];
int n_adjacency_0 = adjacency->dimensions[0];
int n_adjacency_1 = adjacency->dimensions[1];
int n_state_0 = state->dimensions[0];
int n_nodes = n_state_0;
/*
* if the dimensions don't match, return NULL
*/
bool c_mask[n_nodes][n_nodes];
if (n_mask_0 != n_mask_1 || n_adjacency_0 != n_adjacency_1 ||
n_adjacency_0 != n_mask_0 || n_adjacency_0 != n_adjacency_1) {
return NULL;
}
/*
* The 2D arrays are introduced as follows
* array[i][j] = (array->data + i*array->strides[0] + j*array->strides[1])
*/
for (counter_node = 0; i < n_mask; i++){
*row_start = (array->data + i*array->strides[0]);
}
//Py_DECREF();
//return PyFloat_FromDouble();
}
谢谢!
4 个回答
2
我不确定这是否能回答你的问题,不过,如果你想在C语言中访问你的NumPy数据,可以尝试创建一个迭代器来遍历你的数组。这样做虽然不能让你像用[i][j]那样直接索引,但可以遍历整个数组。
static PyObject *func1(PyObject *self, PyObject *args) {
PyArrayObject *X;
int ndX;
npy_intp *shapeX;
NpyIter *iter;
NpyIter_IterNextFunc *iternext;
PyArray_Descr *dtype;
double **dataptr;
PyArg_ParseTuple(args, "O!", &PyArray_Type, &X);
ndX = PyArray_NDIM(X);
shapeX = PyArray_SHAPE(X);
dtype = PyArray_DescrFromType(NPY_DOUBLE);
iter = NpyIter_New(X, NPY_ITER_READONLY, NPY_KEEPORDER, NPY_NO_CASTING, dtype);
iternext = NpyIter_GetIterNext(iter, NULL);
dataptr = (double **) NpyIter_GetDataPtrArray(iter);
do {
cout << **dataptr << endl; //Do something with the data in your array
} while (iternext(iter));
NpyIter_Deallocate(iter);
return Py_BuildValue(...);
}
1
我觉得你应该看看这个链接:http://docs.scipy.org/doc/numpy/reference/c-api.array.html
特别是,
void* PyArray_GETPTR3(PyObject* obj, <npy_intp> i, <npy_intp> j, <npy_intp> k)
还有其他相关的函数。大卫·赫弗南会很惊讶如果这个API没有提供这些功能。