本文共 1369 字,大约阅读时间需要 4 分钟。
Q: 假设 aVal 的地址为 6684156,求 pVal 的值?
pVal = &aVal + offset,需要考虑指针的类型。aVal 的地址为 6684156,不是可以简单地加上 offset = 2。正确的做法是将 offset 转换为与 int 类型相关联的字节大小(通常是 4 个字节),然后进行偏置操作。因此,pVal 的值为 6684156 + 2 * sizeof(int)。在 32 位系统中,sizeof(int) 为 4,所以 pVal 的值为 6684164。 typedef struct{ uint16_t LAeq; uint16_t LBeq; uint16_t LCeq; uint16_t LZeq;} myData_t; #pragma pack() int j = 0; uint16_t measVal = 0; myData_t data; for(int m = 0; m < 4; m++){ measVal = m * 10 + j; memcpy((uint8_t*)(unsigned long)&data + j, (uint8_t*)&measVal, 2); j += 2; printf("data[%d], %04x, addr=%d\r\n", m, *(uint16_t*)((uint8_t*)(unsigned long)&data + m * 2), (uint8_t*)&data + m * 2); } 在该代码中,memcpy 函数用于将 measVal 的值拷贝到 data 结构体的相应位置。memcpy 的第四个参数为 2,表示每次拷贝 2 个字节。j 变量用于遍历 data 结构体的各个成员,并逐步增加 2 个字节以访问下一个成员。
在 printf 语句中,第二个参数使用了多次强制转换:
*(uint16_t*):表示取地址操作,获取地址值。((uint8_t*)(unsigned long)):首先将地址强制转换为 unsigned long,以确保在 32 位和 64 位系统中都能正确使用地址。+ m * 2:表示在结构体起始地址的基础上增加 m * 2 个字节,访问下一个成员的位置。(uint8_t*):将地址进一步强制转换为 uint8_t*,以便按照单字节偏置操作对象的内存。通过对指针进行强制转换,可以确保内存操作的正确性,并兼容不同类型的系统架构。
转载地址:http://mbwp.baihongyu.com/