void *vp;
void*是一种特别的指针,因为它没有指向的类型,或者说不能根据这个类型判断出指向对象的长度。
void *指针具有以下特点:
1、任何指针(包括函数指针)都可以赋值给void指针;
type *p;
vp=p; //不需转换,只获得变量/对象地址而不获得大小
2. void指针赋值给其他类型的指针时都要进行转换;
type * p=(type *)vp; //转换类型也就是获得指向变量/对象大小
3. void指针在强制转换成具体类型前,不能解引用;
*vp //错误 (因为void指针只知道指向变量/对象的起始地址,而不知道指向变量/对象的大小(占几个字节)所以无法正确引用 )
4. void指针不能参与指针运算,除非进行转换。
(type*)vp++; //等价于:vp=vp+sizeof(type)
void*的作用
1、传参:通用类型
可以作为函数模板,链表等参数的通用参数。在使用时,只需要强制类型转换就可以。
例如内存操作函数memcpy和memset的函数原型分别为:
void* memcpy(void *dest, constvoid *src, size_t len);
void* memset(void *buffer, int c, size_t num);
这样,任何类型的指针都可以传入memcpy和memset中,这也真实地体现了内存操作函数的意义,
因为它操作的对象仅仅是一片内存,而不论这片内存是什么类型。
2、强制类型转换
有时候由于重载等的干扰,导致需要转换成void *,来进行取地址。
例如,(void *)obj.member,就可以取到member的地址;直接&(obj.member)取到的实际上是obj的开始地址。
3、指向0的地址
(void *)0,指向全是0的地址,相当于NULL。
简单的小例子:
#include <stdio.h>
int main(){
void *data = "Hello World";//一个无类型指针代替所有的类型,这里是存储字符型数据
printf("%s\n",data);//将其输出
return 0;
}
下面是无类型指针,访问是一个字节一个字节的访问的。
#include <stdio.h>
#include <stdlib.h>
int main(){
void *data = malloc(8);//一个无类型指针代替所有的类型
printf("%ld\n",sizeof(data[0]));//没有创建具体的数据类型,访问的时候是一个字节一个字节访问的。
free(data);//记得释放内存
return 0;
}
输出
1
下面创建一个整型数据类型,在访问时将输出4,即整型的数据长度来访问。
#include <stdio.h>
#include <stdlib.h>
int main(){
void *data = malloc(8);//一个无类型指针代替所有的类型
int *intData = data;//为其指定一个整型的数据类型
printf("%ld\n",sizeof(intData[0]));//输出其数据尺寸
free(data);//记得释放内存
return 0;
}
输出
4
空指针并赋值
#include <stdio.h>
#include <stdlib.h>
int main(){
void *data = malloc(8);//一个无类型指针代替所有的类型
int *intData = data;//为其指定数据类型
intData[0] = 10000;
printf("%d\n",intData[0]);//输出其值
free(data);//记得释放内存
return 0;
}
输出
10000