void(*p)()和void*p()区别

本文详细解析了C/C++中复杂函数指针声明void(*b[10])(void(*)())的意义,通过逐步拆解说明了如何理解此类声明,并提供了一种更为清晰的声明方式。
 
整理来源:http://zhidao.baidu.com/link?url=lYBzk1BiYAVy20H4HBOXqAC70E0o616Z0HvvOQeP-2PCoFTjTWZXr-_c1At2OyfyULfUPD4BkVbOFtLSDdzH7q
void (*p)()是一个指向函数的指针,表示是一个指向函数入口的指地变量,该函数的返回类型是void类型。它的用法可参看下例:
例如:有一返加void值的函数swap,(swap用来交换两个数)
void (*p)();    /*定义指向函数的指针变量p*/
p=swap;    /*使指针变量p指向函数max*/
(*p)(a,b);   /*通过指针变量p调用函数max*/
它等价于:
swap(a,b)

void *p()是一个指针型函数,它的函数名为p,返回了一个指针,因为是void,这个指针没有定义类型,所以返回的是一个通用型指针。
给你举一个例子:
#include<stdio.h>
int *max(int *p);
void main()
{
    int a[10]={96,23,45,86,79,63,58,36,29,95};
    int *p;
    p=max(a);
    printf(“max=%d\n”,*p);
}
int *max(int *p)
{
     int i,*q=p;
     for(i=1;i<10;i++)
     if(*(p+i)>*q)
          q=p+1;
     return q;
}
 
 
整理自:http://www.cnblogs.com/leon3000/archive/2007/11/15/2037989.html
 
 
 

void (*b[10]) (void (*)());

void (*b[10]) (void (*)());

看到这行代码,相信程序员们都会倒吸一口冷气吧。如果非常不幸的在维护的代码中看到类似的表述,除了想办法找出写出这种天书的猛人来,只能自己硬着头皮搞清楚了。

在C和C++中,构造这类声明表达式只有一条简单的规则:按照使用的方式来声明。

C变量的声明都是由两部分组成的:类型,以及一组类似表达式的声明符(declarator)。声明符类似于表达式,对它求值应该返回一个声明中给定类型的结果。例如,我们来看一个简单的声明:

float f;

这里f就是声明符,对其求值,应该得到一个float型的数值。

然后看括号的作用,比如:

float ((f));

很简单,这个声明的含义就是,((f))的类型为浮点类型。因为括号和f之间没有其他的修饰了,所以,我们可以推知,f也是浮点型。

再来看稍微复杂点的:

float f();

我们注意到有一个()表达式出现,这个声明符的含义是一个函数。所以这个声明的含义就是,f()的求值结果是一个浮点数,也就是说,f是个返回值为浮点数的函数。

OK,这些都很简单,再看一个:

float *f();

这个就是表示*f()是个浮点表达式,而()的结合优先级高于*,所以,*f()也就是*(f()),f是个函数,返回值是个指针,这个指针指向一个浮点数。

如果*f是先结合的呢?比如:

float (*f)();

这时,f就不是和()结合而成为一个函数名,而是和*结合成为一个指针名,这个指针,是指向函数入口的函数指针,而这个函数,返回值是浮点型。

现在我们知道怎么声明一个指定类型的变量了。在这个声明表达式中,把变量名,和声明末尾的分号去掉,剩余的部分用一个括号整个括起来,这样就得到了这个变量的类型声明了,比如:

float (*f)();

这个表示f是一个指向返回值为浮点型的函数指针。而我们把f这个变量名,和最后的;号去掉,就得到:

( float (*)() )

这个就表示,这个表达式是一个类型,即“指向返回值为浮点型的函数的指针”。如果用这个类型去修饰一个变量名,我们就叫它类型转换符。

现在有了这些预备知识,我们可以回头看标题的声明到底是什么意思了:

void (*b[10]) (void (*)());

首先,表达式的后半部分被两个()分隔开了,我们分别分析它们。( *b[10] ),其中出现了变量名b,很容易就知道,b是一个有10个元素的数组,每个元素都是一个指针。

然后,看(void(*)()),其中没有出现变量名,所以它代表了一个类型,即“指向返回值为void型的函数的指针“,而我们知道,C语法中,类型修饰符是必须出现在变量名的左边的,而在整个表达式中这个类型符是在变量名b的右边,所以, (void(*)())最外层的这个(),表示了定义了一个函数,这个函数有一个参数,就是一个指针,具体来说,就是“指向返回值为void型的函数的指针“。

这样就很清楚了,b数组里,每一个指针元素,都是一个函数指针,这个函数有一个参数,这个参数是一个函数指针。整个表达式最左边的void,则定义了b数组中函数指针所指向函数的返回值类型。

在一串绕口令式的解说后,我们终于看到了事实的真相:这行代码就是逗你玩...

如果不是故意偷懒,这样代码的作者真是在玩弄阅读者的感情,如果非要实现这样一个复杂的定义的话,我们也是有更好的方法的,就是使用typedef来进行类型声明。

在面对void (*b[10]) (void (*)());时,我们可以先声明后半部分的类型:

typdef void (*pFunParam)();

即表示,类型pFunParam,是一个函数指针。

然后,针对整个表达式声明一个类型:

typedef void (*pFun)(pFunParam);

即表示,类型pFun,是一个函数指针。此函数的参数类型为pFunParam。

最后,进行变量的声明:

pFun b[10];

这样,就清晰许多了吧。最重要的时,利用这样的方式,将编写者的设计思路也体现在了代码中。

最后,介绍一个更快速的阅读方法:从右向左。

编译器在进行代码扫描时是从左向右的,而我们在解读这个表达式的时候,从右向左的方法会更容易些,如:

从右向左扫描以上表达式,首先我们看到一个“)”,因为其右边再没有变量,所以我们知道这是一个函数定义,然后,又看到一对“()“,显然也是函数定义,在就是(*),显然和后面的()联合起来表达了一个函数指针,再左边的void,即修饰了此指针的类型;然后出现了和最右边”)"对应的”(”,显然这个函数定义也完成了,再左边的[10]表达了一个数组,左边是它的名字b,而更左边的“*”和后面的“(“一起表达了一个函数指针,这个函数指针定义了b数组中元素的类型。再往左,指针元素的定义void也出现了。这样,我们就又看到了此表达式的真相。

<think>我们正在讨论C语言中的void指针(void *p)。根据引用内容,void指针是一种通用指针类型,可以指向任何类型的数据。下面我将从含义、使用场景使用方法三个方面进行详细解释。 ### 1. 含义 - `void *p` 声明了一个void类型的指针变量p。 - void指针的特殊之处在于它不指向任何具体的数据类型,因此可以存储任意类型变量的地址。 - 例如,可以将int型、char型、float型等任何类型变量的地址赋给void指针: ```c int a = 10; char c = 'c'; float f = 1.2f; void *p = NULL; // 初始化为空 p = &a; // 指向int p = &c; // 指向char p = &f; // 指向float ``` - 注意:void指针不能直接进行解引用操作,因为编译器不知道它指向的数据类型,也就无法确定要访问多少字节。所以在使用前必须将其强制转换为具体类型的指针。 ### 2. 使用场景 void指针主要用于以下场景: - **泛型编程**:在编写通用函数时,void指针可以接收不同类型的数据。例如,C标准库中的内存操作函数(如`memcpy`, `memset`)就是使用void指针作为参数类型,以实现对任意类型内存的操作。 ```c void *memcpy(void *dest, const void *src, size_t n); ``` - **动态内存分配**:函数`malloc``calloc`返回的就是void指针,这样我们可以将其强制转换为所需的类型: ```c int *arr = (int*)malloc(10 * sizeof(int)); // 分配10个整数的空间 ``` - **回调函数**:在实现回调函数时,void指针可以用来传递任意类型的参数,增加函数的灵活性。 - **数据结构**:在实现通用数据结构(如链表、树)时,void指针可以存储任意类型的数据,使得数据结构可以存放不同类型的数据元素。 ### 3. 使用方法 使用void指针时需要注意以下几点: - **赋值**:可以将任何类型的指针直接赋给void指针,无需显式转换(C语言自动完成隐式转换)。 ```c int a = 10; void *p = &a; // 正确,无需转换 ``` - **赋值给具体类型指针**:当把void指针赋给其他具体类型的指针时,必须进行显式的类型转换。 ```c void *p = &a; int *ptr = (int*)p; // 必须显式转换 ``` - **解引用**:不能直接解引用void指针,必须将其转换为具体类型的指针后再解引用。 ```c void *p = &a; // *p = 20; // 错误:不能解引用void指针 *(int*)p = 20; // 正确:先转换为int指针再解引用 ``` - **指针算术运算**void指针不能直接进行算术运算(如p++),因为编译器不知道指向的数据类型大小。所以必须先转换为具体类型的指针再进行运算。 ```c int arr[5] = {1,2,3,4,5}; void *p = arr; // p++; // 错误:void指针的算术运算不合法 int *ptr = (int*)p; ptr++; // 正确:指针移动sizeof(int)字节 ``` ### 总结 void指针是C语言中一种强大的工具,它提供了极大的灵活性,允许我们编写通用的代码。但是使用时需要注意类型转换解引用的规则,避免出现错误。 §§相关问题§§ 1. 为什么void指针不能直接进行解引用操作? 2. 在动态内存分配中,为什么malloc返回void指针? 3. void指针NULL指针有什么区别? 4. 如何正确使用void指针实现泛型函数? </think>### `void *p` 的含义、使用场景使用方法 #### 1. **含义** `void *p` 是 C 语言中的**通用指针**(泛型指针): - **无类型特性**:编译器不假设 `p` 指向的具体数据类型(如 `int`/`char` 等)。 - **地址容器**:仅存储内存地址,不包含数据类型信息。 - **灵活性**:可指向任意类型的数据: ```c int a = 10; char c = 'x'; void *p = &a; // 指向 int p = &c; // 指向 char(不需强制转换) ``` #### 2. **使用场景** - **泛型函数**:处理不同类型的数据(如内存操作函数): ```c void swap(void *a, void *b, size_t size) { char tmp[size]; // 按字节交换 memcpy(tmp, a, size); memcpy(a, b, size); memcpy(b, tmp, size); } ``` - **动态内存分配**:`malloc`/`calloc` 返回 `void*`: ```c int *arr = malloc(10 * sizeof(int)); // 自动转为 int* ``` - **数据结构**:构建通用容器(链表/栈): ```c typedef struct Node { void *data; // 可存任何类型数据 struct Node *next; } Node; ``` - **回调函数**:传递任意类型参数: ```c void callback(void *param) { /* 处理 param */ } ``` #### 3. **使用方法** - **赋值**:无需强制转换即可指向任意类型: ```c float f = 3.14; void *p = &f; // 合法 ``` - **解引用****必须显式类型转换**: ```c float f = 3.14; void *p = &f; // *p = 2.71; // 错误!不能直接解引用 *(float *)p = 2.71; // 正确:先转换为 float* ``` - **指针运算****不能直接运算**(需转为具体类型): ```c int arr[5] = {1, 2, 3, 4, 5}; void *p = arr; // p++; // 错误!void* 步长未知 int *q = (int *)p; q++; // 正确:步长为 sizeof(int) ``` - **类型安全**:使用时需确保类型匹配: ```c double d = 1.23; void *p = &d; int x = *(int *)p; // 危险!类型不匹配导致数据错误 ``` #### 总结 | **特性** | **说明** | |----------------|--------------------------------------------------------------------------| | **核心作用** | 充当“通用指针”,突破类型限制 | | **优势** | 提高代码灵活性,实现泛型编程 | | **注意事项** | 解引用/运算前必须强制类型转换,否则行为未定义 | | **典型应用** | 内存操作函数(`memcpy`)、动态内存分配、多态数据结构、跨类型回调函数[^1][^2][^3] | > ⚠️ 关键原则:**`void*` 是“盲指针”**,使用时开发者需自行维护类型正确性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值