C 语言编程教程

C 语言 - 首页

C 语言基础

C 语言 - 概述 C 语言 - 特性 C 语言 - 发展历史 C 语言 - 环境设置 C 语言 - 程序结构 C 语言 - Hello World C - 编译过程 C - 注释 C - 标记 C - 关键字 C - 标识符 C - 用户输入 C - 基本语法 C - 数据类型 C - 变量 C - 整数提升 C - 类型转换 C - 类型转换 C - 布尔值

C 语言中的常量和文字

C - 常量 C - 字面量 C - 转义序列 C - 格式说明符

C 语言中的运算符

C - 运算符 C - 算术运算符 C - 关系运算符 C - 逻辑运算符 C - 位运算符 C - 赋值运算符 C - 一元运算符 C - 递增和递减运算符 C - 三元运算符 C - sizeof 运算符 C - 运算符优先级 C - 其他运算符

C 语言中的决策

C - 决策 C - if 语句 C - if...else 语句 C - 嵌套 if 语句 C - switch 语句 C - 嵌套 switch 语句

C 语言中的循环

C - 循环 C - While 循环 C - For 循环 C - Do...while 循环 C - 嵌套循环 C - 无限循环 C - Break 语句 C - Continue 语句 C - goto 语句

C 语言中的函数

C - 函数 C - Main 函数 C - 按值调用函数 C - 按引用调用函数 C - 嵌套函数 C - 可变参数函数 C - 用户定义函数 C - 回调函数 C - return 语句 C - 递归

C 语言中的作用域规则

C - 作用域规则 C - 静态变量 C - 全局变量

C 语言中的数组

C - 数组 C - 数组的属性 C - 多维数组 C - 将数组传递给函数 C - 从函数返回数组 C - 可变长度数组

C 语言中的指针

C - 指针 C - 指针和数组 C - 指针的应用 C - 指针运算 C - 指针数组 C - 指向指针的指针 C - 将指针传递给函数 C - 从函数返回指针 C - 函数指针 C - 指向数组的指针 C - 指向结构体的指针 C - 指针链 C - 指针 vs 数组 C - 字符指针和函数 C - NULL 指针 C - void 指针 C - 悬垂指针 C - 解引用指针 C - Near、Far 和 Huge 指针 C - 指针数组的初始化 C - 指针与多维数组

C 语言中的字符串

C - 字符串 C - 字符串数组 C - 特殊字符

C 语言的结构体和联合

C - 结构体 C - 结构体和函数 C - 结构体数组 C - 自引用结构 C - 查找表 C - 点 (.) 运算符 C - 枚举(或 enum) C - 结构填充和打包 C - 嵌套结构 C - 匿名结构和联合 C - 联合 C - Bit 位字段 C - Typedef

C 语言中的文件处理

C - 输入和输出 C - 文件 I/O(文件处理)

C 语言中的预处理器

C - 预处理器 C - #pragma 编译指示 C - 预处理器操作符 C - 宏 C - 头文件

C 语言中的内存管理

C - 内存管理 C - 内存地址 C - 存储类

C 其他主题

C - 错误处理 C - 可变参数 C - 命令执行 C - 数学函数 C - static 静态关键字 C - 随机数生成 C - 命令行参数

C 语言编程资源

C语言问题与解答答案 C语言快速指南 C语言速查表 C语言实用资源 C语言讨论


C 语言中的 Bit 位字段

声明结构体或联合类型时,结构体/联合类型变量的大小取决于其各个元素的大小。除了使用默认内存大小外,您还可以设置位的大小来限制大小。指定的大小称为位字段

以下是声明位字段的语法 -

struct {
    data_type elem : width;
};

假设您的 C 程序包含多个 TRUE/FALSE 变量,这些变量分组在一个名为 status 的结构体中,如下所示 -

struct {
    unsigned int widthValidated;
    unsigned int heightValidated;
} status;

此结构体需要 8 个字节的内存空间,但实际上,我们将在每个变量中存储"0"或"1"。在这种情况下,C 编程语言提供了一种更好的内存空间利用方法。

如果您在结构体中使用此类变量,则可以定义变量的宽度,以告知 C 编译器您将仅使用这几个字节。例如,上述结构体可以重写如下 -

struct {
   unsigned int widthValidated : 1;
   unsigned int heightValidated : 1;
} status;

上述结构体需要 4 个字节的内存空间来存储 status 变量,但仅使用 2 位来存储其值。

示例

如果您最多使用 32 个变量,每个变量的宽度为 1 位,那么 status 结构体也将使用 4 个字节。但是,一旦变量数量达到 33 个,它就会分配下一个内存槽,并开始使用 8 个字节。

让我们通过以下示例来理解这一概念 -

#include <stdio.h>

/* 定义简单结构体 */
struct {
    unsigned int widthValidated;
    unsigned int heightValidated;
} status1;

/* 定义一个包含位字段的结构体 */
struct {
   unsigned int widthValidated : 1;
   unsigned int heightValidated : 1;
} status2;

int main() {

   printf("Memory size occupied by status1: %d
", sizeof(status1));
   printf("Memory size occupied by status2: %d
", sizeof(status2));
   
   return 0;
}

输出

编译并执行上述代码后,将产生以下输出 -

Memory size occupied by status1: 8
Memory size occupied by status2: 4

位字段声明

位字段在结构体中的声明形式如下:

struct {
   type [member_name] : width ;
};

下表描述了位字段的变量元素 -

元素 描述
类型

整数类型,用于确定如何解释位字段的值。

类型可以是 int、signed intunsigned int

member_name 位字段的名称。
宽度 位字段的位数。"宽度"必须小于或等于指定类型的位宽度。

具有预定义宽度的变量称为位字段。位字段可以容纳多个位;例如,如果您需要一个变量来存储从 0 到 7 的值,则可以定义一个宽度为 3 位的位字段,如下所示 -

struct {
    unsigned int age : 3;
} Age;

上述结构体定义指示 C 编译器,变量"Age"将仅使用 3 位来存储值。如果您尝试使用超过 3 位,则系统将不允许您这样做。

示例

让我们尝试以下示例 -

#include <stdio.h>

struct {
   unsigned int age : 3;
} Age;

int main() {

   Age.age = 4;
   printf("Sizeof(Age): %d
", sizeof(Age));
   printf("Age.age: %d
", Age.age);

   Age.age = 7;
   printf("Age.age : %d
", Age.age);

   Age.age = 8;
   printf("Age.age : %d
", Age.age);

   return 0;
}

输出

编译上述代码时,会显示警告 -

warning: unsigned conversion from 'int' to 'unsigned char:3' changes value from '8' to '0' [-Woverflow]|

执行后,将产生以下输出 -

Sizeof(Age): 4
Age.age: 4
Age.age: 7
Age.age: 0

在可用存储空间有限的情况下,可以使用位字段。当设备传输编码为多个位的状态或信息时,位字段也非常有效。当某些加密程序需要访问字节中的位时,位字段用于定义数据结构。