Вопрос по ctypes, python – Как изменить поля структуры с помощью указателей Python ctypes

1

Ниже приведен код доступа к значениям dll с использованием ctypes.

Мое намерение состоит в том, чтобы хранить адреса полей структуры. Всякий раз, когда значения в структуре меняются, я могу получить доступ к адресам и получить измененные значения.

DUMMY_DLL_PATH = "dummyModel.dll"

class MyStruct(ctypes.Structure):  
    _fields_ = [("field_one", ctypes.c_int),  
                ("field_two", ctypes.c_int)]  

d_m = ctypes.cdll.LoadLibrary(DUMMY_DLL_PATH)  

d_i = MyStruct.in_dll(d_m,"dummy_In")  

in_field = ctypes.c_int(d_i.field_one)  

#storing the address  
b = ctypes.addressof(in_field)  
b_v = ctypes.cast(b,ctypes.POINTER(ctypes.c_int))  

k= b_v.contents   

print 'before',d_i.field_one,k.value  
#changing the value  
d_i.field_one = 10  

print 'After',d_i.field_one,k.value  
Output:
Before 0 0  
After 10 0  

Через указатели значения не меняются. Осталось 0

Ваш Ответ

1   ответ
4

Проблема вin_field это новыйc_int объект, занимающий другую память, чем исходная структура. Что вы хотитеc_int.from_buffer (документы) который разделяет память исходного объекта. Вот пример:

Windows DLL source x.c compiled with cl /LD x.c:

struct MyStruct
{
    int one;
    int two;
};

__declspec(dllexport) struct MyStruct myStruct = {1,2};

Python script:

from ctypes import *

class MyStruct(Structure):
    _fields_ = [
        ("one", c_int),
        ("two", c_int)]
    def __repr__(self):
        return 'MyStruct({},{})'.format(self.one,self.two)

dll = CDLL('x')
struct = MyStruct.in_dll(dll,"myStruct")
alias1 = c_int.from_buffer(struct, MyStruct.one.offset)
alias2 = c_int.from_buffer(struct, MyStruct.two.offset)
print struct
print 'before',alias1,alias2
struct.one = 10
struct.two = 20
print 'after',alias1,alias2

Output:

MyStruct(1,2)
before c_long(1) c_long(2)
after c_long(10) c_long(20)

Похожие вопросы