指针和数组的学习

时间:2022-07-24
本文章向大家介绍指针和数组的学习,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。

在msOS的代码里有这个两句话

//first put in data, then increase u8MsgHead
((U8 *)(&(u16MsgArray[u8MsgHead])))[0] = MsgType;
((U8 *)(&(u16MsgArray[u8MsgHead])))[1] = Val;

抛开他们所执行的具体意义,仅从C语言层面上去理解。 u16MsgArray是文中定义的数组: static U16 data u16MsgArray[MSG_ARRAY_SIZE]; MSG_ARRAY_SIZE的值为4,4个short类型的数组,u8MsgHead取值范围0~3 咋一看,不理解什么意思,于是乎在电脑上调试了,平台为eclipse+mingw 实验代码如下:


 unsigned short u16MsgArray[4]={
0,1,2,3
};

 ((unsigned char *)(&(u16MsgArray[0])))[0] = 4;
 ((unsigned char *)(&(u16MsgArray[0])))[1] = 5;

运行后的结果为:

u16MsgArray[4]={
1284,1,2,3
};


 1284=0x504=0b'101 0000 0100

分析: &(u16MsgArray[0])意为首元素的地址,可以理解为指针; (unsigned char *)(&(u16MsgArray[0]))意为转换为unsigned char的指针,为了理解,记: unsigned short *u16_ptr = &(u16MsgArray[0]);(注意不能写成unsigned short *u16_ptr = u16MsgArray;即使指针的值不变,但这两个语句的意义不同,后续的指针转换也就不同了)。 这里的u16_ptr是数组指针,指向数组的指针,(顺便提一句,*u16_ptr[9]中的u16_ptr是指针数组的数组名) 然后再进行如下操作:


((unsigned char *)u16_ptr)[0] = 5;
((unsigned char *)u16_ptr)[1] = 6;
((unsigned char *)u16_ptr)[2] = 4;

u16MsgArray数组的值更新为:


 {0x605,0x4,0x2,0x3}

注意,x86是小端模式,高地址存高位数,和我们平时写的数字是反着的,看着有些别扭,也佩服CPU的设计者。 内存地地址—————>高地址 u16MsgArray[0]—>u16MsgArray[3] 0x0022fee0—————>0x0022fee6 0506 0400 0200 0300 在测试中,添加((unsigned char *)(&(u16MsgArray[0])))[-1] = 7; 内存中显示(u16MsgArray[-1]) = 1792. 其中 &(u16MsgArray[-1]) = 0x0022fede 0x0022fede——>0x0022fee0—————>0x0022fee6 0007 0506 0400 0200 0300 以上的内存分布在eclips的memory里查看的 虽然在内存中,u16MsgArray[0]的显示为0506,但是实际的值为0x0605,即1541,这和变量的值一致的(eclipse的Variables查看的)。 同理:0x700 = 1792。