以int** a为例
1.二级指针的声明
-
a
是一个指向int*
(指向整型的指针)的指针,即二级指针。 -
通俗的讲,a是一个指向指针的指针,对a解引用会是一个指针。
-
它可以用于操作动态分配的二维数组、指针数组或需要间接修改指针的场景。
2.动态二维数组的表示
-
例如动态分配一个
N×M
的二维数组:int** a =(int**)malloc(N*sizeof(int*));for(int i=0;i<N;i++){*(a+i)=(int*)malloc(M*sizeof(int));//a[i]=(int*)malloc(M*sizeof(int));}
3.函数参数传递
-
当二级指针作为函数参数传递是,有以下几种意思:
-
1.作为指针数组使用:
-
void example(int** a,int size){for(int i=0;i<size;i++){a[i]=NULL;//初始化} }
2.作为二维数组使用:
-
void example(int** a,int size,int* aColSize){for(int i=0;i<size;i++){for(int j=0;j<aColSize[i];j++){a[i][j]=0;//初始化}} }
3.作为传回的指针使用:
-
void example(int** a){*a=NULL; }
和这段代码是一样的作用:
-
int* example(int* a){a=NULL;return a; }
注意:
-
1.混淆二维数组和二级指针
-
虽然二级指针作为函数参数传递可以作为二维数组使用,但不能将二维数组作为二级指针接收的参数使用!
-
例如以下是错误做法:
-
#include<stdio.h> void example(int** a,int size,int* aColSize){for(int i=0;i<size;i++){for(int j=0;j<aColSize[i];j++){a[i][j]=0;//初始化}} } int main() {int a[10][10],size=10,aColSize[10];for(int i=0;i<10;i++){aColSize[i]=10;for(int j=0;j<10;j++){scanf("%d",&a[i][j]);}}example(a,size,aColSize);//错误:企图将二维数组作为二级指针接收的参数 return 0; }
正确做法应该这样做:
-
#include<stdio.h> #include<stdlib.h> void example(int** a, int size, int* aColSize) {for (int i = 0; i < size; i++) {for (int j = 0; j < aColSize[i]; j++) {a[i][j] = 0; //初始化printf("%d_%d ", i, j);}} } int main() {int *a[10], size = 10, aColSize[10];for (int i = 0; i < size; i++) {a[i] = (int*)malloc(10*sizeof(int));}for (int i = 0; i < size; i++) {aColSize[i] = size;for (int j = 0; j < size; j++) {scanf("%d", &a[i][j]);}}example(a, size, aColSize);for(int i=0;i<size;i++){free(a[i]);}return 0; }
千万不要写成以下这样,虽然编译器可能不会报错,但向野指针指向的区域赋值是不可取的,程序会崩溃的。
-
#include<stdio.h> #include<stdlib.h> void example(int** a, int size, int* aColSize) {for (int i = 0; i < size; i++) {for (int j = 0; j < aColSize[i]; j++) {a[i][j] = 0; //初始化printf("%d_%d ", i, j);}} } int main() {int *a[10], size = 10, aColSize[10];//a数组未初始化for (int i = 0; i < size; i++) {aColSize[i] = size;for (int j = 0; j < size; j++) {scanf("%d", &a[i][j]);}}example(a, size, aColSize);return 0; }
-
二级指针用来作为动态二维数组时,分配的内存是不连续的,但静态分配的二维数组在内存上是连续的!因此,也不能将静态二维数组的首元素地址赋给二级指针!
-
以下是错误示例:
-
int arr[3][4];int **p = (int**)arr; // ❌ 编译通过但运行崩溃
2.动态内存释放
-
虽然程序一般在运行结束后会自动释放所用内存,但为保证程序长时间运行内存足够,因此用malloc等函数动态分配的内存在使用后要释放!一般称其为避免内存泄漏。养成用完释放的好习惯,避免工作时的项目出错。
-
但内存释放对于二级指针一定要有先后顺序:
-
以下是错误示范1:
-
void example(){int N,M;scanf("%d %d",&N,&M);int** a=(int**)malloc(N*sizeof(int*));for(int i=0;i<N;i++){a[i]=(int*)malloc(M*sizeof(int));}free(a);//只释放了a数组,但未释放a+i(0<i<10)这些数组! }
以下是错误示范2:
-
void example(){int N,M;scanf("%d %d",&N,&M);int** a=(int**)malloc(N*sizeof(int*));for(int i=0;i<N;i++){a[i]=(int*)malloc(M*sizeof(int));}free(a);for(int i=0;i<N;i++){free(a[i]);//由于a数组的内存已经释放,a[i]的指针信息丢失,无法释放内存!} }
以下是正确示例:
-
void example(){int N,M;scanf("%d %d",&N,&M);int** a=(int**)malloc(N*sizeof(int*));for(int i=0;i<N;i++){a[i]=(int*)malloc(M*sizeof(int));}for(int i=0;i<N;i++){free(a[i]);}free(a); }
-
关键要点总结
-
✅ 二级指针本质:指向指针的指针
-
✅ 正确使用场景:动态多维数组、需间接修改指针
-
⚠️ 常见错误:混淆静态/动态内存布局、未初始化指针
-
🔧 最佳实践:分配后立即初始化、使用完毕及时释放