📄 c51中关于指针的种种用法(学单片机必看)--云清燕.htm
字号:
<TD height=0>
<TABLE cellSpacing=0 cellPadding=0 width="100%" border=0>
<TBODY>
<TR>
<TD><SPAN id=ob_logd82231></SPAN>
<DIV class=postText>#pragma src
<P>#i nclude <reg51.H></P>
<P><BR>f(){}<BR>f1(){}<BR>f2(){}</P>
<P>main()<BR>{<BR>
{<BR> int
x;<BR> int *px;</P>
<P>
//下面这些表示虽然很烦,但是生成的代码却及其简洁:(黑体部分实际使用过)</P>
<P> //将 xdata 型指针
0x4000 赋给 px<BR>
px=(int xdata *)0x4000;</P>
<P> //表示从 xdata
0x4000处取一个 char
给x<BR>
<STRONG>x=*((char xdata *)0x4000);
(可以将0X4000处,改成一个整形变量,方便进行操作。)</STRONG></P>
<P><STRONG>
或者*((char xdata *)0x4000)=X;//表示给存在xdata中,地址为0x4000的空间赋值。
</STRONG></P>
<P> // 表示从 code
0x4000处取一个 word 作为 xdata 型的指针
给<BR>px<BR>
px=*((int xdata * xdata *)0x4000);</P>
<P> //表示从 code
0x4000处取一个 word 作为 xdata
型的指针,<BR>
//再把这个指针指向的char数据赋给x<BR>
x=**((char xdata * code *)0x4000);</P>
<P>
//表示把函数f()入口地址当作xdata型指针,再把指向的xdata<BR>
//中的int型数据作为code型指针,把指向的code字节<BR>
//赋给x(晕,这样有意义吗?)<BR>
x=**(int code * xdata *)f;</P>
<P>
//把f()入口地址处的ROM中两个code型字节,<BR>
//赋给堆栈指针SP指向的字节(想干什么?编操作系统?)<BR>
*(unsigned int idata *)SP=*(unsigned int code *)&f;</P>
<P>
//表示把f()入口地址处的ROM中两个code型字节,<BR>
//作为一个xdata指针寻址,<BR>
//把指向的数据作为pdata指针寻址,<BR>
//再把把指向的数据作为idata指针寻址,<BR>
//把该地址处的一个字节赋给x
(我靠,累死了)<BR> x=
****(unsigned int data * idata * pdata * xdata * code
<BR>*)&f;<BR>
<BR>
//总之,一个括号里面外面的"*"一样多就表示指向的是数据。<BR> }</P>
<P>
{<BR>
//数组函数<BR> code void
(*ArrFn[])(void)
=<BR>
{
&f1,<BR>
&f2,<BR>
};<BR>
//可以像引用数组一样调用函数啦:<BR>
(*ArrFn[0])();<BR>
(*ArrFn[1])();<BR> }</P>
<P>
{<BR>
//这样将使函数调用0000H处:<BR>
void (*reset)
(void);<BR>
reset=0x0;<BR>
(*reset)();<BR>
reset();</P>
<P>
//或者直接这样,仅仅生成一条指令LCALL
1234H<BR> ((void
(code *)(void))0x1234)(); <BR> }</P>
<P>
{<BR>
//这样可以调用RETI指令:<BR>
#define INT_NUM 30
//空闲中断号<BR> ((void
(code
*)(void))(INT_NUM*8+3))();<BR>
//当然需要在外面声明 int_rpt()interrupt INT_NUM
{}<BR> }<BR>
{<BR>
//这样调用RETI指令太变态:<BR>
code unsigned char
ret_i=0x32;<BR>
((void (code *)(void))(&ret_i))();<BR>
}<BR>}<BR>int_rpt()interrupt INT_NUM {}</P>
<P>指针类型和存储区的关系详解</P>
<P>一、存储类型与存储区关系</P>
<P> data
---> 可寻址片内ram<BR>
bdata --->
可位寻址的片内ram<BR> idata
--->
可寻址片内ram,允许访问全部内部ram<BR>
pdata ---> 分页寻址片外ram
(MOVX @R0) (256 BYTE/页)<BR>
xdata ---> 可寻址片外ram
(64k 地址范围)<BR> code
---> 程序存储区 (64k 地址范围),对应MOVC @DPTR</P>
<P>二、指针类型和存储区的关系</P>
<P>
对变量进行声明时可以指定变量的存储类型如:<BR> uchar data x和data
uchar x相等价都是在内ram区分配一个字节的变量。</P>
<P>
同样对于指针变量的声明,因涉及到指针变量本身的存储位置和指针所指向的存储区位置不同而进行相应的存储区类型关键字的<BR>使用如:</P>
<P> uchar xdata * data pstr</P>
<P>
是指在内ram区分配一个指针变量("*"号后的data关键字的作用),而且这个指针本身指向xdata区("*"前xdata关键字的作用),<BR>可能初学C51时有点不好懂也不好记。没关系,我们马上就可以看到对应“*”前后不同的关键字的使用在编译时出现什么情况。</P>
<P> ......<BR> uchar xdata
tmp[10];
//在外ram区开辟10个字节的内存空间,地址是外ram的0x0000-0x0009<BR>
......</P>
<P> 第1种情况:</P>
<P> uchar data * data
pstr;<BR> pstr=tmp;</P>
<P> 首先要提醒大家这样的代码是有bug的,
他不能通过这种方式正确的访问到tmp空间。 为什么?我们把编译后看到下面的汇编<BR>代码:</P>
<P> MOV
0x08,#tmp(0x00)
;0x08是指针pstr的存储地址</P>
<P> 看到了吗!本来访问外ram需要2
byte来寻址64k空间,但因为使用data关键字(在"*"号前的那个),所以按KeilC编译环境来说<BR>就把他编译成指向内ram的指针变量了,这也是初学C51的朋友们不理解各个存储类型的关键字定义而造成的bug。特别是当工程中的<BR>默认的存储区类为large时,又把tmp[10]
声明为uchar tmp[10] 时,这样的bug是很隐秘的不容易被发现。</P>
<P> 第2种情况:</P>
<P> uchar xdata * data
pstr;<BR> pstr = tmp;</P>
<P>
这种情况是没问题的,这样的使用方法是指在内ram分配一个指针变量("*"号后的data关键字的作用),而且这个指针本身指向<BR>xdata区("*"前xdata关键字的作用)。编译后的汇编代码如下。</P>
<P> MOV
0x08,#tmp(0x00)
;0x08和0x09是在内ram区分配的pstr指针变量地址空间<BR> MOV
0x09,#tmp(0x00)</P>
<P>
这种情况应该是在这里所有介绍各种情况中效率最高的访问外ram的方法了,请大家记住他。</P>
<P> 第3种情况:</P>
<P> uchar xdata * xdata
pstr;<BR> pstr=tmp;</P>
<P> 这中情况也是对的,但效率不如第2种情况。编译后的汇编代码如下。</P>
<P> MOV DPTR,
#0x000A
;0x000A,0x000B是在外ram区分配的pstr指针变量地址空间<BR> MOV
A, #tmp(0x00)<BR> MOV @DPTR,
A<BR> INC DPTR<BR> MOV A,
#tmp(0x00)<BR> MOVX @DPTR, A</P>
<P> 这种方式一般用在内ram资源相对紧张而且对效率要求不高的项目中。</P>
<P> 第4种情况:</P>
<P> uchar data * xdata
pstr;<BR> pstr=tmp;</P>
<P> 如果详细看了第1种情况的读者发现这种写法和第1种很相似,是的,同第1
种情况一样这样也是有bug的,但是这次是把pstr分<BR>配到了外ram区了。编译后的汇编代码如下。</P>
<P> MOV DPTR,
#0x000A
;0x000A是在外ram区分配的pstr指针变量的地址空间<BR> MOV A,
#tmp(0x00)<BR> MOVX @DPTR, A</P>
<P> 第5种情况:</P>
<P> uchar * data pstr;<BR>
pstr=tmp;</P>
<P>
大家注意到"*"前的关键字声明没有了,是的这样会发生什么事呢?下面这么写呢!对了用齐豫的一首老歌名来说就是
“请跟我<BR>来”,请跟我来看看编译后的汇编代码,有人问这不是在讲C51吗?
为什么还要给我们看汇编代码。C51要想用好就要尽可能提升C51<BR>编译后的效率,看看编译后的汇编会帮助大家尽快成为生产高效C51代码的高手的。还是看代码吧!</P>
<P> MOV 0x08,
#0X01
;0x08-0x0A是在内ram区分配的pstr指针变量的地址空间<BR> MOV
0x09, #tmp(0x00)<BR> MOV 0x0A,
#tmp(0x00)</P>
<P>
注意:这是新介绍给大家的,大家会疑问为什么在前面的几种情况的pstr指针变量都用2 byte空间而到这里就用3
byte空间了<BR>呢?这是KeilC的一个系统内部处理,在KeilC中一个指针变量最多占用 3
byte空间,对于没有声明指针指向存储空间类型的指针,<BR>系统编译代码时都强制加载一个字节的指针类型分辩值。具体的对应关系可以参考KeilC的help中C51
User's Guide。</P>
<P> 第6种情况:</P>
<P> uchar * pstr;<BR>
pstr=tmp;</P>
<P>
这是最直接最简单的指针变量声明,但他的效率也最低。还是那句话,大家一起说好吗!编译后的汇编代码如下。</P>
<P> MOV DPTR,
#0x000A
;0x000A-0x000C是在外ram区分配的pstr指针变量地址空间<BR> MOV
A, #0x01<BR> MOV @DPTR,
A<BR> INC DPTR<BR> MOV
DPTR, #0x000A<BR> MOV A,
#tmp(0x00)<BR> MOV @DPTR,
A<BR> INC DPTR<BR> MOV A,
#tmp(0x00)<BR> MOVX @DPTR, A</P>
<P>
这种情况很类似第5种和第3种情况的组合,既把pstr分配在外ram空间了又增加了指针类型的分辨值。</P>
<P>
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -