为了账号安全,请及时绑定邮箱和手机立即绑定

编辑 Ctypes 结构的缓冲区

编辑 Ctypes 结构的缓冲区

慕村225694 2022-06-28 11:09:55
我有一个 Ctypes 结构,它需要以易于实现的标准方式 [setattr(structure,value)] 进行编辑,但我还需要能够编辑原始缓冲区,因为我希望能够分配某个位到一个值(例如位 25 = 0xd5) 我该怎么做?非常简化的示例代码,如果有帮助的话import ctypes as *#ctypes array with ~250,000 c_uint32 elementshuge_arr = (c_uint32*250,000)(*range(250,000))  # Fill with dummy data for this exampleclass Example(Structure):    _pack_ = 1    _fields_ = [        ("a", c_uint16),        ("b", c_uint16, 14),        ("c", c_uint16, 2),        ("d", c_uint32, 24),        ("e", c_uint32, 8),        ("f", c_uint16),        ("g", c_uint16)    ]offset = 123456example_struct = Example.from_buffer(huge_arr, offset)# Ideally, I'd like to be able to set bits in example_struct. for example, example_struct[2] = 0x2b我知道可以通过执行 huge_arr[offset+2] = 0x2b 来执行 example_struct[2] = 0x2b,但我的程序比这个示例更复杂,huge_arr 被定义(并保留)在主文件中,而 example_struct 被传输作为不同文件中另一个函数的参数,因此 huge_arr 超出范围。有没有办法改变 exmaple_struct 的第 n 位?需要注意的另一件事是,该程序是用 Python 2.7 编写的,但即使是 python3 解决方案也将不胜感激预先感谢您的帮助(我肯定会为任何可以解决此问题的善良的灵魂标记最佳答案)
查看完整描述

1 回答

?
子衿沉夜

TA贡献1828条经验 获得超3个赞

清单[Python 3.Docs]:ctypes - Python 的外部函数库。


您在问题中的几个地方使用了“位”一词,但您的意思可能是“字节”(因为位只能具有0或1的值)。


为了实现您的目标,您可以将结构包装在一个联合中。


code00.py:


#!/usr/bin/env python3


import sys

import ctypes as ct



class ExampleStruct(ct.Structure):

    _pack_ = 1

    _fields_ = [

        ("a", ct.c_uint16),

        ("b", ct.c_uint16, 14),

        ("c", ct.c_uint16, 2),

        ("d", ct.c_uint32, 24),

        ("e", ct.c_uint32, 8),  # @TODO - cfati: Why not c_uint8 ???

        ("f", ct.c_uint16),

        ("g", ct.c_uint16),

        ("test_field", ct.c_uint8),  # One byte field would make the example more eloquent

    ]



class Example(ct.Union):

    _anonymous_ = ["struct"]

    _fields_ = [

        ("struct", ExampleStruct),

        ("raw", ct.c_ubyte * ct.sizeof(ExampleStruct)),

    ]



def main():

    huge_arr_size = 250000

    huge_arr = (ct.c_uint32 * huge_arr_size)(*range(huge_arr_size))

    arr_offset = 123456

    example = Example.from_buffer(huge_arr, arr_offset)

    print("example.test_field: {0:d}".format(example.test_field))

    test_field_offset = Example.test_field.offset

    example.raw[test_field_offset] = 123

    print("example.test_field: {0:d}".format(example.test_field))



if __name__ == "__main__":

    print("Python {0:s} {1:d}bit on {2:s}\n".format(" ".join(item.strip() for item in sys.version.split("\n")), 64 if sys.maxsize > 0x100000000 else 32, sys.platform))

    main()

    print("\nDone.")

输出:


[cfati@CFATI-5510-0:e:\Work\Dev\StackOverflow\q058460001]> "e:\Work\Dev\VEnvs\py_064_03.07.03_test0\Scripts\python.exe" code00.py

Python 3.7.3 (v3.7.3:ef4ec6ed12, Mar 25 2019, 22:22:05) [MSC v.1916 64 bit (AMD64)] 64bit on win32


example.test_field: 147

example.test_field: 123


Done.


查看完整回答
反对 回复 2022-06-28
  • 1 回答
  • 0 关注
  • 92 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
微信客服

购课补贴
联系客服咨询优惠详情

帮助反馈 APP下载

慕课网APP
您的移动学习伙伴

公众号

扫描二维码
关注慕课网微信公众号