【问题标题】:Access data at memory address with ctypes使用 ctypes 访问内存地址处的数据
【发布时间】:2013-07-17 23:33:46
【问题描述】:

我正在尝试为一个将输入作为指针的旧 C 程序创建一个 Python 包装器。此时,我可以让程序运行,但不知道如何取回指定指针处的值。

这是一个简化的 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", c_int),
                ("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 的值列表。

【问题讨论】:

    标签: python c pointers ctypes


    【解决方案1】:

    您的 C 函数返回 void,而不是 CONTEXT *。因此,您的代码只是将随机未初始化的内存投射到 CONTEXT *,然后尝试取消它。

    最重要的是,即使它确实通过引用返回了 CONTEXT 对象,tmp[1] 也会尝试在该对象之后 取消对内存的引用,所以它仍然是垃圾。

    当您尝试将随机内存解释为双精度时,如果幸运的话,您会得到类似 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 函数 did 返回一个与你的 ctypes 声明匹配的 CONTEXT 结构数组,这一切都可以正常工作。例如:

    #include <stdlib.h>
    
    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;
    }
    

    编译它,运行现有的 Python 脚本并添加 print tmp[1].wts[1],它会打印出 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)
    

    【讨论】:

    • 感谢您投入时间和精力来解释和修复我的错误!我发现的唯一一件事是c_void 似乎不存在,但None 代替了它。我不太懂 C,所以我尽量避免更改该代码。
    • 快速跟进:我将如何实现它,以便 C 函数将 n 作为指针 *n 接收,然后通过 m=*n+1 对其进行操作?我当前的代码失败了。具体来说,是否有 Python 代码中的快速修复来执行此操作?
    • @brebs:对,对不起,void *c_void_p,但 void 只是 None。无论如何,对于后续工作,我将编辑答案。
    • @eryksun:我认为没有这种变化会更容易解释,因为 OP 专门询问如何制作指针......但你是对的,最好解释一下他为什么不这样做' t 真的想首先创建一个指针(他可以从 Python/ctypes 访问)。我会编辑答案。
    猜你喜欢
    • 1970-01-01
    • 2018-09-08
    • 2013-02-18
    • 1970-01-01
    • 2013-10-30
    • 1970-01-01
    • 2017-08-28
    • 2018-04-12
    • 2010-09-12
    相关资源
    最近更新 更多