单片机C语言的运算符和表达式
运算符是完成某种特定运算的符号,运算符按其表达式中与运算符的关系可分为单目运算符,双目运算符和三目运算符。单目运算符就是指只有一个运算对象,双目运算符就要求有两个运算对象,三目运算符则要三个运算对象。
表达式则是由运算及运算对象所组成的具有特定含义的式子,根据运算符种类不同,可以产生四种表达式,分别是算术表达式、赋值表达式、关系表达式和逻辑表达式。
赋值运算符与赋值表达式
简单赋值运算符和表达式,简单赋值运算符记为“=”。使用“=”的赋值语句格式如下:
变量=表达式;
它的作用就是把数据赋给变量,如x=10;利用赋值运算符将一个变量与一个表达式连接起来的式子为赋值表达式,在表达式后面加“;”便构成了赋值语句。
示例如下:
a = 0xFF; //将常数十六进制数FF赋于变量a
b = c = 33; //同时赋值给变量b,c
d = e; //将变量e的值赋于变量d
f = a+b; //将变量a+b的值赋于变量f
由上面的例子可以知道赋值语句的意义就是先计算出“=”右边的表达式的值,然后将得到的值赋给左边的变量。而且右边的表达式可以是一个赋值表达式。
如果赋值运算符两边的数据类型不相同,系统将自动进行类型转换,即把赋值号右边的类型换成左边的类型。具体规定如下:
(1)实型赋予整型,舍去小数部分。
(2)整型赋予实型,数值不变,但将以浮点形式存放,即增加小数部分(小数部分的值为0)。
(3)字符型赋予整型,由于字符型为一个字节,而整型为二个字节,故将字符的ASCII码值放到整型量的低八位中,高八位为0。
(4)整型赋予字符型,只把低八位赋予字符量。
提示
往往会出现“==”与“=”这两个符号混淆的错误原码,问为何编译报错,往往就是错在if(a=x)之类的语句中,错将“=”用为“==”。“==”符号是用来进行相等关系运算。
算术运算符与算术表达式
C51中的算术运算符如下:
+ 加或取正值运算符
++ 自增运算符
- 减或取负值运算符
— 自减运算符
* 乘运算符
/ 除运算符
% 取余运算符
其中只有取正值和取负值运算符是单目运算符,其它则都是双目运算符,除法运算符和一般的算术运算规则有所不同,如是两浮点数相除,其结果为浮点数,如10.0/20.0所得值为0.5,而两个整数相除时,所得值就是整数,如7/3,值为2。
算术表达式是由算术运算符和括号连接起来的式子。算术表达式的形式如下:
表达式1 算术运算符 表达式2
如:a+b*(10-a),(x+9)/(y-a)
运算符与有优先级和结合性,可用用括号“()”来改变优先级。
关系运算符与关系表达式
C51语言中有六种关系运算符:
> 大于
< 小于
>= 大于等于
<= 小于等于
== 等于
!= 等于
关系运算符的优先级别:前四个具有相同的优先级,后两个也具有相同的优先级,但是前四个的优先级要高于后两个。
当两个表达式用关系运算符连接起来时,这时就是关系表达式。关系表达式通常是用来判别某个条件是否满足。关系表达式的形式如下:
表达式1 关系运算符 表达式2
如:I(J=3),J+I>J。
要注意的是用关系运算符的运算结果只有0和1两种,也就是逻辑的真与假,当指定的条件满足时结果为1,不满足时结果为0。
逻辑运算符与逻辑表达式
C51的逻辑运算符有如下三个:
&& 逻辑与
|| 逻辑或
! 逻辑非
用逻辑运算符将关系表达式或逻辑量连接起来就是逻辑表达式了。逻辑表达式的一般形式为:
逻辑与:条件式1 && 条件式2
逻辑或:条件式1 || 条件式2
逻辑非:!条件式
逻辑与表达式的值:当条件式1“与”条件式2都为真时结果为真(非0值),否则为假(0值)。也就是说运算会先对条件式1进行判断,如果为真(非0值),则继续对条件式2进行判断,当结果为真时,逻辑运算的结果为真(值为1),如果结果不为真时,逻辑运算的结果为假(0值)。如果在判断条件式1时就不为真的话,就不用再判断条件式2了,而直接给出运算结果为假。
逻辑或表达式的值:只要二个运算条件中有一个为真时,运算结果就为真,只有当条件式都不为真时,逻辑运算结果才为假。
逻辑非表达式的值:如果条件式的运算值为真,进行逻辑非运算后则结果变为假;条件式运算值为假时最后逻辑结果为真。
逻辑运算符的优先级别,从高到低依次为:!(逻辑非)→&&(逻辑与)→||(逻辑或)。
位运算符
位运算符的作用是按位对变量进行运算,但是并不改变参与运算的变量的值。C51中共有6种位运算符,如下所示:
& 按位与
| 按位或
^ 按位异或
~ 按位取反
<< 左移
>> 右移
位运算一般的表达形式如下:
变量1 位运算符 变量2
位运算符的优先级从高到低依次是:“~”(按位取反)→“<<”(左移)→“>>”(右移)→“&”(按位与)→“^”(按位异或)→“|”(按位或)。
例:若a=0x54,b=0x3b
则c=a & b=0x10;
例:若a=0xea
则a=a<<2=0xa8;
复合赋值运算符
复合赋值运算符就是在赋值运算符“=”的前面加上其他运算符。C51中的复合赋值运算符如下:
+= 加法赋值 >>= 右移位赋值
-= 减法赋值 &= 逻辑与赋值
*= 乘法赋值 | = 逻辑或赋值
/= 除法赋值 ^= 逻辑异或赋值
%= 取模赋值 <<= 左移位赋值
复合运算的一般形式为:
变量 复合赋值运算符 表达式
复合运算就是变量与表达式先进行运算符所要求的运算,再把运算结果赋值给参与运算的变量,这是C语言中一种简化程序的一种方法。凡是二目运算都可以用复合赋值运算符去简化表达。
例如:
a+=56等价于a=a+56
y/=x+9等价于y=y/(x+9)
很明显采用复合赋值运算符会降低程序的可读性,但这样却可以使程序代码简单化,并能提高编译的效率。
逗号运算符
在前面的一些程序中,如“int a,b,c”,逗号用于分隔表达式用。但在C语言中,逗号还是一种特殊的运算符,它可以将两个或多个表达式连接起来,形成逗号表达式。逗号表达式的一般形式为:
表达式1,表达式2,表达式3……表达式n
用逗号运算符组成的表达式在程序运行时,是从左到右计算出各个表达式的值,而整个用逗号运算符组成的表达式的值等于最右边表达式的值,就是“表达式n”的值。
例如
void main()
{
int a=2,b=4,c=6,x,y;
y=(x=a+b),(b+c);
printf("y=%d,x=%d",y,x);
}
程序中的赋值关系如下:
a←2,b←4,c←6,x←0,y←0
x←a+b,y←b+c
本例中,y等于整个逗号表达式的值,也就是表达式2的值,x是第1个表达式的值。
在实际的应用中大部分情况下,使用逗号表达式的目的只是为了分别得到名个表达式的值,而并不一定要得到和使用整个逗号表达式的值。
要注意的是,并不是在程序的任何位置出现的逗号,都可以认为是逗号运算符。如函数中的参数,同类型变量的定义中的逗号只是用来间隔之用而不是逗号运算符。
条件运算符
条件运算符“?:”是一个三目运算符,它可以把三个表达式连接构成一个条件表达式。
条件表达式的一般形式如下:
逻辑表达式? 表达式1:表达式2
条件表达式中逻辑表达式的类型可以与表达式1和表达式2的类型不一样。条件运算符的作用简单来说就是根据逻辑表达式的值选择使用表达式的值。当逻辑表达式的值为真时(非0值)时,整个表达式的值为表达式1的值;当逻辑表达式的值为假(值为0)时,整个表达式的值为表达式2的值。
下面我们来看一个例子:
若a=1,b=2,这时我们要求是取a、b两数中的较小的值放入min变量中,也许你会这样写:
if (a
min = a;
else
min = b; //当a
用条件运算符去构成条件表达式如下:
min = (a
很明显它的结果和含意都和上面的一段程序是一样的,但是代码却比上一段程序少很多,编译的效率也相对要高,但有着和复合赋值表达式一样的缺点就是可读性相对效差。在实际应用时可以加上适当的注解,这样可以有助于程序的调试和编写,也便于日后的修改读写。
指针和地址运算符
C语言中提供的两个专门用于指针和地址的运算符:
* 取内容
& 取地址
取内容和地址的一般形式分别为:
变量 = * 指针变量
指针变量 = & 目标变量
取内容运算是将指针变量所指向的目标变量的值赋给左边的变量;取地址运算是将目标变量的地址赋给左边的变量。
例如:
sizeof运算符
sizeof是用来求数据类型、变量或是表达式的字节数的一个运算符,与“=”之类的运算符在程序执行后才能计算出结果不同,它直接在编译时产生结果。语法如下:
sizeof (数据类型)
sizeof (表达式)
例如运行以下语句
printf("char 是多少个字节? %bd 字节\n",sizeof(char));
printf("long 是多少个字节? %bd 字节\n",sizeof(long));
结果为:
char是多少个字节? 1 字节
long是多少个字节? 4 字节
3.2.11 强制类型转换运算符
C语言中的圆括号“()”也可以作为一种运算符来使用,即强制类型转换运算符。它的作用是将表达式或变量的类型强制转换为指定的类型。
在C51程序中进行算术运算时,需要注意数据类型的转换,数据类型转换分为隐式转换和显式转换。隐式转换是在对程序进行编译时由编译器自动处理的,并且只有基本数据类型(即char、int、long和float)可以进行隐式转换。其他数据类型不能进行隐式转换,例如,我们不能把一整型数利用隐式转换赋值给一个指针变量,在这种情况下就必须利用强制类型转换运算符来进行显式转换。
强制类型转换运算符的一般使用形式为:
(类型)(表达式)
例如,预先在8051单片机的片外数据存储器(xdata)中定义了一个字符型指针变量px,如果想给这个指针变量赋一初值0xB000,可以通过强制类型转换的方式,如下:
px=(char xdata*) 0xB000
单片机为什么一直用C语言,不用其他编程语言?只有学过的知道
单片机为什么还在用C语言编程?答案是:C语言是最适合单片机编程的高级语言。
这个问题的意思应该是:现在有很多很好用的高级语言,如java,python,VC等等,为什么这些语言不能用来编写单片机程序呢?那么这个问题的答案就是:不能不能,而是不合适。
一、单片机编程的特点
对单片机编程来说,首先要考虑的是单片机的程序空间和数据空间都是有限的,所以要让程序尽量短小精悍,以节省程序占用的存储空间。
第二、单片机编程的一个主要对象是对单片机的端口和内部寄存器的操作和配置,这个需要比较精确的时序控制。
第三、单片机算法运算中,尽量使用加法、减法、移位运算,因为乘法和除法运算会非常费时间,尤其是除法,会耗费很多时间,这对于速度本身就有限制的单片机来说,是一个很大的负担。
二、高级语言编写单片机程序的缺陷
高级语言可以实现更为优化的算法,更为方便的执行方案,但是,高级语言对程序存储空间的占用要比汇编和C语言多很多。这是最致命的一点,单片机有限的存储空间需要靠精打细算来设计程序,根本经不起高级语言臃肿的代码体积。
高级语言无法实现精确的时序控制。
三、C语言是一个折中选择
其实用C语言开发单片机也是一个折中方案,因为最适合单片机开发的编程语言实在太过晦涩难懂,并且每一种单片机的汇编指令有很大区别,所以想把一个程序从这种单片机移植到另一种单片机简直是痴心妄想,还不如重新写一遍程序。
而C语言代码执行效率高,也比较精简,更便于移植......所以在现今的单片机编程语言中,C语言才会占据绝对主导地位
作者丨悟空问答:老马识途单片机
想要在程序员生涯内有更高的成就的话,C/C++就是一个既可以强化思维能力,又可以打好编程基础的编程语言,你想要做软件开发,成为核心程序员的话,学习C/C++的话笔者有一个C/C++的编程俩千人羣(Q艘索:C/C++编程学习13群 )你如果感觉自学C语言/C++语言有困难的话,有兴趣学习或者了解一下C/C++编程的小伙伴就可以进来交流。
相关问答
为什么 单片机 在进行 乘法 运算不产生进位?_作业帮[最佳回答]你是说51吗?因为不可能产生进位呀,8位乘8位乘积放在AB中,不可能超过16位,所以不会产生进位的,因此进位C始终处于复位状态.但当乘积大于255时会影响...
单片机乘法 究竟怎么运算?51单片机乘法指令是MULABA*B结果高八位为B,低八位为A。51单片机乘法指令是MULABA*B结果高八位为B,低八位为A。
单片机乘法 指令功能详解?乘法指令是单片机指令中最常用的一种指令,乘法指令的功能是让CPU进行乘法计算,得到乘法结果。根据乘法指令的实现的具体方式,乘法指令可以分为以下几类:1....
单片机乘法 指令:假定(A)=50H(B)=0A0H.执行指令MULAB后,累加...[最佳回答]50H*0A0H=320H你可以把十六进制都换成十进制来计算!即80*10=800换成十六进制就是320H执行指令之后,A=20H,B=03H,(结果高八位放在寄存器B,低八...
51 单片机乘法 指令MUL,求大虾?A就是累加器ACCB是B寄存器。这个执行完以后,B放结果高8位,A放结果低8位你的数据,执行以后A=56H,B=1CHBA=1C56HA就是累加器ACCB是B寄存器。这个执行完以后,B放...
单片机 中计算器程序 c 语言加减乘除肿么写?-ZOL问答#include
1.原码表示方法的优点,是在数的真值和它的原码表示之间的对应关系简单,相互转换容易,用原码实现乘除运算的规则简单.缺点是用原码实现加减运算很不方便.要比...
51 单片机 中执行一条 乘法 指令或者除法指令需要多少个机器周期?一条乘法指令或者一条除法指令都需要4个机器周期。指令周期是不确定的,因为她和该条指令所包含的机器周期有关。一个指令周期=1个(或2个或3个或4个)机器周...
单片机 ds是什么意思?单片机ds就是DSP芯片。DSP芯片是数字处理芯片,可以在一个时钟周期里完成一个MAC,就是16位乘上16位数,其积再加上一个32位数这两个操作一个时钟周期完成。在ds...
松翰 单片机 SN8P2711B怎么没有 乘法 指令?8bit和16bit都不会包含乘法指令,成本太高了。可使用a×bwhile(b--){a+=a;}不过注意a是否会溢出8bit和16bit都不会包含乘法指令,成本太高了。可使用a×bwhi.....