转自:阳光VS心情
嵌入式系统总是要用户对变量或寄存器进行位操作。给定一个整型变量a,写两段代码,第一个设置a的bit3,第二个清除a的bit3。在以上两个操作中,要保持其它位不变。
这是一个嵌入式面试常考的一个问题,主要考察宏定义的使用和嵌入式里最常用的设置和清除寄存器某一位时常用的操作(也叫掩码),具体可查看C Primer Plus一书中第15章位操作的15.3 C按位运算符章节。
参考答案如下:
#define BIT3 (0x01 << 3)
static int a;
void set_bit3(void) {
a |= BIT3;
}
void clear_bit3(void) {
a &= ~BIT3;
}
以上的代码能够实现功能,但个人感觉移植性不太好。如果想更换一个位,比如设置a的bit4和清除a的bit4,那么就需要重新修改代码,需要将代码中所有的数字3变成数字4,这样需要修改的部分一共有6处,如果程序调用这两个函数的地方比较多,那么需要修改的位置就更多了。
#define BIT4 (0x01 << 4)
static int a;
void set_bit4(void) {
a |= BIT4;
}
void clear_bit3(void) {
a &= ~BIT4;
}
自己想了一下,写了如下代码,全部用宏定义实现
#include <stdio.h>
#include <stdlib.h>
// 带参数的宏定义参数一定要加括号
#define SET_BIT(REG, BIT) ((REG) |= (BIT))
#define CLEAR_BIT(REG, BIT) ((REG) &= ~(BIT))
#define BIT(n) (0x00000001u << (n)) // 注意:这里n一定要加括号
static int a;
int main(void) {
SET_BIT(a, BIT(3));
itoa(a, str3, 2); // 2即是代表转换为2进制
printf("%d的二进制是: %s\n", a, str3);
CLEAR_BIT(a, BIT(3));
itoa(a, str4, 2); // 2即是代表转换为2进制
printf("%d的二进制是: %s\n", a, str4);
return 0;
}
如果这样写的话,需要修改的时候直接把函数内宏定义SET_BIT
和CLEAR_BIT
中的数字3修改成数字4就可以了,这样相比之前的代码最少省了6个步骤的修改,并且不需要增加函数,节省了程序的空间。
不知道我写的宏定义有没有问题,欢迎探讨。