嵌入式C语言位操作的几种常见用法
作为一名老单片机工程师,我承认,当年刚入行的时候,最怕的就是看那些密密麻麻的寄存器定义,以及那些让人眼花缭乱的位操作。
尤其是遇到那种“明明改了寄存器,硬件就是不听话”的情况,简直想把示波器砸了!那时心里默默吐槽:这谁设计的寄存器,就不能给个明确的开关按钮吗,非要让我扭来扭去?
其实,每个单片机工程师都经历过这段“痛苦”的旅程。在第一家公司,我特别佩服那个把NXP单片机寄存器玩得溜溜转的大佬,同时又对那些藏在代码深处的位操作充满恐惧。毕竟,一个不小心,就可能让你的程序跑飞,硬件罢工。

如果你也正为位操作而苦恼,那么恭喜你,找到了组织!这篇文章不会教你背诵晦涩的位操作定义,而是会用最通俗易懂的语言,带你掌握嵌入式C语言中位操作的几种常见用法。
学完之后,你不仅能轻松应对各种硬件控制任务,还能在代码优化方面更上一层楼。告别抓耳挠腮,让你也能在位操作的世界里“横着走”!
记得有一次做消费类产品,我负责一个资源非常紧张的51单片机项目,Flash和RAM都快要爆炸了。当时,我想尽各种办法优化代码,最后发现,使用位操作可以极大地压缩数据存储空间,提高程序的运行效率。
通过巧妙地运用位操作,我成功盘下了这个项目,节省了硬件成本,还赢得了老板欢心,后面我离职了几年,又以技术入股的方式把我请回去,从此踏入更大的坑,算了,血泪史,不说也罢。。。从那以后,我就也深刻体会到,掌握位操作,真的是单片机工程师的必备技能。
1.位操作,其实没那么可怕!
1.1 位操作的基石:二进制世界
在深入位操作之前,我们需要先回到二进制的世界。
单片机本质上就是处理二进制数据的机器,一切指令、数据,最终都会转化为0和1。所以,理解二进制是掌握位操作的基础。
举个例子,我们常说的“8位单片机”,指的是它的数据总线宽度是8位,也就是一次可以处理8个二进制位的数据。比如,0xAA(十六进制)在二进制中表示为10101010。而位操作,就是对这些二进制位进行各种各样的操作。
1.2 位与(&):提取信息的过滤器
位与操作符(&)的作用是,将两个操作数的对应位进行“与”运算。只有当两个位都为1时,结果才为1,否则为0。
uint8_t a = 0b10101010;
uint8_t b = 0b00001111;
uint8_t result = a & b; // result 的值为 0b00001010
位与操作最常见的应用场景是清除特定位和提取特定位。
-
清除特定位: 假设我们需要清除一个寄存器reg的第3位(从0开始计数),我们可以使用以下代码:
reg = reg & (~(1 << 3)); // 将第3位清零
这里,(1 << 3) 会生成一个掩码0b00001000,然后取反得到0b11110111。再与reg进行位与操作,就可以将第3位清零,而其他位保持不变。
-
提取特定位: 假设我们需要提取reg的第4位到第7位,可以使用以下代码:
uint8_t extracted = (reg >> 4) & 0x0F; // 提取第4位到第7位
这里,(reg >> 4) 会将reg右移4位,使得第4位到第7位移动到最低位。然后与0x0F(0b00001111)进行位与操作,就可以提取出第4位到第7位的值。
1.3 位或(|):设置信息的开关
位或操作符(|)的作用是,将两个操作数的对应位进行“或”运算。只要两个位中有一个为1,结果就为1,否则为0。
uint8_t a = 0b10101010;
uint8_t b = 0b00001111;
uint8_t result = a | b; // result 的值为 0b10101111
位或操作最常见的应用场景是设置特定位。
-
设置特定位: 假设我们需要设置reg的第2位为1,可以使用以下代码:
reg = reg | (1 << 2); // 将第2位设置为1
这里,(1 << 2) 会生成一个掩码0b00000100。然后与reg进行位或操作,就可以将第2位设置为1,而其他位保持不变。
1.4 位异或(^):翻转信息的魔法棒
位异或操作符(^)的作用是,将两个操作数的对应位进行“异或”运算。当两个位不同时,结果为1,相同时为0。
uint8_t a = 0b10101010;
uint8_t b = 0b00001111;
uint8_t result = a ^ b; // result 的值为 0b10100101
位异或操作最常见的应用场景是翻转特定位和简单加密。
-
翻转特定位: 假设我们需要翻转reg的第5位,可以使用以下代码:
-
reg = reg ^ (1 << 5); // 翻转第5位
这里,(1 << 5) 会生成一个掩码0b00100000。然后与reg进行位异或操作,就可以将第5位翻转(0变1,1变0)。
-
简单加密: 位异或操作可以用于简单的加密和解密。同一个数据与同一个密钥进行两次位异或操作,就可以恢复原始数据。
uint8_t data = 0x5A;
uint8_t key = 0x3C;
uint8_t encrypted = data ^ key; // 加密
uint8_t decrypted = encrypted ^ key; // 解密,恢复为 data
1.5 位取反(~):反转世界的钥匙
位取反操作符(~)的作用是,将操作数的每一位取反。
uint8_t a = 0b10101010;
uint8_t result = ~a; // result 的值为 0b01010101
位取反操作通常用于生成掩码,配合其他位操作实现更复杂的功能。比如,前面清除特定位的例子中,我们就用到了位取反。
1.6 左移(<<)和右移(>>):移形换影的魔术
左移操作符(<<)的作用是,将操作数的每一位向左移动指定的位数,右边补0。
右移操作符(>>)的作用是,将操作数的每一位向右移动指定的位数,左边补0(无符号数)或补符号位(有符号数)。
uint8_t a = 0b00000011;
uint8_t result_left = a << 2; // result_left 的值为 0b00001100
uint8_t result_right = a >> 1; // result_right 的值为 0b00000001
左移和右移操作最常见的应用场景是**乘以或除以2的幂**、**提取特定位**和**组合数据**。
-
乘以或除以2的幂: 左移n位相当于乘以2的n次方,右移n位相当于除以2的n次方。这比直接使用乘除法运算更快。
-
提取特定位: 就像前面提取reg的第4位到第7位的例子。
-
组合数据: 假设我们有两个8位数据,需要将它们组合成一个16位数据:
uint8_t high = 0x12;
uint8_t low = 0x34;
uint16_t combined = (high << 8) | low; // combined 的值为 0x1234
2. 实战演练:GPIO控制
说了这么多,我们来一个实战演练:使用位操作控制GPIO。
假设我们需要控制一个LED的亮灭,LED连接到GPIO的第5个引脚。
#define LED_PIN (1 << 5) // 定义LED引脚对应的掩码// 点亮LED
void led_on() {GPIO_PORT |= LED_PIN; // 设置GPIO引脚为高电平
}// 熄灭LED
void led_off() {GPIO_PORT &= ~LED_PIN; // 设置GPIO引脚为低电平
}// 翻转LED状态
void led_toggle() {GPIO_PORT ^= LED_PIN; // 翻转GPIO引脚状态
}
这个例子清晰地展示了位操作在控制硬件方面的简洁和高效。
3. 注意事项:别踩这些坑!
-
位宽问题: 确保操作的变量类型足够容纳所需的位数,避免数据溢出。
-
符号扩展: 在对有符号数进行右移操作时,注意符号位的扩展。
-
移位溢出: 移位位数不应超过变量的位宽,否则行为未定义。
-
优先级: 位操作符的优先级比较低,需要注意加括号,避免运算顺序错误。
位操作是嵌入式C语言的精髓,也是单片机工程师的必备技能。掌握位操作,你就能更高效地控制硬件,更巧妙地优化代码,在单片机世界里施展你的魔法。
希望这篇文章能帮助你打开位操作的大门,让你在嵌入式开发的道路上越走越远!记住,位操作不仅是技术,更是一种思考方式,它能让你以更精巧、更高效的方式解决问题。干吧,骚年。
最近很多粉丝问我单片机怎么学,我根据自己从业十年经验,累积耗时一个月,精心整理一份「单
片机最佳学习路径+单片机入门到高级教程+工具包」,全部无偿分享给铁粉!!!
除此以外,再含泪分享我压箱底的22个热门开源项目,包含源码+原理图+PCB+说明文档,让你迅速进阶成高手!

教程资料包和详细的学习路径可以看我下面这篇文章的开头。
《单片机入门到高级开挂学习路径(附教程+工具)》
《单片机入门到高级开挂学习路径(附教程+工具)》
《单片机入门到高级开挂学习路径(附教程+工具)》
相关文章:
嵌入式C语言位操作的几种常见用法
作为一名老单片机工程师,我承认,当年刚入行的时候,最怕的就是看那些密密麻麻的寄存器定义,以及那些让人眼花缭乱的位操作。 尤其是遇到那种“明明改了寄存器,硬件就是不听话”的情况,简直想把示波器砸了&am…...
基于Djiango实现中药材数据分析与可视化系统
中药材数据分析与可视化系统 项目截图 登录 注册 首页 药材Top20 药材价格 产地占比 历史价格 新闻资讯 后台管理 一、项目概述 中药材数据分析与可视化系统是一个基于Django框架开发的专业Web应用,致力于对各类中药材数据进行全面、系统的采集、分析和可视化展示…...
stm32(gpio的四种输出)
其实GPIO这个片上外设的功能: 用于控制IO引脚。 CPU就如同大脑,而这些片上外设就如同四肢一样的关系 如图 —————————————————————————————— OK类比了以上 其实GPIO是有 八种工作模式的 这八种工作模式 因为GPIO是面向IO…...
系统架构设计师:计算机组成与体系结构(如CPU、存储系统、I/O系统)案例分析与简答题、详细解析与评分要点
计算机组成与体系结构 10道案例分析与简答题 案例分析题(5道) 1. Cache映射与主存编址计算 场景:某计算机系统采用32位地址总线,主存容量为4GB,Cache容量为512KB,块大小为64B,使用4路组相联映射…...
Zookeeper 可观测性最佳实践
Zookeeper 介绍 ZooKeeper 是一个开源的分布式协调服务,用于管理和协调分布式系统中的节点。它提供了一种高效、可靠的方式来解决分布式系统中的常见问题,如数据同步、配置管理、命名服务和集群管理等。本文介绍通过 DataKit 采集 Zookeeper 指标&#…...
位运算---总结
位运算 基础 1. & 运算符 : 有 0 就是 0 2. | 运算符 : 有 1 就是 1 3. ^ 运算符 : 相同为0 相异为1 and 无进位相加位运算的优选级 不用在意优先级,能加括号就加括号给一个数 n ,确定它的二进制位中第 x 位是 0 还是 1? 规定: 题中所说的第x位指:int 在32位机器下4个…...
2. 什么是最普通的自动化“裸奔状态”?
什么是最普通的自动化"裸奔状态"?从大厂案例看测试代码的生存困境 一个典型的"裸奔代码"示例 # 打开目标网站 driver.get(http://test-site.com/login-page)# 登录操作 driver.find_element_by_id(user).send_keys(tester) driver.find_eleme…...
头歌java课程实验(函数式接口及lambda表达式)
第1关:利用lambda表达式对Book数组按多个字段进行排序 任务描述 本关任务:利用Comparator接口完成对Book数组同时按多个字段进行排序。 编程要求 1、本任务共有三个文件,可查看各文件的内容 2、无需修改SortBy.java枚举文件及Book.java类文…...
微信小程序三种裁剪动画有效果图
效果图 .wxml <image class"img inset {{status?action1:}}" src"{{src}}" /> <image class"img circle {{status?action2:}}" src"{{src}}" /> <image class"img polygon {{status?action3:}}" src&quo…...
C语言笔记(鹏哥)上课板书+课件汇总(结构体)-----数据结构常用
结构体 目录: 1、结构体类型声明 2、结构体变量的创建和初始化 3、结构体成员访问操作符 4、结构体内存对齐*****(重要指数五颗星) 5、结构体传参 6、结构体实现位段 一、结构体类型声明 其实在指针中我们已经讲解了一些结构体内容了&…...
git清理--解决.git文件过大问题
背景:为什么.git比我仓库中的文件大很多 为什么我的git中只有一个1KB的README,但是.git却又1G多?当我想把这个git库push到gitee时,还会报错: 根据报错信息,可看出失败的原因是:有文件的大小超过…...
Jetson Orin NX 部署YOLOv12笔记
步骤一.创建虚拟环境 conda create -n yolov12 python3.8.20 注意:YOLOv12/YOLOv11/YOLOv10/YOLOv9/YOLOv8/YOLOv7a/YOLOv5 环境通用 步骤二.激活虚拟环境 conda activate yolov12 #激活环境 步骤三.查询Jetpack出厂版本 Jetson系列平台各型号支持的最高Jetp…...
微服务2--服务治理与服务调用
前言 :本文主要阐述微服务架构中的服务治理,以及Nacos环境搭建、服务注册、服务调用,负载均衡以及Feign实现服务调用。 服务治理 服务治理是微服务架构中最核心最基本的模块。用于实现各个微服务的自动化注册与发现。 服务注册:在…...
Arduino示例代码讲解:Project 08 - Digital Hourglass 数字沙漏
Arduino示例代码讲解:Project 08 - Digital Hourglass 数字沙漏 Project 08 - Digital Hourglass 数字沙漏程序功能概述功能:硬件要求:输出:代码结构全局变量`setup()` 函数`loop()` 函数计时和点亮LED:读取倾斜开关状态:重置LED和计时器:运行过程注意事项Project 08 - …...
python生成项目依赖文件requirements.txt
文章目录 通过pip freeze去生成通过pipreqs去生成 通过pip freeze去生成 pip freeze > requirements.txt会将整个python的Interceptor的环境下lib包下所有的依赖都生成到这个文件当中,取决于我们使用的python的版本下所有的安装包。不建议使用这种方式ÿ…...
C语言之高校学生信息快速查询系统的实现
🌟 嗨,我是LucianaiB! 🌍 总有人间一两风,填我十万八千梦。 🚀 路漫漫其修远兮,吾将上下而求索。 C语言之高校学生信息快速查询系统的实现 目录 任务陈述与分析 问题陈述问题分析 数据结构设…...
精益数据分析(7/126):打破创业幻想,拥抱数据驱动
精益数据分析(7/126):打破创业幻想,拥抱数据驱动 在创业的道路上,我们都怀揣着梦想,但往往容易陷入自我编织的幻想中。我希望通过和大家一起学习《精益数据分析》,能帮助我们更清醒地认识创业过…...
Spring Boot 项目中发布流式接口支持实时数据向客户端推送
1、pom依赖添加 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-webflux</artifactId></dependency>2、事例代码 package com.pojo.prj.controller;import com.pojo.common.core.utils.String…...
Ubuntu安装MySQL步骤及注意事项
一、安装前准备 1. 系统更新:在安装 MySQL 之前,确保你的 Ubuntu 系统软件包是最新的,这能避免因软件包版本问题导致的安装错误,并获取最新的安全补丁。打开终端,执行以下两条命令: sudo apt update sudo …...
【网络篇】从零写UDP客户端/服务器:回显程序源码解析
大家好呀 我是浪前 今天讲解的是网络篇的第四章:从零写UDP客户端/服务器:回显程序源码解析 从零写UDP客户端/服务器:回显程序源码解析 UDP 协议特性核心类介绍 UDP的socket应该如何使用:1: DatagramSocket2: DatagramPacket回…...
MATLAB 控制系统设计与仿真 - 38
多变量系统控制器设计实例1 考虑如下给出的多变量系统模型: 考虑混合灵敏度问题,引入加权矩阵: 设计鲁棒控制器,并绘制闭环系统的阶跃响应曲线及开环系统的奇异值曲线。 MATLAB代码如下: clear all;clc; stf(s); g1…...
轻量化高精度的视频语义分割
Video semantic segmentation (VSS)视频语义分割 Compact Models(紧凑模型) 在深度学习中,相对于传统模型具有更小尺寸和更少参数数量的模型。这些模型的设计旨在在保持合理性能的同时,减少模型的计算和存储成本。 紧凑模型的设计可以涉及以下一些技术: 深度剪枝(Deep…...
Spring Boot 版本与对应 JDK 版本兼容性
Spring Boot 版本与对应 JDK 版本兼容性 以下是 Spring Boot 主要版本与所需 JDK 版本的对应关系,以及长期支持(LTS)信息: 最新版本对应关系 (截至2024年) Spring Boot 版本发布日期支持的 JDK 版本备注3.2.x (最新)2023-11JDK 17-21推荐使用 JDK 173…...
[密码学实战]详解gmssl库与第三方工具兼容性问题及解决方案
[密码学实战]详解gmssl库与第三方工具兼容性问题及解决方案 引言 国密算法(SM2/SM3/SM4)在金融、政务等领域广泛应用,但开发者在集成gmssl库实现SM2签名时,常遇到与第三方工具(如OpenSSL、国密网关)验证不…...
JavaScript模块化开发:CommonJS、AMD到ES模块
引言 在Web开发的早期阶段,JavaScript代码通常被编写在一个庞大的文件中或分散在多个脚本标签里,这种方式导致了全局变量污染、依赖关系难以管理、代码复用困难等问题。随着Web应用日益复杂,模块化编程成为了解决这些问题的关键。本文将带您…...
【k8s系列1】一主两从结构的环境准备
环境准备 虚拟机软件准备及安装,这里就不详细展开了,可以看文章:【一、虚拟机vmware安装】 linux环境准备及下载,下载镜像centOS7.9,以前也有写过这个步骤的文章,可以看:【二、安装centOS】 开始进入正题…...
dubbo SPI插件扩展点使用
参考:SPI插件扩展点 Dubbo SPI概述 使用IoC容器帮助管理组件的生命周期、依赖关系注入等是很多开发框架的常用设计,Dubbo中内置了一个轻量版本的IoC容器,用来管理框架内部的插件,实现包括插件实例化、生命周期、依赖关系自动注入…...
青少年编程与数学 02-016 Python数据结构与算法 26课题、生物信息学算法
青少年编程与数学 02-016 Python数据结构与算法 26课题、生物信息学算法 一、序列比对算法二、基因表达分析算法三、蛋白质结构预测算法四、系统生物学模型构建算法五、单细胞分析算法六、遗传关联分析算法七、机器学习与数据挖掘算法八、数据可视化算法九、代谢组学分析算法十…...
【Rust 精进之路之第2篇-初体验】安装、配置与 Hello Cargo:踏出 Rust 开发第一步
系列: Rust 精进之路:构建可靠、高效软件的底层逻辑 **作者:**码觉客 发布日期: 2025-04-20 引言:磨刀不误砍柴工,装备先行! 在上一篇文章中,我们一起探索了 Rust 诞生的缘由&…...
洛谷题目:P7775 [COCI 2009/2010 #2] VUK 题解 (本题简)
题目传送门: P7775 [COCI 2009/2010 #2] VUK - 洛谷 (luogu.com.cn) 前言: 这道题的核心目标是找出狼从起点 V 到终点 J 的路径,使得狼在途中离它最近的树的距离的最小值最大。下面为大家详细讲解: #整体思路概述: 这道题我们可以采用“先计算距离,再来二分查找”的…...
