<meta name="keywords" content="必胜时时彩开户,keywords" /> 第九课 C51运算符和表达式(指针和地址运算符) 联系我们

第九课 C51运算符和表达式(指针和地址运算符)

在第 3 课我们学习数据类型时,学习过指针类型,知道它是一种存放指向此外一个数据的地址的变量类型。指针是单片机C语言中一个很是主要的看法,也是学习单片机C语言中的一个难点。关于指针将会在第九课中做详细的解说。在这里我们先来明确一下单片机C语言中供应的两个专门用于指针和地址的运算符:

*    取内容

&    取地址取内容和地址的浅易形式划分为:

变量  =  *  指针变量 指针变量  =  &  目的变量

取内容运算是将指针变量所指向的目的变量的值赋给左边的变量;取地址运算是将目的变量的地址赋给左边的变量。要重视的是:指针变量中只能存放地址(也就是指针型数据), 浅易情形下不要将非指针类型的数据赋值给一个指针变量。

下面来看一个例子,并用一个图表和实例去质朴明确指针的应用措施和寄义。

设有两个 unsigned  int  变量    ABC 处 CBA  存放在 0x0028,0x002A 中 尚有一个指针变量    portA  存放在 0x002C 中 那么我们写这样一段法式模范模范去看看*,&的运算效果

unsigned  int  data  ABC  _at_  0x0028; unsigned  int  data  CBA  _at_  0x002A; unsigned  int  data  *Port  _at_  0x002C;

#include  <at89x51.h>

#include  <stdio.h>

void  main(void)

{

SCON  =  0x50;  //串行口要领 1,允许吸收 TMOD  =  0x20;  //准时器 1 准时要领 2

TH1  =  0xE8;  //11.0592MHz  1200 波特率 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 字节

强迫类型转换运算符 不知你们能否有自己去试着编一些法式模范模范,从中能否有遇到一些效果?泉源学习时我就遇到过

这样一个效果:两个纷歧样数据类型的数在相互赋值时会泛起纰谬的值。以下面的一段小法式模范模范:

void  main(void)

{

unsigned  char  a;

unsigned  int  b;

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  <at89x51.h>

#include  <stdio.h>

void  main(void)

{

char  xdata  *  XROM;

char  a;

int  Aa  =  0xFB1C;

long  Ba  =  0x893B7832;

float  Ca  =  3.4534;

SCON  =  0x50;  //串行口要领 1,允许吸收 TMOD  =  0x20;  //准时器 1 准时要领 2

TH1  =  0xE8;  //11.0592MHz  1200 波特率 TL1  =  0xE8;

TI  =  1;

TR1  =  1;  //启动准时器

XROM=(char  xdata  *)  0xB012;  //给指针变量赋 XROM 初值

*XROM  =  ‘R’;  //给 XROM 指向的相对地址赋值

a  =  *((char  xdata  *)  0xB012);  //分歧于 a  =  *XROM

printf  (“%bx  %x  %d  %c  \n”,(char)  Aa,  (int)  Ba,(int)Ca,  a);//转换类型并输入

while(1);

}

法式模范模范运转效果:1c  7832  3  R 不才面这段法式模范模范中,能很清晰到到种种类型阻拦强迫类型转换的基本应用措施,法式模范模范中先

在外部数据存储器 XDATA 中界说了一个字符型指针变量 XROM,当用 XROM=(char  xdata  *)

0xB012 这一语句时,便把 0xB012 这个地址指针赋于了 XROM,如你用 XROM 则会是非法的, 这类措施特殊合适于用标识符来存取相对地址,如在法式模范模范前用#define  ROM  0xB012  这样的 语句,在法式模范模范中便可以用下面的措施用 ROM 对相对地址 0xB012 阻拦存取操作了。运算符的优先级诠释表格能在笔者的 本教程附录 中检查。