1、void的作用
c语言中,void为“不确定类型”,不可以用void来声明变量。如:void a = 10;如果出现这样语句编译器会报错:variable or field ‘a’ declared void。在C语言中void 常常用于:对函数返回类型的限定和对函数参数限定:
(1)对函数返回类型的限定:当函数不需要返回类型是必须用void 来限定返回类型,限定了函数的返回类型为void后函数不能有返回值;如:void fun(int a);
(2)对函数参数类型的限定:当函数不允许接受参数时必须用void 来限定函数参数,限定了函数的参数类型为void后函数不能有参数;如:int fun(void);
2、void * 的作用
void * 为无类型指针,任何类型的指针都可以直接赋值给它,无需进行强制类型转换。sizeof(void*)大小为4,因为为指针类型。任何类型的指针都可以直接赋值给它,无需进行强制类型转换。但这并不意味着,void * 也可以无需强制类型转换地赋给其它类型的指针。因为"无类型"可以包容"有类型",而"有类型"则不能包容"无类型"。
C语言中void * 为 “不确定类型指针”,void *可以用来声明指针。如:void * a;
(1)void *可以接受任何类型的赋值:
void *a = NULL;
int * b = NULL;
a = b;//a是void * 型指针,任何类型的指针都可以直接赋值给它,无需进行强制类型转换
我们可以认为void就是一张白纸可以在上班写任何类型的数值。
(2)void *可以赋值给任何类型的变量 但是需要进行强制转换:
例:
int * a = NULL ;
void * b ;
a = (int *)b;
但是有意思的是:void* 在转换为其他数据类型时,赋值给void* 的类型 和目标类型必须保持一致。简单点来说:void* 类型接受了int * 的赋值后 这个void * 不能转化为其他类型,必须转换为int *类型;
例如:
#include <stdio.h>
int main (){
int a= 10;
void *b = &a;printf("int a = %d\n",a);
printf("void (int *)b =%d \n",*(int *)b);
printf("void (double *)b =%d \n",*(double*)b); //编译器并不会报错但是其结果却有点出人意料
return 0;
}
其结果也是各种各样:
//---------结果1------------
int a = 10
void (int *)b = 10
void (double *)b = 0.000000
//---------结果2------------
int a = 10
void (int *)b = 10
void (double *)b = -169647784854594875714536834205830109813973725378637218342762753874027478154937823064700382247683481669586551809051934352771907584.000000
3、(void**)&的理解 的作用
我在另一篇文章讲过为啥使用双重指针,void**就是双重指针的一个具体应用,就是因为函数调用之后需要使用指针来传参,所以才需要定义指针的指针,本质还是c语言的传值和传址不同所导致的。其实C语言指针的学习,不管它是几层指针,只需要搞明白一个事情,在 C 语言中,赋值符号“=”号两边的数据类型必须是相同的,如果不同,则需要显示或隐式类型转换,所以不管几层指针,赋值的时候两边指针类型保持一致就可以了。既然如此为啥使用void**类型呢,我的理解是还是为了代码的工程化,可以代码复用的目的。
这里看下例子,这是双向链表删除数据的一个接口,这里就用到了void**类型:
int dlist_remove(DList *list, DListElmt *element, void **data) {
if (element == NULL || dlist_size(list) == 0)
return -1;
*data = element->data;
if (element == list->head) {
list->head = element->next;
if (list->head == NULL)
list->tail = NULL;
else
element->next->prev = NULL;
}
else{
element->prev->next = element->next;
if (element->next == NULL)
list->tail = element->prev;
else
element->next->prev = element->prev;
}
free(element);
list->size--;
return 0;
}
对应的调用关系截图如下:
这里就可以看出&data就是(int **)&data,(void **)&data和&data还是同一块内存,只不过数据类型发生变化了,这里为了代码复用,不管什么类型数据都可以使用这个借口函数了。
参考:
https://www.cnblogs.com/yuanyongbin/p/8058755.html
https://blog.csdn.net/wcybrain/article/details/78300445