首页 > 解决方案 > Cython:将指针传递给采用 void ** 的函数中的指针

问题描述

我正在尝试编写一个函数,在某些条件下,将指向结构的指针更改为指向不同的结构。

我的约束是我想保留初始函数签名,它将指向指针(而不是特定结构类型)的通用指针作为参数。

这不起作用:

[nav] In [5]: %%cython -f 
         ...: ctypedef struct A: 
         ...:     int x 
         ...:     int y 
         ...:  
         ...: cdef fn(void **m): 
         ...:     # Arbitrary code that changes m[0] to point to another structure
         ...:     pass
         ...:      
         ...: cdef A a 
         ...: cdef A *ap = &a 
         ...: a.x = 2 
         ...: a.y = 3 
         ...: print(ap.x) 
         ...: fn(&ap) 
         ...: print(ap.x) 

Error compiling Cython file:
------------------------------------------------------------
...
cdef A a
cdef A *ap = &a
a.x = 2
a.y = 3
print(a.x)
fn(&ap)
  ^
------------------------------------------------------------

/home/me/.cache/ipython/cython/_cython_magic_3c3902694eafae18c66cb761d4a6b210.pyx:20:3: Cannot assign type 'A **' to 'void **'

我想那是因为即使我可以编写一个以 avoid *作为参数并自动将任何传递的指针转换为a 的函数,void *它也不适用于指向 void 指针的指针,对吗?

如果是这样,我如何传递我的结构指针指针,以便ap可以指向不同的结构?

谢谢。

编辑:

进一步阅读后,我意识到这是 C 架构的“功能”。这篇有用的文章给出了一些简短的解释:

关于指针的指针和内存分配的一方面:虽然void *返回的类型malloc是“通用指针”,适合分配给任何类型的指针或从任何类型的指针分配,但假设的类型void **不是“指向指针的通用指针”。我们的allocstr示例只能用于分配指向 char 的指针。不可能使用通过指针间接返回通用指针的函数void **,因为当您尝试使用它时,例如通过声明和调用

double *dptr;
if(!hypotheticalwrapperfunc(100, sizeof(double), &dptr))
    fprintf(stderr, "out of memory\n");

您不会传递 a void **,而是传递 a double **

我收集到,此时我可以做的是在我的函数签名中指定数据类型,或者返回新分配的值?

标签: cpointerscython

解决方案


我通过传递函数要求的内容来解决,即 a void **,然后在需要时进行转换:

[nav] In [5]: %%cython -f 
         ...: ctypedef struct A: 
         ...:     int x 
         ...:     int y 
         ...:  
         ...: cdef fn(void **m): 
         ...:     # Arbitrary code that changes m[0] to point to another structure
         ...:     pass
         ...:      
         ...: cdef A a 
         ...: cdef void *avp = <void *>&a
         ...: cdef A *ap = <A*>avp
         ...: a.x = 2 
         ...: a.y = 3 
         ...: print(ap.x) 
         ...: fn(&ap)
         ...: ap = <A*>avp # Must reassign
         ...: print(ap.x) 

推荐阅读