单片机C环境下位操作的实现方法
c语言既有高级语言的各种特点,又可对硬件进行操作,并对进行结构化程序设计,用c语言编写的程序较容易移植,它们可生成简洁可靠的目标代码,在代码效率和代码执行速度上完全可以和汇编媲美。采用c语言进行单片机编程是嵌入式程序设计的发展趋势。但是,在嵌入式控制等领域,经常需要控制某一个二进制位,然而除了keil c51等c环境外,很多单片机c环境都没有扩充对位变量定义的关键字,甚至单片机本身的硬件上也没有对单个位操作的汇编指令,这使得已习惯mcs-51内核单片机keil c51编程的用户都为其c环境不能对位变量进行位操作而烦恼。
1 用“读-修改-写”方法实现对单个位的位操作
本文引用地址:https://www.eepw.com.cn/article/21218.htmansic中,一般采用“读-修改-写”的方法实现单个位的位操作,通过与0“与”操作,将某一位清0。如使i变量的b0位为0,实现方法为i=i&0xfe。通过与1“或”操作,将某一位置1。如使i变量的b0位为1,实现方法为i=i|0x01。通过与1“异或”操作,将某一位取反。如使i变量的b0位取反,实现方法为i=i^0x01。
注意:错误“读-修改-写”方法时不要影响其他位,即某位清零时,其他位与1“与”;某位置1时,其他位与0“或”;取反时,其他位与0“异或”。
为了方便程序设计和增加程序可读性,很多程序员喜欢采用下面的移位方式实现单个位的位操作,语句简练,可读性强,比如在某单片机的b口连接1个发光二极管,其点亮操作方法如下:
#define bit(x) (1<<(x))
#define led 2
使用方法如下:
portb|=bit(led);//将portb第3位置1,点
//亮连接在i/o口的led
该方式下,程序运行时会增加移位操作,生成的代码较大,而且执行时间长,实时性差,一些c环境按如下方式直接定义位,则生成的代码就不会有移位操作:

2 通过位域的方法实现位操作
标准c提供了一种基于结构体的数据结构--位域(bitfield),位域就是把一个存储单元中的二进制划分为几个不同的区域。并说明每个区域的位数。每一个域有一个域名,允许在程序中按域名进行操作,位域的定义格式如下:
struct 位域结构名{
位域列表 };
位域列表格式为:类型说明符 位域名:位域长度如:
struct k{
unsigned
int a:1
unsigned int :2
unsigned int b:3
unsigned int :0 //空域
}k1;
说明:
1)各位依次从低位到高位排列,排满一个存储单元,按地址接着排下一单元;
2)位域可以无域名,但不能被引用,如第二域,这时其只用来填充或调整位置;
3)第四行称空域,目的是将目前存储单元的剩余部分分为一个域,且填充0。
位域的引用很简单,如:
k1.a=1; //置k1的b0位为1
k1.b=7; //将k1的b3-5位置111
通过位域定义位变量,是实现单个位位操作的重要途径和方法,采用位域定义位变量,产生的代码紧凑、高效。定义的方法如下:

通过位域定义位,再通过宏进行定义,可以方便地将keil c51等程序移植到其他c编译器,从而不再为没有位操作而苦恼。
对一个单片机的所有i/o口,通过将位域结构指定到i/o端口地址,i/o口便都可以采用位域进行宏定义,这样,操作i/o口就可以像keil c51编程一样方便。
3 基于位域实现位操作应用举例
很多单片机没有硬件的spi,而很多板级外围器件为spi接口,而且某些外围器件不是标准的spi,即通信的总二进制位数不是8的整数倍,这里编制一个例程,为同时收和发0-16位,全双工方式,具体使用见例程注解。注意:很多单片机使用前要对使用的i/o口进行初始化,clk和din为输出口,dout为输入口,同时这里使用了前面通过读-修改-写宏定义的一个函数get_bit(x,y)。

该spi模块已经成功应用到单片机与max7219和ch451等spi通信上。
对于没有扩展位变量的c语言环境,在汇编下没有对单个位进行位操作执行的mcu,通过位域的方法操作i/o口是最佳的方法,汇编下有单个位的位操作指令mcu。可以嵌入汇编,但是程序的可移植性等性能会下降,建议使用位域的方法。
评论