当前位置: 首页 > news >正文

C语言位域与结构体打包技术

位域(Bit Fields)

位域是C语言中一种特殊的数据结构,允许在结构体中以位为单位来指定成员变量所占用的内存空间。

基本语法

struct {type [member_name] : width;
};
  • type: 整数类型(int, unsigned int, signed int等)

  • member_name: 位域成员名称

  • width: 位域占用的位数(必须小于等于指定类型的位数)

示例

// 用于存储日期,节省空间
struct date {unsigned int day : 5;    // 1-31 (需要5位)unsigned int month : 4;  // 1-12 (需要4位)unsigned int year : 12;  // 0-4095 (需要12位)
};// 硬件寄存器映射示例
struct control_reg {unsigned int enable : 1;unsigned int mode : 2;unsigned int reserved : 4;  // 保留位unsigned int status : 1;
};
struct {unsigned int age : 3;  // 3位表示年龄(0-7)unsigned int gender : 1; // 1位表示性别(0/1)unsigned int reserved : 4; // 保留位
} Person;
#include <stdio.h>struct {unsigned int age : 3;  // 使用3位存储age
} Age;int main() {Age.age = 4;printf("Sizeof(Age): %lu\n", sizeof(Age));  // 通常输出4(取决于编译器)printf("Age.age: %d\n", Age.age);          // 输出4Age.age = 8;  // 3位最大值为7(二进制111),8会导致溢出printf("Age.age: %d\n", Age.age);  // 输出0(取决于编译器实现)return 0;
}

位域的特点

  1. 节省空间:当变量只需要少量位表示时,可以节省内存

  2. 位操作:方便对硬件寄存器等需要位操作的场景

  3. 平台依赖性:位域的具体实现依赖于编译器和硬件平台

注意事项

  1. 不能对位域成员取地址(因为位域可能不按字节对齐)

  2. 位域成员不能是数组

  3. 未命名的位域可用于填充(如unsigned int : 4;

  4. 宽度为0的无名位域强制下一个位域从下一个存储单元开始

结构体打包(Packing)

结构体打包是指通过编译器指令或属性来控制结构体成员的内存对齐方式,以减少内存占用。

常见方法

  1. 使用编译器指令:

    • GCC/Clang: __attribute__((packed))

    • MSVC: #pragma pack(push, 1) 和 #pragma pack(pop)

  2. 重新排列结构体成员:
    按照从大到小的顺序排列成员可以减少填充字节

GCC/Clang示例

struct __attribute__((packed)) PackedStruct {char a;int b;char c;
};

MSVC示例

#pragma pack(push, 1)  // 设置为1字节对齐
struct PackedStruct {char a;int b;char c;
};
#pragma pack(pop)      // 恢复默认对齐

注意事项

  1. 打包结构体可能导致性能下降,因为未对齐的内存访问在某些架构上较慢

  2. 打包结构体在不同平台间的可移植性可能有问题

  3. 常用于网络协议、硬件寄存器映射等需要精确控制内存布局的场景

结合使用示例

#pragma pack(push, 1)  //设置以1个字节为对齐长度
typedef struct _18E6EFF3_Frame{uint32_t                accumulated_charge_capacity     :24;        //累计充电容量 factor 0.1,offset 0 ahuint32_t                accumulated_discharge_capacity  :24;        //累计放电容量 factor 0.1,offset 0 ahuint16_t                single_charge_capacity;                     //单次充电容量 factor 0.1,offset 0 ah
}_18E6EFF3_Frame;
#pragma pack(pop)      //恢复默认对齐方式.
// 一个紧凑的IP头结构表示(简化版)
struct __attribute__((packed)) IPHeader {unsigned int version : 4;    // IP版本unsigned int ihl : 4;        // 头部长度(以32位字计)unsigned int tos : 8;        // 服务类型unsigned int tot_len : 16;   // 总长度unsigned int id : 16;        // 标识unsigned int frag_off : 16;  // 分片偏移unsigned int ttl : 8;        // 生存时间unsigned int protocol : 8;   // 协议unsigned int check : 16;     // 校验和unsigned int saddr;          // 源地址unsigned int daddr;          // 目的地址
};

总结

位域和结构体打包技术可以有效地减少内存使用,特别适用于嵌入式系统或需要与硬件/网络协议交互的场景。但使用时需要注意潜在的性能影响和可移植性问题。

相关文章:

  • 【漫话机器学习系列】243.数值下溢(Underflow)
  • ​Spring + Shiro 整合的核心要点及详细实现说明
  • 【QT】QT软件编译生成exe后,需要拷贝依赖库使用方法
  • QT键盘触发按钮
  • Android开发-创建、运行、调试App工程
  • 01_线性表
  • Java中的Classpath 包含哪些目录?
  • linux -shell原理与运用
  • openwrt 使用quilt 打补丁(patch)
  • 【Harbor v2.13.0 详细安装步骤 安装证书启用 HTTPS】
  • WebRTC并非万能:RTMP与RTSP的工程级价值再认识
  • Flutter开发IOS蓝牙APP的大坑
  • 【2025】Visio 2024安装教程保姆级一键安装教程(附安装包)
  • 【奔跑吧!Linux 内核(第二版)】第1章:Linux 系统基础知识
  • Ros工作空间
  • IDEA 占用C盘太大清理
  • LangChain:大语言模型应用的“瑞士军刀”入门指南
  • 上市公司-企业上下游供应链数据(2003-2023年)-社科数据
  • 推导部分和-图论+dfs+连通块
  • 【数据挖掘】Apriori算法
  • 纪念苏联伟大卫国战争胜利80周年阅兵彩排,解放军仪仗队亮相
  • 4月深圳新房、二手房成交同比均上涨,“5月有望延续积极向好的发展态势”
  • 上海虹桥高铁站拦门事件反转,谁在带偏网友?
  • 娱见 | 为了撕番而脱粉,内娱粉丝为何如此在乎番位
  • 许昌市场监管部门对胖东来玉石开展日常检查:平均毛利率不超20%
  • 特朗普要征100%关税,好莱坞这批境外摄制新片有麻烦了