由浅到深认识C语言(12):位段/位域

2024-03-17 11:12
文章标签 语言 认识 位域 位段

本文主要是介绍由浅到深认识C语言(12):位段/位域,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

该文章Github地址:https://github.com/AntonyCheng/c-notes

在此介绍一下作者开源的SpringBoot项目初始化模板(Github仓库地址:https://github.com/AntonyCheng/spring-boot-init-template & CSDN文章地址:https://blog.csdn.net/AntonyCheng/article/details/136555245),该模板集成了最常见的开发组件,同时基于修改配置文件实现组件的装载,除了这些,模板中还有非常丰富的整合示例,同时单体架构也非常适合SpringBoot框架入门,如果觉得有意义或者有帮助,欢迎Star & Issues & PR!

上一章:由浅到深认识C语言(11):结构体

12.位段/位域

这里需要记住,在内存中数据插入输出的顺序,即逆序插入,顺序输出!详解在8.3指针变量中变量的类型;

问题提出:信息在计算机的存取长度一般以 Byte 为单位,但是有时候存储一个信息不必用到一个或者多个 Byte,例如”真“或”假“用 1 或 0 表示即可,只需要 1 bit,此时我们就需要使用位段

**定义:**C语言允许在一个结构体中以 bit 为单位来指定其成员所占内存长度,以 bit 为单位的成员称为”位段“或称”位域“,而且这些成员的数据类型大都为 unsigned int ,也可以用 unsigned char

格式示例如下:

struct data
{unsigned int a : 2;		//a占2位unsigned int b : 6;		//b占6位unsigned int c : 4;		//c占4位unsigned int d : 4;		//d占4位unsigned int i;			//i占32位
}data;

图解如下:

idcba
32bit4bit4bit6bit2bit

而且相邻位域是可以被压缩的,且相邻位域压缩的位数不能超过成员自身的大小;

解释如下: 以下数据类型仅为 unsigned int 和 unsigned char;

若相邻成员数据类型相同(可压缩):

  • 如果我们使用的是 unsigned int 类型,最多能够压缩 32 个连续的 unsigned int num : 1; 即最多压缩 320/1 编码,如果连续的相同数据类型成员位数超过了 4B ,即 32b ,如果大小溢出了 4B ,即大于等于了 33b 的话,进而成为 8B ,如果只压缩了 33unsigned int num : 1 ,无论打印出来是多少 B ,实际大小还是为 33b ,往后以此类推;

  • 如果我们使用的是 unsigned char 类型,最多能够压缩 8 个连续的 unsigned char ch : 1; 即最多压缩 80/1 编码,如果连续的相同数据类型成员位数超过了 1B ,即 8b ,如果大小溢出了 1B ,即大于等于了 9b 的话,进而成为 2B ,如果只压缩了 9unsigned char ch : 1 ,无论打印出来是多少 B ,实际大小还是为 9b ,往后以此类推;

若相邻成员数据类型不同(不可压缩):

不相邻的两个成员或者两类成员,则既要将两类数据内存按照 int 或者 char 基本数据类型的内存进行加减,又要将每一类中的各个成员内存进行压缩;

示例如下:

#include<stdio.h>
#pragma pack(1)
struct bit_data
{unsigned char a : 1;unsigned char b : 1;unsigned char c : 1;unsigned char d : 1;unsigned char e : 1;unsigned char f : 1;unsigned char g : 1;unsigned char h : 1;
}data;int main(int argc, char* argv[]) {printf("data = %d\n", sizeof(data));
}

打印效果如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

12.1.位段的注意事项

  • 位段不能取地址:

    因为内存地址是以 Byte 为单位,因为系统内存是以 Byte 为基本单位,系统不会识别在 bit 上的地址;

  • 位段的赋值:

    不要操作越界的数据,解释如下:

    #include<stdio.h>struct bit_data
    {unsigned char a : 1;	//0-1				==>		0-1unsigned char b : 2;	//00-11				==>		0-3unsigned char c : 3;	//000-111			==>		0-7unsigned char d : 4;	//0000-1111			==>		0-15unsigned char e : 5;	//00000-11111		==>		0-31unsigned char f : 6;	//000000-111111		==>		0-63unsigned char g : 7;	//0000000-1111111	==>		0-127unsigned char h : 8;	//00000000-11111111	==>		0-255
    }data;int main(int argc, char* argv[]) {data.a = 1;printf("%u\n", data.a);data.b = 3;printf("%u\n", data.b);data.c = 7;printf("%u\n", data.c);data.d = 15;printf("%u\n", data.d);data.e = 31;printf("%u\n", data.e);data.f = 63;printf("%u\n", data.f);data.g = 127;printf("%u\n", data.g);data.h = 255;printf("%u\n", data.h);
    }
    

    打印效果如下:

    外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

  • 无意义位段:

    无意义位段用来限制压缩,解释如下:

    以下示例将位段定义成了全局变量,可以不用初始化为零值(因为全局变量会自动初始化),但是如果将该位段定义成局部变量,那么就需要我们手动初始化为零,因为局部变量载入时是一个不确定的值;

    struct bit_data
    {unsigned char a : 2;unsigned char : 2;    //无意义的位段,仅作占位符,占两个 bitunsigned char b : 2;unsigned char c : 2;
    }data;
    

    图示如下:

    ccbbaa

    验证如下:

    #include<stdio.h>
    #include<string.h>
    struct bit_data
    {unsigned char a : 2;unsigned char : 2;    //无意义的位段,仅作占位符,占两个 bitunsigned char b : 2;unsigned char c : 2;
    }data;int main(int argc, char* argv[]) {//如果是定义成了局部变量,还需要包含string.h头文件,然后://memset(&data,0,1);data.a = 0;data.b = 2;data.c = 3;//此时data转换成二进制为 1110 0000 ==> 十六进制 0xe0printf("data = %#x\n",data);
    }
    

    打印效果如下:

    外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

  • 另起一个位段:

    当我们将无意义位段设置为 0 时,我们就成功另起了一个位段,即在内存的另一处重新开辟了一个容器来装下一个位段的内容,证明如下:

    #include<stdio.h>
    #include<string.h>
    typedef struct data{unsigned char a : 2;unsigned char : 0;unsigned char b : 2;unsigned char : 0;unsigned char c : 2;unsigned char : 0;unsigned char d : 2;
    }DATA;int main(int argc, char* argv[]) {DATA data;memset(&data, 0, 4);//此时这个结构体里包含了四个位段地址printf("sizeof(data) = %d\n", sizeof(data));data.a = 0;data.b = 1;data.c = 2;data.d = 3;printf("%x\n", data);
    }
    

    打印效果如下:

    外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

12.2.位域和单片机

图解如下:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

代码如下:

#include<stdio.h>typedef struct {unsigned char data : 1;unsigned char : 1;unsigned char addr : 2;unsigned char : 1;unsigned char opt : 2;unsigned char : 1;
}DATA;int main(int argc, char* argv[]) {//下面是单片机定义DATA msg;msg.data = 1;msg.addr = 2;msg.opt = 1;
}

上一章:由浅到深认识C语言(13):共用体

这篇关于由浅到深认识C语言(12):位段/位域的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/818779

相关文章

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

每天认识几个maven依赖(ActiveMQ+activemq-jaxb+activesoap+activespace+adarwin)

八、ActiveMQ 1、是什么? ActiveMQ 是一个开源的消息中间件(Message Broker),由 Apache 软件基金会开发和维护。它实现了 Java 消息服务(Java Message Service, JMS)规范,并支持多种消息传递协议,包括 AMQP、MQTT 和 OpenWire 等。 2、有什么用? 可靠性:ActiveMQ 提供了消息持久性和事务支持,确保消

认识、理解、分类——acm之搜索

普通搜索方法有两种:1、广度优先搜索;2、深度优先搜索; 更多搜索方法: 3、双向广度优先搜索; 4、启发式搜索(包括A*算法等); 搜索通常会用到的知识点:状态压缩(位压缩,利用hash思想压缩)。

科研绘图系列:R语言扩展物种堆积图(Extended Stacked Barplot)

介绍 R语言的扩展物种堆积图是一种数据可视化工具,它不仅展示了物种的堆积结果,还整合了不同样本分组之间的差异性分析结果。这种图形表示方法能够直观地比较不同物种在各个分组中的显著性差异,为研究者提供了一种有效的数据解读方式。 加载R包 knitr::opts_chunk$set(warning = F, message = F)library(tidyverse)library(phyl

透彻!驯服大型语言模型(LLMs)的五种方法,及具体方法选择思路

引言 随着时间的发展,大型语言模型不再停留在演示阶段而是逐步面向生产系统的应用,随着人们期望的不断增加,目标也发生了巨大的变化。在短短的几个月的时间里,人们对大模型的认识已经从对其zero-shot能力感到惊讶,转变为考虑改进模型质量、提高模型可用性。 「大语言模型(LLMs)其实就是利用高容量的模型架构(例如Transformer)对海量的、多种多样的数据分布进行建模得到,它包含了大量的先验

C语言 | Leetcode C语言题解之第393题UTF-8编码验证

题目: 题解: static const int MASK1 = 1 << 7;static const int MASK2 = (1 << 7) + (1 << 6);bool isValid(int num) {return (num & MASK2) == MASK1;}int getBytes(int num) {if ((num & MASK1) == 0) {return

MiniGPT-3D, 首个高效的3D点云大语言模型,仅需一张RTX3090显卡,训练一天时间,已开源

项目主页:https://tangyuan96.github.io/minigpt_3d_project_page/ 代码:https://github.com/TangYuan96/MiniGPT-3D 论文:https://arxiv.org/pdf/2405.01413 MiniGPT-3D在多个任务上取得了SoTA,被ACM MM2024接收,只拥有47.8M的可训练参数,在一张RTX

如何确定 Go 语言中 HTTP 连接池的最佳参数?

确定 Go 语言中 HTTP 连接池的最佳参数可以通过以下几种方式: 一、分析应用场景和需求 并发请求量: 确定应用程序在特定时间段内可能同时发起的 HTTP 请求数量。如果并发请求量很高,需要设置较大的连接池参数以满足需求。例如,对于一个高并发的 Web 服务,可能同时有数百个请求在处理,此时需要较大的连接池大小。可以通过压力测试工具模拟高并发场景,观察系统在不同并发请求下的性能表现,从而

C语言:柔性数组

数组定义 柔性数组 err int arr[0] = {0}; // ERROR 柔性数组 // 常见struct Test{int len;char arr[1024];} // 柔性数组struct Test{int len;char arr[0];}struct Test *t;t = malloc(sizeof(Test) + 11);strcpy(t->arr,

C语言指针入门 《C语言非常道》

C语言指针入门 《C语言非常道》 作为一个程序员,我接触 C 语言有十年了。有的朋友让我推荐 C 语言的参考书,我不敢乱推荐,尤其是国内作者写的书,往往七拼八凑,漏洞百出。 但是,李忠老师的《C语言非常道》值得一读。对了,李老师有个官网,网址是: 李忠老师官网 最棒的是,有配套的教学视频,可以试看。 试看点这里 接下来言归正传,讲解指针。以下内容很多都参考了李忠老师的《C语言非