- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我有一个用 C 语言编写的打包结构,我想用 Python 对其进行解析。我注意到 C(使用 GCC 4.9.2)和 Python 3.4.2 中的 ctypes 库之间的运算符 sizeof
返回的位域结构大小存在差异。
以下 C 代码按预期打印 5:
#include <stdio.h>
#include <stdint.h>
typedef struct __attribute__((packed)) {
uint32_t ch0 : 20;
uint32_t ch1 : 20;
} pkt_t;
int main(){
printf("sizeof(pkt_t): %d\n", sizeof(pkt_t));
return 0;
}
虽然 Python 中的(相同)代码打印出 8
import ctypes
class Packet(ctypes.LittleEndianStructure):
_pack_ = 1
_fields_ = [
('ch0', ctypes.c_uint32, 20),
('ch1', ctypes.c_uint32, 20),
]
print(ctypes.sizeof(Packet()))
看起来 _pack_ = 1
等同于 C 中的 __attribute__((aligned(1)))
,而不是 __attribute__((packed, aligned (1)))
这将使结构尽可能紧密地打包。有没有办法为 ctypes 结构启用 packed
属性?
最佳答案
ctypes 文档 https://docs.python.org/3/library/ctypes.html#structure-union-alignment-and-byte-order明确指出
By default, Structure and Union fields are aligned in the same way theC compiler does it. It is possible to override this behavior byspecifying a pack class attribute in the subclass definition. Thismust be set to a positive integer and specifies the maximum alignmentfor the fields. This is what #pragma pack(n) also does in MSVC.
这意味着它们不引用 gcc 的作用,而是引用 MSVC #pragma pack
。请参阅:https://learn.microsoft.com/en-us/cpp/preprocessor/pack?view=vs-2019
n
(Optional) Specifies the value, in bytes, to be used for packing. Ifthe compiler option /Zp isn't set for the module, the default valuefor n is 8. Valid values are 1, 2, 4, 8, and 16. The alignment of amember is on a boundary that's either a multiple of n, or a multipleof the size of the member, whichever is smaller.
因此,观察到的行为的原因之一在于 MSVC 与 gcc 的编译器细节。
gcc 文档 https://gcc.gnu.org/onlinedocs/gcc-4.9.4/gcc/Type-Attributes.html#Type-Attributes (抱歉,gcc 页面上没有 gcc-4.9.2 的文档),另一方面,cleary 告诉我们:
This attribute(packed), attached to struct or union type definition, specifiesthat each member (other than zero-width bit-fields) of the structureor union is placed to minimize the memory required. When attached toan enum definition, it indicates that the smallest integral typeshould be used.
在这种情况下,它们并没有说明任何边界要求,而内存占用量已最小化。跨字节边界放置位域是否有意义可能是另一个讨论的主题。为什么 __attribute__((aligned(1)))
实际上按预期设置结构,这对我来说似乎很合乎逻辑且一致,因为它明确地强制执行结构成员的字节对齐,这似乎不是通过指定 __attribute__((packed))
来区分大小写。在第一种情况下,gcc 编译器也会对位字段强制执行字节对齐。
总结:
__attribute__ ((aligned (n)))
要求结构(成员)的对齐__attribute__ ((__packed__))
要求最小化内存占用,即使这意味着位域字节边界被跨越关于c - ctypes.struct(打包)中的 sizeof 与 C 中的打包结构不匹配,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40454606/
我有一个数组 items[] items[] 中的每一项都是一个结构体。 item 有键 id、date、value(即 item.id、item.date、item.value) 我想使用 Stru
我想存储 100 名员工。 RollNo,姓名,工资,时间(各种数据,我无法在这里解释,但你可以看下面的代码片段来理解 main() { struct day { int hour
这个问题在这里已经有了答案: storage size of ‘names’ isn’t known (3 个答案) 关闭 5 年前。 我正在尝试蓝牙编程,遇到了这个我不明白的问题。基本上,当我使用
这是一个奇怪的事情: 我有一个结构,它包含指向相同类型结构的指针和指向其他类型结构的指针,以及一些其他值。 struct animal { struct animal * father;
我有一个结构定义如下(名称不同) struct str1 { int field1; struct str2; } 我在一个函数中有一个*str1。我想要一个指向 str2 的指针。 所以
DISK_DETECTION_INFO is defined as有什么原因吗? typedef struct _DISK_DETECTION_INFO { DWORD Size
我正在尝试打包一个字符串和一个字符串的长度。 fmt = '
我在创建结构时遇到问题。 我的结构: public struct Device: Codable { let data: DeviceData let meta: Meta? } pu
struct Item { var name:String? var type:String? var value:Int? var tag:Int? } ... ..
// NewReaderSize returns a new Reader whose buffer has at least the specified 43 // size. If the ar
这个问题在这里已经有了答案: Sorting a vector of custom objects (14 个答案) 关闭 3 年前。 在下面的 C++ 片段中, 如何基于 TwoInts 结构中的
#include struct Header { unsigned long long int alignment; }; int main(void) { struct Heade
我有一个目前看起来像这样的结构(缩写为仅显示基本部分): typedef struct { uint32_t baudrate; ... some other internally u
对此没有太多解释,这就是我所拥有的: public struct PACKET_HEADER { public string computerIp; publi
我有以下代码: struct MyStruct{ data: &'a str, } fn get(S: &'a MyStruct) -> &'a str{ S.data } fn se
struct S1 { char c; int i; }; struct S3 { char c1; struct S1 s; double c2; }; 我正
我有一个名为 Parameter 的协议(protocol): protocol Parameter { var name: String { get } var unit: Unit
有 2 个 struct 定义 A 和 A。我知道 struct A 可以包含指向 struct A 的 POINTER 但我不明白为什么 struct A 不能包含struct A(不是指针) 最佳
我有以下代码: struct MyStruct{ data: &'a str, } fn get(S: &'a MyStruct) -> &'a str{ S.data } fn se
为了说明这一点,这里有一个小的不可变结构和一个更新它的函数: (struct timeseries (variable observations) #:transparent) (define (ad
我是一名优秀的程序员,十分优秀!