底层解析C指针(二)

描述

上一篇主要讨论了C指针的本质,但并没有解释指针的类型问题,这次我们重点来讨论指针的类型与指针内存分配之间的关系。

对比以下两个指针

int *p;

char *p;

单纯从上面看这两个指针有何区别?

很多读者第一反应就是其类型不同,确实没错,但是我们现在需要从底层出发,搞清楚指针类型的意义。

其实单纯从上面来看,两个指针是没有任何区别的,其内存的大小完全相同。

但是由于指针有++和--操作,导致指针的偏移量大小到底多大,这个就需要由类型决定。 因此类型仅仅只决定指针移动时偏移的大小。

下面我们来彻底分析指针这个问题。

从程序员开始写int *p这句时,计算机仅仅为p分配了一个四字节的内存地址(假设编译器为32位),而这个内存地址中存储的数据是多少并不知道,由于其数据可能是一个随机数,因此我们强行访问p时会得到一些随机的数字,这又一步说明了指针为啥需要进行初始化,但是,我们如果查看&p时,我们可以看到这是合法的,因为我们查看的是p的地址, 即计算机为p分配的一个内存地址,如图所示:

C指针

这个可能比较好理解

我们现在开始深入,看如下:(下面代码摘自作者写的嵌入式操作系统)

typedef struct PCB_STRUCT
{
    struct PCB_STRUCT *TCBPrev;       //前趋指针
    struct PCB_STRUCT *TCBNext;       //后继指针

    UINT32             TCBDlyCount;   //延时计数器
    UINT8              TCBState;      //任务状态       255为头部   254尾部    0表示阻塞态,1表示就绪态,2表示挂起态

    INT8              *TCBName;       //任务名
    FUN                TCBTask;       //任务函数指针
    UINT8             *TCBStack;      //人工堆栈
    UINT8              TCBPrio;       //任务优先级
    pTCB              *TCBHandler;    //任务句柄  也叫任务ID   

    UINT16             TCB_SP;        //任务当前的SP指针 
}pTCB;

现在我们定义

pTCB *RdyTab[33];        //就绪查找表   0为空闲任务

我们现在思考:

RdyTab[33]的内存分配结构?

部分读者会有第一反应就是内存结构当然是33个结构体的大小。 如下:

C指针

其实上面是错误的

因为之前说过,即使是结构体指针数组,其数组的元素本质上仍是指针,因此,其大小也只是一个四字节的地址单元,因此,其正确的结构如下:

C指针

那有人会有疑惑,为啥

RdyTab[i]->TCBPrev等等,这些为什么存在呢?
其实这个并不存在,这里单纯指RdyTab[i]的内存单元中,而我们需要使用RdyTab[i]->TCBPrev等等
时,必须先初始化RdyTab[i]的值,而这个RdyTab[i]->TCBPrev就是指初始化时指向内存的首地址
偏移位置,这个工作是由编译器完成的,如下图所示!
C指针

显然当我们使用RdyTab[i]->TCBPrev时,其地址就是分配的结构体所在的地址,而不是指针数组的地址。

打开APP阅读更多精彩内容
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉

全部0条评论

快来发表一下你的评论吧 !

×
20
完善资料,
赚取积分