十五届省二无颜面对江东父老
痛定思痛~~ 这个系列打算以指针开个头,欢迎大家提出备赛建议!!!
目录
🌸个人主页:Yang-ai-cao
📕系列专栏:蓝桥杯 C语言
🍍博学而日参省乎己,知明而行无过矣
希望通过这篇详细的介绍,咱们能更好地理解C语言中的指针和内存管理。如果咱在学习过程中遇到任何问题,或者有任何疑问,请随时告诉我,我会尽力帮助咱。祝咱在编程的道路上不断进步!
前言
在C语言编程中,指针和内存管理是非常重要的概念,掌握这些内容不仅能提升咱的编程能力,还能帮助咱编写更高效、灵活的代码。虽然这些概念可能听起来有些复杂,但请放心,我会尽量用通俗易懂的语言来解释,希望能帮助咱更好地理解。如果有任何不清楚的地方,欢迎随时提问和指正!
一、指针的基本概念
“指针”是一个变量,它存储另一个变量的内存地址。
int a = 10;
int *p = &a; // p 是一个指针,存储了变量 a 的地址
1.1 指针的声明和初始化
int *p; // 声明一个指向 int 类型的指针 p
int a = 10;
p = &a; // 将变量 a 的地址赋值给指针 p
1.2 指针的解引用
通过指针访问其指向的变量的值,称为解引用。
int a = 10;
int *p = &a;
printf("%d\n", *p); // 输出 10
*p = 20; // 修改 a 的值为 20
printf("%d\n", a); // 输出 20
二、 指针运算
指针可以进行一些运算,如加减运算、比较运算等。
1、指针的加减运算
指针加减整数会移动指针指向的位置,移动的步长取决于指针的类型。
int arr[5] = {1, 2, 3, 4, 5};
int *p = arr;
p++; // p 现在指向 arr[1]
printf("%d\n", *p); // 输出 2
2、指针的比较运算
指针可以进行比较运算,比较的是指针所指向的内存地址。
int a = 10, b = 20;
int *p1 = &a, *p2 = &b;
if (p1 < p2) {
printf("p1 指向的地址小于 p2 指向的地址\n");
}
三、 指针数组
指针数组是一个数组,其元素是指针。
int a = 1, b = 2, c = 3;
int *arr[3] = {&a, &b, &c};
for (int i = 0; i < 3; i++) {
printf("%d\n", *arr[i]); // 输出 1, 2, 3
}
四、函数指针
函数指针是指向函数的指针,可以用来调用函数或实现回调函数。
1、函数指针的声明和使用
int add(int x, int y) {
return x + y;
}
int main() {
int (*func_ptr)(int, int) = add; // 声明一个函数指针并初始化
printf("%d\n", func_ptr(2, 3)); // 输出 5
return 0;
}
2、函数指针数组
int add(int x, int y) { return x + y; }
int subtract(int x, int y) { return x - y; }
int main() {
int (*func_ptr[2])(int, int) = {add, subtract};
printf("%d\n", func_ptr[0](5, 3)); // 输出 8
printf("%d\n", func_ptr[1](5, 3)); // 输出 2
return 0;
}
五、动态内存分配和释放
C语言提供了动态内存管理函数,可以在运行时分配和释放内存。
1、"malloc"和 "free"
"malloc" 用于分配指定大小的内存,返回指向分配内存的指针;`free` 用于释放动态分配的内存。
#include <stdlib.h>
int main() {
int *p = (int *)malloc(5 * sizeof(int)); // 分配内存
if (p == NULL) {
printf("内存分配失败\n");
return 1;
}
for (int i = 0; i < 5; i++) {
p[i] = i + 1; // 使用分配的内存
}
for (int i = 0; i < 5; i++) {
printf("%d\n", p[i]); // 输出 1, 2, 3, 4, 5
}
free(p); // 释放内存
return 0;
}
2、 "calloc"
"calloc`"用于分配内存并初始化为零。
#include <stdlib.h>
int main() {
int *p = (int *)calloc(5, sizeof(int)); // 分配并初始化内存
if (p == NULL) {
printf("内存分配失败\n");
return 1;
}
for (int i = 0; i < 5; i++) {
printf("%d\n", p[i]); // 输出 0
}
free(p); // 释放内存
return 0;
}
3、 "realloc"
"realloc`"用于调整已分配内存的大小。
#include <stdlib.h>
int main() {
int *p = (int *)malloc(5 * sizeof(int));
if (p == NULL) {
printf("内存分配失败\n");
return 1;
}
for (int i = 0; i < 5; i++) {
p[i] = i + 1;
}
p = (int *)realloc(p, 10 * sizeof(int)); // 调整内存大小
if (p == NULL) {
printf("内存调整失败\n");
return 1;
}
for (int i = 5; i < 10; i++) {
p[i] = i + 1;
}
for (int i = 0; i < 10; i++) {
printf("%d\n", p[i]); // 输出 1, 2, 3, 4, 5, 6, 7, 8, 9, 10
}
free(p); // 释放内存
return 0;
}
六、指针的地址与地址的指针
1、 指针的地址
指针本身也是一个变量,因此它也有自己的内存地址。可以使用&
运算符获取指针的地址。
int a = 10;
int *p = &a;
int **pp = &p; // pp 是一个指向指针 p 的指针
printf("a 的值: %d\n", a); // 输出 10
printf("a 的地址: %p\n", &a); // 输出 a 的地址
printf("p 的值 (a 的地址): %p\n", p); // 输出 a 的地址
printf("p 的地址: %p\n", &p); // 输出 p 的地址
printf("pp 的值 (p 的地址): %p\n", pp); // 输出 p 的地址
printf("pp 的地址: %p\n", &pp); // 输出 pp 的地址
在这个例子中:
a
是一个整数变量,其值为 10。p
是一个指向a
的指针,存储了a
的地址。pp
是一个指向p
的指针,存储了p
的地址。
2、地址的指针
地址的指针是指向某个内存地址的指针变量。例如,int *p
是一个指向int
类型变量地址的指针。
int a = 10;
int *p = &a; // p 是一个指向 int 类型变量 a 地址的指针
printf("a 的值: %d\n", *p); // 通过解引用 p,输出 a 的值 10
在这个例子中:
p
是一个指向a
的指针,通过*p
可以访问a
的值。
3、多级指针
多级指针是指指向指针的指针,可以扩展到更高级别。
int a = 10;
int *p = &a; // p 是一个指向 a 的指针
int **pp = &p; // pp 是一个指向 p 的指针
int ***ppp = &pp; // ppp 是一个指向 pp 的指针
printf("a 的值: %d\n", ***ppp); // 通过三级解引用,输出 a 的值 10
在这个例子中:
ppp
是一个指向pp
的指针,通过三级解引用***ppp
可以访问a
的值。
总结
1. 指针的基本概念
- 指针是一个变量,存储另一个变量的内存地址。
- 通过指针可以访问和修改其指向的变量值,这称为解引用。
2. 指针运算
- 指针可以进行加减运算,移动指针指向的位置。
- 指针可以进行比较运算,比较的是指针所指向的内存地址。
3. 指针数组
- 指针数组是一个数组,其元素是指针,用于存储多个变量的地址。
4. 函数指针
- 函数指针是指向函数的指针,可以用来调用函数或实现回调函数。
- 可以创建函数指针数组,存储多个函数指针。
5. 动态内存分配和释放
malloc
用于分配指定大小的内存,free
用于释放动态分配的内存。calloc
用于分配内存并初始化为零。realloc
用于调整已分配内存的大小。
6.
- 指针的地址:每个指针变量都有自己的内存地址,可以通过
&
运算符获取。 - 地址的指针:指针变量存储的是某个变量的地址,通过解引用可以访问该变量的值。
- 多级指针:指向指针的指针,可以扩展到更高级别,通过多级解引用可以访问最终指向的变量。