空指针和void *类型指针

简介今天,有个师弟问我:“师兄,空指针和void *类型指针是怎样的?” 当时,我大概的说法是: 1、空指针是没有指向的指针,将暂时用不到的指针定义成空指针,能防止误用。 2、而void * 类型的指针是有指向的指针,但它的指向的数据的类型暂时不确定,所以先弄成void * 类型,后期一般要强制转换的。 回来后,发现这种说法虽然勉强能够接受,但依然有不足的地方,现在补充更正一下: 1、空指针实质上是有指向的指针,但它指向的地址是很小的地址,约定俗成为地址0,我来解释一下为
今天,有个师弟问我:“师兄,空指针和void *类型指针是怎样的?”

当时,我大概的说法是:

1、空指针是没有指向的指针,将暂时用不到的指针定义成空指针,能防止误用。

2、而void * 类型的指针是有指向的指针,但它的指向的数据的类型暂时不确定,所以先弄成void * 类型,后期一般要强制转换的。



回来后,发现这种说法虽然勉强能够接受,但依然有不足的地方,现在补充更正一下:

1、空指针实质上是有指向的指针,但它指向的地址是很小的地址,约定俗成为地址0,我来解释一下为什么。

#include
void main(){
int * str = NULL;
gets(str);
printf("%s",str);
}
这个程序在语法上是没有问题的,但是我们编译、链接都没有报错,但是我们打开的时候却出错了。
原因是空指针指向的地址是不保存数据,同时不允许程序访问的。

那么,这个NULL到底是什么呢?

我打开了用F12对着NULL 单击打开,看看它到底是什么,结果打开了头文件stdio.h,显示   #define NULL    ((void *)0)

显然,这是一个宏定义。NULL 实际上是((void*)0) ,容易看出,外层的括号纯粹是为了防止歧义;里层的括号则是强制类型转换,转换成void * 类型,本来void * 类型是用来存放地址的,那么这里的0自然就是地址0了。

在内存分配方面,较小的地址是不用来存放数据的,也不允许程序访问的。所以,指针指向了它,就是这个指针不能操作它指向的这块较小的地址。

哈哈,至此,空指针算是解释通了。

简单来说,空指针有指向,但是它指向的地址是特殊的,该地址不允许存放数据和不允许程序访问,所以空指针不能操作该地址里的东西,我们就理解为“指针指向了空,无法操作了”。



2、void * 类型指针,这个类型指针指向了实实在在的存放数据的地址,但是该地址存放的数据的数据类型我们暂时不知道。

举个例子,我们的动态内存分配就是这样,一开始只是分配地址,但没有知道这块地址用了存放什么,接着强制类型转换,使得它用来存放我们想要存放的内容。


char*str=(char*)malloc(sizeof(char)*13);

上面这条代码,malloc()函数分配的地址一开始是void * 类型的,因为我们用来存放char类型数据,所以强制转换为 char *  。
本文转自:https://blog.csdn.net/luo_technically/article/details/52714389
新加评论 评论标题:

文章评论

    PVOID和VOID* 无类型指针

        2020-09-25    
    修改 删除

    https://blog.csdn.net/u012370255/article/details/28257043

    P表示指针,那么PVOID表示:void *  ---无类型指针


    所有指针都是一个32位二进制数(32位系统下),这个意义上说所有指针都是一样的,它们的大小一样,用于指向内存中的某处地址,然而指针为什么要有类型之分呢?答案是指针偏移。例如p为一个指针,它指向内存某处地址,那么p+1(或者写p[1])是什么意思呢?答案是p指向地址的后面那个地址,那么后面多少呢?这就看指针类型了,假如它是字符指针,那么就是后面一个字节,假如它是整型指针,那就是后面第四字节,假如它是一个结构体,那就是后面sizeof(结构体)个字节。可以说,指针有类型之分,完全就是为了计算地址偏移。这一区别到了汇编级就没有分别了,汇编级不存在指针类型,只有指针偏移数。


    那么void 指针是啥呢?答案是无类型指针。干啥用呢?它只是一个地址指向,从不用计算偏移(void 了也没法确定如何偏移了…),它只能指向一整块内存,只能通过它来访问这块内存,不能用偏移访问(p 1,p[1]等,千万不要用在void指针上)。它的好处是什么呢?答案是不用强制转换,任何类型指针都可直接赋值给一个void 指针,而不用转换。
    例如:void *p :
    char a :
    int b :
    p =&a :可以,不用转换
    p =&b :可以,不用转换
    那么通过p 怎么输出a和b呢?可以,答案是强制转换。*((int *)p ).*((char *)p )得到了对应类型的内容。
    指针,真灵活……

评论列表
PVOID和VOID* 无类型指针