二维数组在c语言中用指针遍历时需区分行指针和列指针。1. 行指针指向一行,如int (p)[4],+i跳转到第i行;2. 列指针指向具体元素,如int p,+j在当前行移动;3. 用行指针遍历时可通过arr+i获取行地址再加列索引;4. 用列指针遍历时可将数组视为一维,按i*列数+j计算偏移量;5. 常见误区包括混淆指针类型、错误计算偏移及对数组名误操作。理解结构与访问方式是关键。
二维数组和指针操作在C语言中是个容易混淆的地方,尤其是用指针遍历二维数组时,很多人会搞不清行指针和列指针的区别。其实只要理解了它们的结构和访问方式,就很容易掌握。
二维数组在内存中的布局
二维数组在内存里是按行优先顺序存储的,也就是说,先放第一行的所有元素,接着是第二行,依此类推。例如:
int arr[3][4] = { {1, 2, 3, 4}, {5, 6, 7, 8}, {9, 10, 11, 12} };
这个数组在内存中其实就是连续的12个整数:1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12。
所以,无论你是用下标还是用指针访问,最终都是按照这种线性结构来操作的。
行指针与列指针的区别
- 行指针:指向的是整个一行(比如
int (*p)[4]
就是指向一个含有4个整型元素的一维数组)
- 列指针:指向的是某一行中的某个元素(比如
int *p
)
举个例子:
int arr[3][4]; int (*rowPtr)[4] = arr; // 行指针,指向每一行 int *colPtr = arr[0]; // 列指针,指向第一个元素
使用上也有所不同:
-
rowPtr + i
会跳到第i行开头的位置
-
colPtr + j
是在当前行中移动j个位置
用行指针遍历二维数组
如果你用行指针来遍历,可以这样写:
for (int i = 0; i < 3; i++) { for (int j = 0; j < 4; j++) { printf("%d ", (* (arr + i))[j]); // 或者 (*(rowPtr + i))[j] } printf("n"); }
这里的关键点在于:
-
arr + i
得到的是第i行的地址
-
* (arr + i)
就是这一行的第一个元素的地址,也就是一维数组的首地址
- 再加上
[j]
就能取到具体的值
或者更简洁地配合行指针变量:
int (*rowPtr)[4] = arr; for (int i = 0; i < 3; i++) { for (int j = 0; j < 4; j++) { printf("%d ", (rowPtr[i])[j]); } printf("n"); }
用列指针遍历二维数组
如果你想用列指针的方式访问,也可以:
int *colPtr = &arr[0][0]; // 或者直接 int *colPtr = arr[0]; for (int i = 0; i < 3; i++) { for (int j = 0; j < 4; j++) { printf("%d ", *(colPtr + i * 4 + j)); } printf("n"); }
这种方式的本质是把二维数组当作一维来看待。注意这里的偏移量是
i * 4 + j
,其中4是列数。
你也可以每次只移动指针:
int *p = arr[0]; for (int i = 0; i < 12; i++) { printf("%d ", *(p + i)); if ((i + 1) % 4 == 0) printf("n"); }
常见误区提醒
- 混淆行指针和列指针的作用范围
- 忘记二维数组的列数,导致偏移计算错误
- 把二维数组名当成普通指针随意赋值(比如不能直接
arr++
)
基本上就这些。指针遍历二维数组不难,关键是搞清楚类型和偏移逻辑。多练几个例子就能熟练掌握了。
评论(已关闭)
评论已关闭