【ARM 嵌入式 C 入门及渐进 12.3 -- 将数值的第 s 位到 e 位清零】

请阅读【嵌入式开发学习必备专栏】


文章目录

    • [将数值的第 s 位到 e 位清零](#将数值的第 s 位到 e 位清零)

将数值的第 s 位到 e 位清零

为了定义一个VAL_CLR_BITS(val, s, n)宏,该宏将变量val的第s位到第n位清零(假设n >= s),其余位的值保持不变,我们可以使用位操作来实现这一目标。以下是宏的一个可能实现:

c 复制代码
#define VAL_CLR_BITS(val, s, n) ((val) & (~(((1ULL << ((n) - (s) + 1)) - 1) << (s))))

这个宏的工作原理如下:

  1. 1ULL << ((n) - (s) + 1):首先将1(使用unsigned long long类型以确保宽度)左移(n - s + 1)位。这样做的目的是创建一个值,其中有(n - s + 1)个1位,与val中从sn位对应的位数相同。
  2. -1:然后,从这个值中减去1,这会将所有的1位左移后的结果变为0,其余为1。
  3. << (s):接着,将上一步得到的值左移s位,这样就将清零的位对准了val的第s位到第n位。
  4. ~:然后对上一步的结果取反,这会将要清零的位变为0,其余位变为1。
  5. &:最后,使用与操作&val与第4步得到的掩码进行位与操作。由于掩码中第s位到第n位为0,其余位为1,因此这一操作将清零val的第s位到第n位,而保留其余位的值不变。
    下面是一个如何使用VAL_CLR_BITS宏的示例:
c 复制代码
#include <stdio.h>

#define VAL_CLR_BITS(val, s, n) ((val) & (~(((1ULL << ((n) - (s) + 1)) - 1) << (s))))

int main(void) 
{
    unsigned int val = 0xFFFF; // 16位宽的值,初始全为1
    unsigned int s = 4; // 起始位
    unsigned int n = 7; // 结束位
    
    unsigned int result = VAL_CLR_BITS(val, s, n);
    
    printf("Original value: 0x%X\n", val);
    printf("After masking: 0x%X\n", result);
    
    return 0;
}

这段代码将输出val的原始值和应用VAL_CLR_BITS宏后的结果。在这个示例中,val初始值为0xFFFF,即所有位都是1。应用VAL_CLR_BITS(val, 4, 7)后,结果将是0xFF0F,即第4位到第7位被清零。

相关推荐
whoarethenext1 分钟前
使用 C++ 实现 MFCC 特征提取与说话人识别系统
开发语言·c++·语音识别·mfcc
ITfeib10 分钟前
Flutter
开发语言·javascript·flutter
想躺平的咸鱼干1 小时前
Volatile解决指令重排和单例模式
java·开发语言·单例模式·线程·并发编程
Owen_Q1 小时前
Denso Create Programming Contest 2025(AtCoder Beginner Contest 413)
开发语言·算法·职场和发展
·云扬·1 小时前
【Java源码阅读系列37】深度解读Java BufferedReader 源码
java·开发语言
liulilittle2 小时前
C++ i386/AMD64平台汇编指令对齐长度获取实现
c语言·开发语言·汇编·c++
Thomas_YXQ2 小时前
Unity URP法线贴图实现教程
开发语言·unity·性能优化·游戏引擎·unity3d·贴图·单一职责原则
Zz_waiting.3 小时前
Javaweb - 10.4 ServletConfig 和 ServletContext
java·开发语言·前端·servlet·servletconfig·servletcontext·域对象
Touper.3 小时前
JavaSE -- 泛型详细介绍
java·开发语言·算法
sun0077003 小时前
std::forward作用
开发语言·c++·算法