指针类型和存储区的关系详解

上传人:豆浆 文档编号:31232690 上传时间:2018-02-06 格式:DOC 页数:3 大小:30KB
返回 下载 相关 举报
指针类型和存储区的关系详解_第1页
第1页 / 共3页
指针类型和存储区的关系详解_第2页
第2页 / 共3页
指针类型和存储区的关系详解_第3页
第3页 / 共3页
亲,该文档总共3页,全部预览完了,如果喜欢就下载吧!
资源描述

《指针类型和存储区的关系详解》由会员分享,可在线阅读,更多相关《指针类型和存储区的关系详解(3页珍藏版)》请在金锄头文库上搜索。

1、指针类型和存储区的关系详解 一、存储类型与存储区关系data - 可寻址片内 rambdata - 可位寻址的片内 ramidata - 可寻址片内 ram,允许访问全部内部 rampdata - 分页寻址片外 ram (MOVX R0) (256 BYTE/页)xdata - 可寻址片外 ram (64k 地址范围)code - 程序存储区 (64k 地址范围),对应 MOVC DPTR二、指针类型和存储区的关系对变量进行声明时可以指定变量的存储类型如:uchar data x 和 data uchar x 相等价都是在内 ram 区分配一个字节的变量。同样对于指针变量的声明,因涉及到指针变

2、量本身的存储位置和指针所指向的存储区位置不同而进行相应的存储区类型关键字的使用如:uchar xdata *data pstr是指在内 ram 区分配一个指针变量(*号后的 data 关键字的作用),而且这个指针本身指向 xdata 区(*前 xdata 关键字的作用),可能初学 C51 时有点不好懂也不好记。没关系,我们马上就可以看到对应“*”前后不同的关键字的使用在编译时出现什么情况。.uchar xdata tmp10; /在外 ram 区开辟 10 个字节的内存空间,地址是外 ram 的 0x00000x0009.第 1 种情况:uchar data * data pstr;pstr=

3、tmp;首先要提醒大家这样的代码是有 bug 的, 他不能通过这种方式正确的访问到 tmp 空间。 为什么?我们把编译后看到下面的汇编代码:MOV 0x08,#tmp(0x00) ;0x08 是指针pstr 的存储地址看到了吗!本来访问外 ram 需要 2 byte 来寻址 64k 空间,但因为使用 data 关键字(在*号前的那个),所以按 KeilC 编译环境来说就把他编译成指向内 ram 的指针变量了,这也是初学 C51 的朋友们不理解各个存储类型的关键字定义而造成的 bug。特别是当工程中的默认的存储区类为 large 时,又把 tmp10 声明为 uchar tmp10 时,这样的

4、bug 是很隐秘的不容易被发现。第 2 种情况:uchar xdata * data pstr;pstr = tmp;这种情况是没问题的,这样的使用方法是指在内 ram 分配一个指针变量(*号后的 data 关键字的作用),而且这个指针本身指向 xdata 区(*前 xdata 关键字的作用)。编译后的汇编代码如下。MOV 0x08,#tmp(0x00) ;0x08 和 0x09 是在内 ram 区分配的 pstr 指针变量地址空间MOV 0x09,#tmp(0x00)这种情况应该是在这里所有介绍各种情况中效率最高的访问外 ram 的方法了,请大家记住他。第 3 种情况:uchar xdata

5、 * xdata pstr;pstr=tmp; 这中情况也是对的,但效率不如第 2 种情况。编译后的汇编代码如下。MOV DPTR, #0x000A ;0x000A,0x000B是在外 ram 区分配的 pstr 指针变量地址空间MOV A, #tmp(0x00)MOVX DPTR, AINC DPTRMOV A, #tmp(0x00)MOVX DPTR, A这种方式一般用在内 ram 资源相对紧张而且对效率要求不高的项目中。第 4 种情况:uchar data * xdata pstr;pstr=tmp;如果详细看了第 1 种情况的读者发现这种写法和第 1 种很相似,是的,同第 1 种情况一

6、样这样也是有 bug 的,但是这次是把pstr 分配到了外 ram 区了。编译后的汇编代码如下。MOV DPTR, #0x000A ;0x000A 是在外ram 区分配的 pstr 指针变量的地址空间MOV A, #tmp(0x00)MOVX DPTR, A第 5 种情况:uchar * data pstr;pstr=tmp;大家注意到*前的关键字声明没有了,是的这样会发生什么事呢?下面这么写呢!对了用齐豫的一首老歌名来说就是 “请跟我来”,请跟我来看看编译后的汇编代码,有人问这不是在讲 C51 吗? 为什么还要给我们看汇编代码。C51 要想用好就要尽可能提升 C51 编译后的效率,看看编译后

7、的汇编会帮助大家尽快成为生产高效 C51 代码的高手的。还是看代码吧!MOV 0x08, #0X01 ;0x080x0A 是在内 ram 区分配的 pstr 指针变量的地址空间MOV 0x09, #tmp(0x00)MOV 0x0A, #tmp(0x00)注意:这是新介绍给大家的,大家会疑问为什么在前面的几种情况的 pstr 指针变量都用 2 byte 空间而到这里就用 3 byte空间了呢?这是 KeilC 的一个系统内部处理,在 KeilC 中一个指针变量最多占用 3 byte 空间,对于没有声明指针指向存储空间类型的指针,系统编译代码时都强制加载一个字节的指针类型分辩值。具体的对应关系可

8、以参考 KeilC 的 help 中 C51 Users Guide。第 6 种情况:uchar * pstr;pstr=tmp;这是最直接最简单的指针变量声明,但他的效率也最低。还是那句话,大家一起说好吗!编译后的汇编代码如下。MOV DPTR, #0x000A ;0x000A0x000C是在外 ram 区分配的 pstr 指针变量地址空间MOV A, #0x01MOV DPTR, AINC DPTRMOV A, #tmp(0x00)MOV DPTR, AINC DPTRMOV A, #tmp(0x00)MOVX DPTR, A这种情况很类似第 5 种和第 3 种情况的组合,既把 pstr

9、分配在外 ram 空间了又增加了指针类型的分辨值。小结一下:大家看到了以上的 6 种情况,其中效率最高的是第 2 种情况,既可以正确访问 ram区又节约了代码,效率最差的是第 6 种,但不是说大家只使用第 2 种方式就可以了,还要因情况而定,一般说来应用 51 系列的系统架构的内部 ram 资源都很紧张,最好大家在定义函数内部或程序段内部的局部变量使用内 ram,而尽量不要把全局变量声明为内 ram 区中。所以对于全局指针变量我建议使用第 3 种情况,而对于局部的指针变量使用第 2 种方式。C51 是很灵活的,也很好理解和使用,但要成为笑傲江湖的一代高手还是要多想多练,没有实际项目的锻炼是不容易提高的。希望这篇文章对大家一点用处。

展开阅读全文
相关资源
相关搜索

当前位置:首页 > 行业资料 > 其它行业文档

电脑版 |金锄头文库版权所有
经营许可证:蜀ICP备13022795号 | 川公网安备 51140202000112号