插手珍藏夹
 公司简介 || 产物列表 || 采办编制 || 汇款订单提交 || 论坛 || Universal Programmer || 淘宝网店 || 诚聘英才 || 单片机名目斥地
指针类型和存储区的干系详解
    
一、存储类型与存储区干系

    data     --->    可寻址片内ram
    bdata    --->    可位寻址的片内ram
    idata    --->    可寻址片内ram,答应拜候全数内部ram
    pdata    --->    分页寻址片外ram (MOVX @R0) (256 BYTE/页)
    xdata    --->    可寻址片外ram (64k 地址范畴)
    code     --->    法式存储区 (64k 地址范畴),对应MOVC @DPTR

二、指针类型和存储区的干系

    对变量进行声明时能够指定变量的存储类型如:
    uchar data x和data uchar x相等价都是在内ram区分派一个字节的变量。

    同样对于指针变量的声明,因涉及到指针变量本身的存储位置和指针所指向的存储区位置不合而进行响应的存储区类型环节字的
使用如:

    uchar xdata * data pstr

    是指在内ram区分派一个指针变量("*"号后的data环节字的传染感动),并且这个指针本身指向xdata区("*"前xdata环节字的传染感动),
可能初学C51时有点不好懂也不好记。不妨,咱们登时就能够看到对应“*”前后不合的环节字的使用在编译时出现什么环境。

    ......
    uchar xdata tmp[10];    //在外ram区斥地10个字节的内存空间,地址是外ram的0x0000-0x0009
    ......

    第1种环境:

    uchar data * data pstr;
    pstr=tmp;

    起首要提示大师多么的代码是有bug的, 他不能经由这类编制精确的拜候到tmp空间。 为什么?咱们把编译后看到下面的汇编
代码:

    MOV 0x08,#tmp(0x00)        ;0x08是指针pstr的存储地址

    看到了吗!本来拜候外ram需要2 byte来寻址64k空间,但由于使用data环节字(在"*"号前的阿谁),所以按KeilC编译环境来说
就把他编译成指向内ram的指针变量了,这也是初学C51的伴侣们不睬解各个存储类型的环节字定义而构成的bug。出格是当工程中的
默认的存储区类为large时,又把tmp[10] 声明为uchar tmp[10] 时,多么的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 * xdata pstr;
    pstr=tmp;

    这中环境也是对的,但效率不如第2种环境。编译后的汇编代码如下。

    MOV DPTR, #0x000A        ;0x000A,0x000B是在外ram区分派的pstr指针变量地址空间
    MOV A, #tmp(0x00)
    MOV @DPTR, A
    INC DPTR
    MOV A, #tmp(0x00)
    MOVX @DPTR, A

    这类编制一般用在内ram资底细对严峻并且对效率要求不高的名目中。

    第4种环境:

    uchar data * xdata pstr;
    pstr=tmp;

    若是细致看了第1种环境的读者发觉这类写法和第1种很类似,是的,同第1 种环境一样多么也是有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
编译后的效率,看看编译后的汇编会赞助大师尽快成为生产高效C51代码的高手的。仍是看代码吧!

    MOV 0x08, #0X01            ;0x08-0x0A是在内ram区分派的pstr指针变量的地址空间
    MOV 0x09, #tmp(0x00)
    MOV 0x0A, #tmp(0x00)

    留意:这是新引见给大师的,大师会疑问为什么在后面的几种环境的pstr指针变量都用2 byte空间而到这里就用3 byte空间了
呢?这是KeilC的一个系统内部处置,在KeilC中一个指针变量最多占用 3 byte空间,对于没有声明指针指向存储空间类型的指针,
系统编译代码时都强制加载一个字节的指针类型分辩值。具体的对应干系能够参考KeilC的help中C51 User's Guide。

    第6种环境:

    uchar * pstr;
    pstr=tmp;

    这是最间接最简单的指针变量声明,但他的效率也最低。仍是那句话,大师一路说好吗!编译后的汇编代码如下。

    MOV DPTR, #0x000A        ;0x000A-0x000C是在外ram区分派的pstr指针变量地址空间
    MOV A, #0x01
    MOV @DPTR, A
    INC DPTR
    MOV DPTR, #0x000A
    MOV A, #tmp(0x00)
    MOV @DPTR, A
    INC DPTR
    MOV A, #tmp(0x00)
    MOVX @DPTR, A

    这类环境很类似第5种和第3种环境的组合,既把pstr分派在外ram空间了又添加了指针类型的分辩值。

    小结一下:大师看到了以上的6种环境,此中效率较高的是第2种环境,既能够精确拜候ram区又节约了代码,效率最差的是第 6
种,但不是说大师只使用第2种编制就能够了,又一次要因环境而定,一般说来使用51系列的系统架构的内部ram本钱都很严峻,较好大师
在定义函数内部或法式段内部的局部变量使用内ram,而尽量不要把全局变量声明为内ram区中。所以对于全局指针变量我建议使用第
3 种环境,而对于局部的指针变量使用第2种编制。

    C51是很矫捷的,也很好理解和使用,但要成为笑傲江湖的一代高手仍是要多想多练,没有现实项方针熬炼是不容易提高的。希
望这篇文章对大师一点用处。
      

--->>>前去首页



专业生产 单片机 斥地板 进修板 电子制造散件 电子制造套件 GSM模块 GPS模块 GPRS模块 GPRS MODEM DTU 短信猫 产物
以上部门内容转载于网上,如有涉及到版权问题,请即通知本人删除 浙ICP备11001927号 MSN:hificat@hotmail.com
联系地址:浙江省杭州市西湖科技园西园七路3号4层 邮政编码:310011 Email:hificat@163.com
德律总机:0571-87615070   产物征询:转分机1   技术支撑:转分机2   传真:转分机3   手机:13185018567
发卖QQ:1198450005 发卖征询 技术QQ熊工:1275636157 技术征询 技术QQ徐工:420951892 技术征询
杭州澳门新濠天地电子有限公司 版权所有 COPYRIGHT2003——2011 HANGZHOU KinCony ELECTRONICS CO.,LTD All rights reserved