关于指针数组指针做为返回类型的函数

本文详细解析了一段复杂的C/C++代码中关于指针的定义与使用,特别是对函数返回指针类型进行了深入探讨。
今天被一哥们叫了看了一段程序。
里面有这样一段代码:
C/C++ code
int * (*func(int n))[3];
反应如下:
这是一个数组。
这是一个指针数组。
这是一个函数指针数组。
到后面又看到然后有这样的用法:
C/C++ code
func(n_temp);
这难道是一个函数???
那么[3]又表示什么呢。
然后就分析了下。
后来才发现原来()的优先级比*高。
那么这是一个函数指针的想法肯定是错的。
下边写下午简单的分析。
int func(int n);
这当然是一个函数(具有一个int型参数)。
int (*func(int n));
这是一个具有一个int型参数,并且返回值是一个指针的函数。
int (*func(int n))[3];
这又是什么意思呢?
先看
int (*arr)[3];
arr是指向 具有三个整型元素数组的指针。
那么
int (*func(int n))[3];
是一个函数指针数组?没这种说法。
是一个数组指针函数?好像还是不对。。。
因为c/c++里面不允许返回数组的。
所以只能是返回类型是一个数组指针。
那么:
int * (*func(int n))[3];
这是一个具有一个int型参数,并且返回值是一个指向数组的指针。
而这个数组又是具有三个int指针元素的数组。
函数返回指针是没问题的,数组指针、函数指针等都行
返回数组、函数则不行
仅供聊资。。。。表骂人。
 
<think>我们详细解释C语言中四个容易混淆的概念:指针函数函数指针指针数组数组指针。 注意:指针函数本质是函数函数指针本质是指针指针数组本质是数组数组指针本质是指针。 1. **指针函数(返回指针函数)** - 定义:返回值指针类型的函数。 - 语法:`返回类型 *函数名(参数列表)` - 示例: ```c int *create_array(int size) { int *arr = (int*)malloc(size * sizeof(int)); return arr; // 返回一个指向整型的指针 } ``` - 调用:`int *myArray = create_array(10);` 2. **函数指针(指向函数指针)** - 定义:指向函数指针变量,用于调用函数或作为回调函数。 - 语法:`返回类型 (*指针名)(参数类型列表)` - 示例: ```c int add(int a, int b) { return a+b; } int (*pFunc)(int, int) = add; // pFunc指向add函数 ``` - 调用:`int result = pFunc(3, 5); // 相当于调用add(3,5)` 3. **指针数组(元素为指针数组)** - 定义:数组元素均为指针类型。 - 语法:`数据类型 *数组名[数组长度]` - 示例: ```c char *names[] = {"Alice", "Bob", "Charlie"}; // 每个元素是char*类型 ``` - 访问:`printf("%s", names[0]); // 输出"Alice"` 4. **数组指针(指向数组指针)** - 定义:指向整个数组指针,常用于二维数组操作。 - 语法:`数据类型 (*指针名)[数组长度]` - 示例: ```c int matrix[3][4] = {{1,2,3,4}, {5,6,7,8}, {9,10,11,12}}; int (*pArr)[4] = matrix; // pArr指向一个包含4个整数的数组 ``` - 访问:`printf("%d", pArr[1][2]); // 输出第1行第2列元素,即7` **总结对比表** | 类型 | 本质 | 声明形式 | 用途 | |------------|--------|------------------------|--------------------------| | 指针函数 | 函数 | `int* func(...)` | 返回指针 | | 函数指针 | 指针 | `int (*pFunc)(...)` | 动态调用函数、回调 | | 指针数组 | 数组 | `int* arr[10]` | 存储多个指针 | | 数组指针 | 指针 | `int (*pArr)[10]` | 操作二维数组的行 | **深入理解函数指针数组** 结合引用[2][^2]和引用[3][^3],函数指针数组是存储函数指针数组,其声明为: ```c // 声明一个包含5个函数指针数组,每个指针指向返回void、参数为const char*的函数 void (*funcArray[5])(const char*); ``` 而指向函数指针数组指针(二级指针): ```c void (*(*pFuncArray)[5])(const char*) = &funcArray; // pFuncArray是指向函数指针数组指针 ``` **易混淆点解析** - `int *p[10]`:指针数组,p是数组,包含10个int*元素。 - `int (*p)[10]`:数组指针,p指向一个包含10个int的数组。 - `int *func()`:指针函数func返回int*。 - `int (*pFunc)()`:函数指针,pFunc指向返回int的函数。 **应用场景** - 指针函数:动态内存分配函数(如malloc)、字符串处理函数(如strcpy)。 - 函数指针:回调函数(如qsort)、状态机实现。 - 指针数组:命令行参数argv、字符串数组。 - 数组指针:二维数组传递、矩阵运算。 **代码示例:综合运用** ```c #include <stdio.h> // 指针函数:返回字符串长度 size_t* str_len_ptr(const char* s) { static size_t len; // 静态存储,避免返回局部变量地址 for(len=0; s[len]; len++); return &len; } // 函数指针应用:回调函数 void process(int a, int b, int (*op)(int,int)) { printf("Result: %d\n", op(a,b)); } int add(int a, int b) { return a+b; } int mul(int a, int b) { return a*b; } int main() { // 指针数组:存储字符串 char* fruits[] = {"Apple", "Banana", "Cherry"}; // 数组指针:操作二维数组 int arr[2][3] = {{1,2,3}, {4,5,6}}; int (*pArr)[3] = arr; // 使用指针函数 size_t *len = str_len_ptr(fruits[0]); printf("Length of '%s': %zu\n", fruits[0], *len); // 使用函数指针数组 int (*ops[2])(int,int) = {add, mul}; for(int i=0; i<2; i++) { process(3, 4, ops[i]); } // 使用数组指针访问二维数组 printf("arr[1][2] = %d\n", pArr[1][2]); // 输出6 return 0; } ``` 输出: ``` Length of 'Apple': 5 Result: 7 Result: 12 arr[1][2] = 6 ``` **注意事项** - 指针函数返回的指针不能指向局部变量(除非是静态或动态分配的内存)。 - 函数指针需确保签名(返回类型和参数类型)匹配。 - 指针数组数组指针在内存布局上有本质区别。 §§相关问题§§ 1. 函数指针作为参数传递时有什么注意事项? 2. 如何用函数指针实现状态机? 3. 指针数组和二维字符数组在内存结构上的区别是什么? 4. 数组指针在动态二维数组中的应用? </think> ### 详细解析C语言中的指针函数函数指针指针数组数组指针 --- #### **1. 指针函数(返回指针函数)** **本质**:普通函数,但返回值指针类型。 **语法**:`返回类型 *函数名(参数列表)` **特点**: - 返回一个内存地址(指针),可指向变量、数组或动态分配的内存。 - **必须确保返回的指针有效**(避免返回局部变量地址)。 **示例**: ```c // 指针函数:返回动态分配的整型数组 int* create_array(int size) { int *arr = (int*)malloc(size * sizeof(int)); // 动态分配内存 for (int i = 0; i < size; i++) arr[i] = i * 10; return arr; // 返回指针 } int main() { int *ptr = create_array(5); printf("%d", ptr[2]); // 输出:20 free(ptr); // 释放内存 } ``` --- #### **2. 函数指针(指向函数指针)** **本质**:指针变量,存储函数的入口地址。 **语法**:`返回类型 (*指针名)(参数类型列表)` **用途**: - 实现回调函数(如 `qsort` 中的比较函数)。 - 动态调用不同函数(如状态机)。 **示例**: ```c int add(int a, int b) { return a + b; } int sub(int a, int b) { return a - b; } int main() { int (*pFunc)(int, int); // 声明函数指针 pFunc = add; // 指向 add 函数 printf("5+3=%d\n", pFunc(5, 3)); // 输出:8 pFunc = sub; // 改为指向 sub 函数 printf("5-3=%d\n", pFunc(5, 3)); // 输出:2 } ``` --- #### **3. 指针数组(元素是指针数组)** **本质**:数组,每个元素都是指针类型。 **语法**:`数据类型 *数组名[数组长度]` **用途**: - 存储多个字符串(如命令行参数 `char *argv[]`)。 - 管理多个指针的集合。 **示例**: ```c int main() { char *names[] = {"Alice", "Bob", "Charlie"}; // 指针数组(每个元素是 char*) for (int i = 0; i < 3; i++) printf("Name %d: %s\n", i, names[i]); } ``` **输出**: ``` Name 0: Alice Name 1: Bob Name 2: Charlie ``` --- #### **4. 数组指针(指向数组指针)** **本质**:指针,指向整个数组(而非数组首元素)。 **语法**:`数据类型 (*指针名)[数组长度]` **用途**: - 操作二维数组(传递二维数组函数)。 - 处理动态分配的二维数组。 **示例**: ```c void print_matrix(int (*p)[3], int rows) { // 数组指针(指向含3个int的数组) for (int i = 0; i < rows; i++) { for (int j = 0; j < 3; j++) printf("%d ", p[i][j]); // 等效于 *(*(p+i)+j) printf("\n"); } } int main() { int matrix[2][3] = {{1,2,3}, {4,5,6}}; int (*pArr)[3] = matrix; // 数组指针指向二维数组 print_matrix(pArr, 2); } ``` **输出**: ``` 1 2 3 4 5 6 ``` --- ### **对比总结** | **类型** | **本质** | **声明示例** | **核心用途** | |----------------|------------|--------------------------|----------------------------| | 指针函数 | 函数 | `int* func()` | 返回指针(如动态数组) | | 函数指针 | 指针 | `int (*pFunc)(int, int)` | 动态调用函数、回调 | | 指针数组 | 数组 | `char* arr[10]` | 存储多个字符串或指针 | | 数组指针 | 指针 | `int (*p)[5]` | 操作二维数组的行 | --- ### **常见混淆点解析** 1. **指针函数 vs 函数指针** - `int* f()`:指针函数,`f` 是函数,返回 `int*`。 - `int (*f)()`:函数指针,`f` 是指针,指向返回 `int` 的函数。 2. **指针数组 vs 数组指针** - `int* a[5]`:指针数组,`a` 是数组,元素为 `int*`(5个指针)。 - `int (*a)[5]`:数组指针,`a` 是指针,指向含5个 `int` 的数组。 3. **函数指针数组**(结合引用[2][^2]和[3][^3]) ```c void (*funcArray[3])(int); // 数组元素是函数指针 void (*(*pFuncArray)[3])(int) = &funcArray; // 指向该数组指针 ``` --- ### **应用场景** - **指针函数**:工厂模式(创建对象)、返回动态数据。 - **函数指针**:事件处理、算法策略切换。 - **指针数组**:命令行参数处理、多字符串管理。 - **数组指针**:矩阵运算、传递二维数组函数。 ---
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值