51系列单片机学习4C编程相关变量赋值和转换.docx
《51系列单片机学习4C编程相关变量赋值和转换.docx》由会员分享,可在线阅读,更多相关《51系列单片机学习4C编程相关变量赋值和转换.docx(16页珍藏版)》请在冰豆网上搜索。
51系列单片机学习4C编程相关变量赋值和转换
学过汇编的朋友都知道汇编对位的处理能力是很强的,但是单片机C语言也能对运算对象进行按位操作,从而使单片机C语言也能具有一定的对硬件直接进行操作的能力。
位运算符的作用是按位对变量进行运算,但是并不改变参与运算的变量的值。
如果要求按位改变变量的值,则要利用相应的赋值运算。
还有就是位运算符是不能用来对浮点型数据进行操作的。
单片机c语言中共有6种位运算符。
位运算一般的表达形式如下:
变量1位运算符变量2位运算符也有优先级,从高到低依次是:
“~”(按位取反)→“<<”(左移)→“>>”(右移)→“&”(按位与)→“^”(按位异或)→“|”(按位或)
表8-1是位逻辑运算符的真值表,X表示变量1,Y表示变量2
XY~X~YX&YX|YX^Y
0011000
0110011
1001011
1100110
表8-1按位取反,与,或和异或的逻辑真值表
利用以前建立起来的实验板,我们来做个实验验证一下位运算是否真是不改变参与变量的值,同时学习位运算的表达形式。
程序很简单,用P1口做运算变量,P1.0-P1.7对应P1变量的最低位到最高位,通过连接在P1口上的LED我们便能直观看到每个位运算后变量是否有改变或如何改变。
程序如下:
#include
voidmain(void)
{
unsignedinta;
unsignedintb;
unsignedchartemp;//临时变量
P1=0xAA;//点亮D1,D3,D5,D7P1口的二进制为10101010,为0时点亮LED
for(a=0;a<1000;a++)
for(b=0;b<1000;b++);//延时
temp=P1&0x7;//单纯的写P1|0x7是没有意义的,因为没有变量被影响,不会被编译
//执行P1|0x7后结果存入temp,这个时候改变的是temp,但P1不会被影响。
//这个时候LED没有变化,仍然是D1,D3,D5,D7亮
for(a=0;a<1000;a++)
for(b=0;b<1000;b++);//延时P1=0xFF;//熄灭LED
for(a=0;a<1000;a++)
for(b=0;b<1000;b++);//延时
P1=0xAA;//点亮D1,D3,D5,D7P1口的二进制为10101010,为0时点亮LED
for(a=0;a<1000;a++)
for(b=0;b<1000;b++);//延时
P1=P1&0x7;//这个时候LED会变得只有D2灭
//因为之前P1=0xAA=10101010
//与0x7位与0x7=00000111
//结果存入P1P1=00000010//位为O时点亮LED,电路看第三课
for(a=0;a<1000;a++)
for(b=0;b<1000;b++);//延时P1=0xFF;//熄灭LED
while
(1);
//大家能根据上面的程序去做位或,左移,取反等等。
}
复合赋值运算符
复合赋值运算符就是在赋值运算符“=”的前面加上其他运算符。
以下是C语言中的复合赋值运算符:
+=加法赋值>>=右移位赋值
-=减法赋值&=逻辑与赋值
*=乘法赋值|=逻辑或赋值
/=除法赋值^=逻辑异或赋值
%=取模赋值-=逻辑非赋值
<<=左移位赋值
复合运算的一般形式为:
变量复合赋值运算符表达式其含义就是变量与表达式先进行运算符所要求的运算,再把运算结果赋值给参与运算的
变量。
其实这是C语言中一种简化程序的一种方法,凡是二目运算都能用复合赋值运算符去简化表达。
例如:
a+=56等价于a=a+56
y/=x+9等价于y=y/(x+9)很明显采用复合赋值运算符会降低程序的可读性,但这样却能使程序代码简单化,并
能提高编译的效率。
对于开始学习C语言的朋友在编程时最好还是根据自己的理解力和习惯去使用程序表达的方式,不要一味追求程序代码的短小。
逗号运算符
如果你有编程的经验,那么对逗号的作用也不会陌生了。
如在VB中“Dima,b,c”的逗号就是把多个变量定义为同一类型的变量,在C也一样,如“inta,b,c”,这些例子说明逗号用于分隔表达式用。
但在C语言中逗号还是一种特殊的运算符,也就是逗号运算符,能用它将两个或多个表达式连接起来,形成逗号表达式。
逗号表达式的一般形式为:
表达式1,表达式2,表达式3……表达式n
这样用逗号运算符组成的表达式在程序运行时,是从左到右计算出各个表达式的值,而整个用逗号运算符组成的表达式的值等于最右边表达式的值,就是“表达式n”的值。
在实际的应用中,大部分情况下,使用逗号表达式的目的只是为了分别得到名个表达式的值,而并不一定要得到和使用整个逗号表达式的值。
要注意的还有,并不是在程序的任何位置出现的逗号,都能认为是逗号运算符。
如函数中的参数,同类型变量的定义中的逗号只是用来间隔之用而不是逗号运算符。
条件运算符
上面我们说过单片机C语言中有一个三目运算符,它就是“?
:
”条件运算符,它要求有三个运算对象。
它能把三个表达式连接构成一个条件表达式。
条件表达式的一般形式如下:
逻辑表达式?
表达式1:
表达式2条件运算符的作用简单来说就是根据逻辑表达式的值选择使用表达式的值。
当逻辑表达式的值为真时(非0值)时,整个表达式的值为表达式1的值;当逻辑表达式的值为假(值为0)时,整个表达式的值为表达式2的值。
要注意的是条件表达式中逻辑表达式的类型可以与表达式1和表达式2的类型不一样。
下面是一个逻辑表达式的例子。
如有a=1,b=2这个时候我们要求是取ab两数中的较小的值放入min变量中,也许你会这样写:
if(a
elsemin=b;//这一段的意思是当a用条件运算符去构成条件表达式就变得简单明了了.
第3课我们学习数据类型时,学习过指针类型,知道它是一种存放指向另一个数据的地址的变量类型。
指针是单片机C语言中一个十分重要的概念,也是学习单片机C语言中的一个难点。
对于指针将会在第九课中做详细的讲解。
在这里我们先来了解一下单片机C语言中供给的两个专门用于指针和地址的运算符:
*取内容;&取地址取内容和地址的一般形式分别为:
变量=*指针变量;指针变量=&目标变量
取内容运算是将指针变量所指向的目标变量的值赋给左边的变量;取地址运算是将目标变量的地址赋给左边的变量。
要注意的是:
指针变量中只能存放地址(也就是指针型数据),一般情况下不要将非指针类型的数据赋值给一个指针变量。
下面来看一个例子,并用一个图表和实例去简单理解指针的使用方法和含义。
设有两个unsignedint变量ABC处CBA存放在0x0028,0x002A中另有一个指针变量portA存放在0x002C中那么我们写这样一段程序去看看*,&的运算结果
unsignedintdataABC_at_0x0028;
unsignedintdataCBA_at_0x002A;
unsignedintdata*Port_at_0x002C;
#include
#include
voidmain(void)
{
SCON=0x50;//串行口方式1,允许接收TMOD=0x20;//定时器1定时方式2
TH1=0xE8;//11.0592MHz1200波特率TL1=0xE8;
TI=1;
TR1=1;//启动定时器
ABC=10;//设初值CBA=20;
Port=&CBA;//取CBA的地址放到指针变量Port
*Port=100;//更改指针变量Port所指向的地址的内容
printf("1:
CBA=%d\n",CBA);//显示此时CBA的值
Port=&ABC;//取ABC的地址放到指针变量Port
CBA=*Port;//把当前Port所指的地址的内容赋给变量CBA
printf("2:
CBA=%d\n",CBA);//显示此时CBA的值
printf("ABC=%d\n",ABC);//显示ABC的值
}
程序初始时
值
地址
说明
0x00
0x002DH
0x00
0x002CH
0x00
0x002BH
0x00
0x002AH
0x0A
0x0029H
0x00
0x0028H
执行ABC=10;向ABC所指的地址0x28H写入10(0xA),因ABC是int类型要占用0x28H和
0x29H两个字节的内存空间,低位字节会放入高地址中,所以0x28H中放入0x00,0x29H中放入0x0A
值
地址
说明
0x00
0x002DH
0x00
0x002CH
0x00
0x002BH
0x00
0x002AH
0x0A
0x0029H
ABC为int类型占用两字节
0x00
0x0028H
执行CBA=20;原理和上一句一样
值
地址
说明
0x00
0x002DH
0x00
0x002CH
0x14
0x002BH
CBA为int类型占用两字节
0x00
0x002AH
0x0A
0x0029H
ABC为int类型占用两字节
0x00
0x0028H
执行Port=&CBA;取CBA的首地址放到指针变量Port
值
地址
说明
0x00
0x002DH
0x2A
0x002CH
CBA的首地址存入Port
0x14
0x002BH
0x00
0x002AH
0x0A
0x0029H
0x00
0x0028H
*Port=100;更改指针变量Port所指向的地址的内容
值
地址
说明
0x00
0x002DH
0x2A
0x002CH
0x64
0x002BH
Port指向了CBA所在地址2AH
0x00
0x002AH
并存入100
0x0A
0x0029H
0x00
0x0028H
其它的语句也是一样的道理,大家能用Keil的单步执行和打开存储器查看器一看,这样
就更不难理解了。
图9-1存储器查看窗
图9-2在串行调试窗口的最终结果
sizeof运算符
看上去这确实是个奇怪的运算符,有点像函数,却又不是。
大家看到size应该就猜到是和大小有关的吧?
是的,sizeof是用来求数据类型、变量或是表达式的字节数的一个运算符,但它并不像“=”之类运算符那样在程序执行后才能计算出结果,它是直接在编译时产生结果的。
它的语法如下:
sizeof(数据类型)
sizeof(表达式)下面是两句应用例句,程序大家能试着编写一下。
printf("char是多少个字节?
½字节\n",sizeof(char));
printf("long是多少个字节?
½字节\n",sizeof(long));
结果是:
char是多少个字节?
1字节
long是多少个字节?
4字节
强制类型转换运算符不知你们是否有自己去试着编一些程序,从中是否有遇到一些问题?
开始学习时我就遇到过
这样一个问题:
两个不一样数据类型的数在相互赋值时会出现不对的值。
如下面的一段小程序:
voidmain(void)
{
unsignedchara;
unsignedintb;
b=100*4;
a=b;
while
(1);
}
这段小程序并没有什么实际的应用意义,如果你是细心的朋友定会发现a的值是不会等于100*4的。
是的a和b一个是char类型一个是int类型,从以前的学习可知char只占一个字节值最大只能是255。
但编译时为何不出错呢?
先来看看这程序的运行情况:
图9-3小程序的运行情况b=100*4就能得知b=0x190,这个时候我们能在Watches查看a的值,对于watches窗口我们在第5课时简单学习过,在这个窗口Locals页里能查看程序运行中的变量的值,也能
在watch页中输入所要查看的变量名对它的值进行查看。
做法是按图中1的watch#1(或watch#2),然后光标移到图中的2按F2键,这样就能输入变量名了。
在这里我们能查看到a的值为0x90,也就是b的低8位。
这是因为执行了数据类型的隐式转换。
隐式转换是在程序进行编译时由编译器自动去处理完成的。
所以有必要了解隐式转换的规则:
1.变量赋值时发生的隐式转换,“=”号右边的表达式的数据类型转换成左边变量的数据类型。
就如上面例子中的把INT赋值给CHAR字符型变量,得到的CHAR将会是INT的低8位。
如把浮点数赋值给整形变量,小数部分将丢失。
2.所有char型的操作数转换成int型。
3.两个具有不一样数据类型的操作数用运算符连接时,隐式转换会按以下次序进行:
如有一操作数是float类型,则另一个操作数也会转换成float类型;如果一个操作数为long类型,另一个也转换成long;如果一个操作数是unsigned类型,则另一个操作会被转换成unsigned类型。
从上面的规则能大概知道有那几种数据类型是能进行隐式转换的。
是的,在单片机c语言中只有char,int,long及float这几种基本的数据类型能被隐式转换。
而其它的数据类型就只能用到显示转换。
要使用强制转换运算符应遵循以下的表达形式:
(类型)表达式
用显示类型转换来处理不一样类型的数据间运算和赋值是十分方便和方便的,特别对指针变量赋值是很有用的。
看一面一段小程序:
#include
#include
voidmain(void)
{
charxdata*XROM;
chara;
intAa=0xFB1C;
longBa=0x893B7832;
floatCa=3.4534;
SCON=0x50;//串行口方式1,允许接收TMOD=0x20;//定时器1定时方式2
TH1=0xE8;//11.0592MHz1200波特率TL1=0xE8;
TI=1;
TR1=1;//启动定时器
XROM=(charxdata*)0xB012;//给指针变量赋XROM初值
*XROM=‘R’;//给XROM指向的绝对地址赋值
a=*((charxdata*)0xB012);//等同于a=*XROM
printf(“%bx%x%d%c\n”,(char)Aa,(int)Ba,(int)Ca,a);//转换类型并输出
while
(1);
}
程序运行结果:
1c78323R在上面这段程序中,能很清楚到到各种类型进行强制类型转换的基本使用方法,程序中先在外部数据存储器XDATA中定义了一个字符型指针变量XROM,当用XROM=(charxdata*)0xB012这一语句时,便把0xB012这个地址指针赋于了XROM,如你用XROM则会是非法的,这种方法特别适合于用标识符来存取绝对地址,如在程序前用#defineROM0xB012这样的语句,在程序中就能用上面的方法用ROM对绝对地址0xB012进行存取操作了。
运算符的优先级说明表格能在笔者的本教程附录中查看。
前面学习了大部分的基本语法,以下所要学习的各种基本语句的语法能说是组成程序的灵魂。
在前面的课程中的例子里,也简单理解过一些语句的使用方法,能看出C语言是一种结构化的程序设计语言。
C语言供给了相当丰富的程序控制语句。
学习掌握这些语句的使用方法也是单片机C语言学习中的重点。
表达式语句是最基本的一种语句。
不一样的程序设计语言都会有不一样的表达式语句,如VB就是在表达式后面加入回车就构成了VB的表达式语句,而在51单片机的C语言中则是加入分号“;”构成表达式语句。
举例如下:
b=b*10;Count++;
X=A;Y=B;
Page=(a+b)/a-1;
以上的都是合法的表达式语句。
在我收到的一些网友的Email中,发现很多开始学习的朋友一般在编写调试程序时忽略了分号“;”,造成程序不能被正常的编译。
我本人的经验是在遇到编译错误时先语法是否有误,这在开始学习时一般会因在程序中加入了全角符号、运算符打错漏掉或没有在后面加“;”。
在C语言中有一个特殊的表达式语句,称为空语句,它仅仅是由一个分号“;”组成。
有时候为了使语法正确,那么就要求有一个语句,但这个语句又没有实际的运行效果那么这时就要有一个空语句。
说起来就像大家在晚自修的时候用书包占位一样,呵呵。
空语句通常用会以下两种使用方法。
(1)while,for构成的循环语句后面加一个分号,形成一个不执行其它操作的空循环体。
我会会常常用它来写等待事件发生的程序。
大家要注意的是“;”号作为空语句使用时,要与语句中有效组成部分的分号相区别,如for(;a<50000;a++);第一个分号也应该算是空语句,它会使a赋值为0(但要注意的是如程序前有a值,则a的初值为a的当前值),最后一个分号则使整个语句行成一个空循环。
若此时a=0,那么for(;a<50000;a++);就相当于for(a=0;a<50000;a++);我本人习惯是写后面的写法,这样能使人更不难读明白。
(2)在程序中为有关语句供给标号,标记程序执行的位置,使相关语句能跳转到要执行的位置。
这会用在goto语句中。
下面的示例程序是简单说明while空语句的使用方法。
硬件的功能很简单,就是在P3.7上接一个开关,当开关按下时P1上的灯会全亮起来。
当然实际应用中按钮的功能实现并没有这么的简单,一般还要进行防抖动处理等。
先在我们的实验板上加一个按钮。
电路图如图10-1。
图10-1加了按钮的实验电路图
程序如下:
#include
voidmain(void)
{unsignedinta;
do
{
P1=0xFF;//关闭P1上的LED
while(P3_7);//空语句,等待P3_7按下为低电平,低电平时执行下面的语句P1=0;//点亮LED
for(;a<60000;a++);//这也是空语句的使用方法,注意a的初值为当前值
}//这样第一次按下时会有一延时点亮一段时间,以后按多久就亮多久
while
(1);//点亮一段时间后关闭再次判断P3_7,如此循环
}
上面的实验电路已加入了RS232串行口电路,只要稍微改变一下,就能变为具有仿真功能的实验电路。
这个改变的关键就是把芯片改用SST89C58,并在芯片中烧入仿真监控程序。
SST89C58同样也是一种51架构的单片机,它具有24K+8K的两个程序存储区,能选择其一做为程序的启动区。
只要把一个叫SOFTICE.HEX的监控程序用支持SST89C58的编程器烧录到芯片中(使用编程器或用CA版的SST89C58烧录SOFTICE的具体方法和文件能参考/),就能把上面的电路升级为MON51仿真实验器。
那么怎么用它和KEIL实现联机仿真呢?
图10-2项目设置菜单
图10-3项目设置首先要在你要仿真的程序项目设置仿真器所使用的驱动,在Debug页中选择对应本仿真器的KeilMon51驱动,如图10中1所示。
图10-3的3是选择在仿真时能使用的工具窗口,如内存显示,断点等等。
按2进行图10-4中的仿真器设置。
设置好串行口号,波特率,晶体震荡器为11.0592M时选38400。
CacheOptions为仿真缓选取后会加快仿真的运行的速度。
设好后编译运行程序就能连接仿真器了,连接成功会出现如图10-5的画面。
如连接不成功就出现图10-6的图,这个时候能先复位电路再按"TryAgain",还不成功连接的话则应检查软件设置和硬件电路。
图10-5中1是指示仿真器的固件版本为F-MON51V3.4版。
点击3中小红点位置时为设置和取消断点,点击2则运行到下一个断点。
图10-7则是变量和存储器的查看。
仿真器在软件大概的使用方法和软件仿真相差不多。