论void void*及void**类型

本文详细介绍了C语言中`void`和`void*`的作用。`void`用于限定不返回任何类型的函数和不接受参数的函数。`void*`作为无类型指针,可以接受任何类型指针的赋值,但在使用时需要进行类型转换。文章还探讨了`void**`在函数参数传递中的应用,特别是在代码复用和双向链表操作中的重要性。通过示例代码说明了如何使用`void**`在不同数据类型间转换。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值