当前位置: 首页 > backend >正文

C语言指针进阶

文章目录

  • 1. 指向指针的指针(int pp)
    • 1.1 基本概念
    • 1.2 内存模型
    • 1.3 实战应用
  • 2. 函数指针
    • 2.1 什么是函数指针?
    • 2.2 声明与使用
    • 2.3 高级应用
  • 3. 结构体指针
    • 3.1 基本用法
    • 3.2 动态内存分配
    • 3.3 链表实现
  • 4. 综合实战
  • 5. 避坑指南

1. 指向指针的指针(int pp)

1.1 基本概念

  • 指向指针的指针(多级指针)是指存储指针变量地址的指针,常用于动态多维数组、链式数据结构等场景。
int a = 10;
int *p = &a;      // p 存储 a 的地址
int **pp = &p;    // pp 存储 p 的地址

1.2 内存模型

a = 10       (地址: 0x1000)
p = 0x1000   (地址: 0x2000)
pp = 0x2000  (地址: 0x3000)

1.3 实战应用

  • 动态二维数组
int rows = 3, cols = 4;
int **matrix = (int **)malloc(rows * sizeof(int *));
for (int i = 0; i < rows; i++) {matrix[i] = (int *)malloc(cols * sizeof(int));
}
  • // 使用 matrix[i][j]
  • // 释放内存略
  • 修改指针变量
void changePointer(int **pp) {int b = 20;*pp = &b;  // 修改外部指针的指向
}
int main() {int a = 10;int *p = &a;changePointer(&p);  // p 现在指向 breturn 0;
}

2. 函数指针

2.1 什么是函数指针?

函数指针是指向函数入口地址的指针,可用于实现回调机制、动态函数调用等。

2.2 声明与使用

int add(int a, int b) { return a + b; }
int sub(int a, int b) { return a - b; }
int main() {// 声明函数指针int (*op)(int, int);  op = add;  // 指向 add 函数printf("%d\n", op(2, 3));  // 输出 5op = sub;  // 切换指向 sub 函数printf("%d\n", op(5, 2));  // 输出 3return 0;
}

2.3 高级应用

  • 回调函数(Callback)
void process(int x, int y, int (*callback)(int, int)) {printf("Result: %d\n", callback(x, y));
}
int main() {process(10, 5, add);  // 输出 15process(10, 5, sub);  // 输出 5return 0;
}
  • 函数指针数组
int (*ops[2])(int, int) = {add, sub};
printf("%d\n", ops[0](3, 2));  // 调用 add
printf("%d\n", ops[1](3, 2));  // 调用 sub

3. 结构体指针

3.1 基本用法

结构体指针用于高效操作复杂数据结构,避免值传递的内存拷贝

typedef struct {int id;char name[20];
} Student;
int main() {Student s = {1, "Alice"};Student *p = &s;printf("%s\n", p->name);  // 使用 -> 访问成员return 0;
}

3.2 动态内存分配

Student *createStudent(int id, const char *name) {Student *s = (Student *)malloc(sizeof(Student));s->id = id;strcpy(s->name, name);return s;
}
int main() {Student *s = createStudent(2, "Bob");printf("%d: %s\n", s->id, s->name);free(s);  // 释放内存return 0;
}

3.3 链表实现

typedef struct Node {int data;struct Node *next;
} Node;
int main() {Node *head = NULL;// 插入节点Node *newNode = (Node *)malloc(sizeof(Node));newNode->data = 10;newNode->next = head;head = newNode;// 遍历链表略return 0;
}

4. 综合实战

案例:通用排序函数

// 比较函数
int cmpInt(const void *a, const void *b) {return *(int *)a - *(int *)b;
}
void sort(void *arr, int n, int size, int (*cmp)(const void *, const void *)) {for (int i = 0; i < n-1; i++) {for (int j = 0; j < n-i-1; j++) {void *p1 = (char *)arr + j * size;      // 计算元素地址void *p2 = (char *)arr + (j+1) * size;if (cmp(p1, p2) > 0) {swap(p1, p2, size);  // 交换内存块}}}
}
int main() {int arr[] = {5, 2, 8, 1};sort(arr, 4, sizeof(int), cmpInt);// 输出排序结果略return 0;
}

5. 避坑指南

问题类型错误示例修正方法
多级指针解引用错误***pp 无意义明确层级:**pp 或 *pp
函数指针类型不匹配int (*p)() 指向 void func()严格匹配返回值和参数类型
结构体指针未初始化Student *p; p->id=1;先分配内存或指向有效结构体变量
  1. 总结
  • 指向指针的指针:用于动态多维数据结构、跨函数修改指针。
  • 函数指针:实现回调、策略模式等灵活编程。
  • 结构体指针:高效操作复杂数据,实现链表/树等动态结构。
http://www.xdnf.cn/news/8796.html

相关文章:

  • 互联网大厂Java求职面试:AI与大模型应用集成中的架构难题与解决方案
  • 向量数据库选型实战指南:Milvus架构深度解析与技术对比
  • OPENEULER搭建私有云存储服务器
  • 使用 Python 库中自带的数据集来实现上述 50 个数据分析和数据可视化程序的示例代码
  • Go 语言基础1 Slice,map,string
  • 在PyCharm中使用pyenv指定的Python:配置指南
  • 机器学习--分类算法
  • xml双引号可以不转义
  • 【python实战】中国主要城市经济统计数据分析与预测
  • 力扣395做题笔记
  • 深度学习论文idea:多模态检索
  • 计算机网络总结(物理层,链路层)
  • 【深度学习】2. 从梯度推导到优化策略:反向传播与 SGD, Mini SGD
  • 最好用的wordpress外贸主题
  • 【概率论基本概念02】最大似然性
  • 正则表达式:字符串模式匹配的利器
  • Bochs下去运行linux-0.11
  • 云原生安全基石:深度解析HTTPS协议(从原理到实战)
  • 图论核心:深度搜索DFS 与广度搜索BFS
  • 【React】createPortal - 简单的Message和Modal组件
  • JAVA集合(含List、Map、Set)(超详细版)
  • 阿里云国际版香港轻量云服务器:CN2 GIA加持,征服海外网络的“速度与激情”!
  • 搭建 C/C++_CMake_Boost_git 开发环境
  • 【最新版】Arduino IDE的安装入门Demo
  • 异步编程与axios技术
  • 代码随想录算法训练营 Day53 图论Ⅳ 字符串接龙 有向图 岛屿周长
  • C#索引器详解:让对象像数组一样被访问
  • 自用git记录
  • Java反射机制详细笔记
  • 项目管理学习-CSPM4(2)