C語(yǔ)言 指針與二維數(shù)組詳解
二維數(shù)組在概念上是二維的,有行和列,但在內(nèi)存中所有的數(shù)組元素都是連續(xù)排列的,它們之間沒(méi)有“縫隙”。以下面的二維數(shù)組 a 為例:
int a[3][4] = { {0, 1, 2, 3}, {4, 5, 6, 7}, {8, 9, 10, 11} };
從概念上理解,a 的分布像一個(gè)矩陣:
0 1 2 3
4 5 6 7
8 9 10 11
但在內(nèi)存中,a 的分布是一維線(xiàn)性的,整個(gè)數(shù)組占用一塊連續(xù)的內(nèi)存:

C語(yǔ)言中的二維數(shù)組是按行排列的,也就是先存放 a[0] 行,再存放 a[1] 行,最后存放 a[2] 行;每行中的 4 個(gè)元素也是依次存放。數(shù)組 a 為 int 類(lèi)型,每個(gè)元素占用 4 個(gè)字節(jié),整個(gè)數(shù)組共占用 4×(3×4) = 48 個(gè)字節(jié)。
C語(yǔ)言允許把一個(gè)二維數(shù)組分解成多個(gè)一維數(shù)組來(lái)處理。對(duì)于數(shù)組 a,它可以分解成三個(gè)一維數(shù)組,即 a[0]、a[1]、a[2]。每一個(gè)一維數(shù)組又包含了 4 個(gè)元素,例如 a[0] 包含 a[0][0]、a[0][1]、a[0][2]、a[0][3]。
假設(shè)數(shù)組 a 中第 0 個(gè)元素的地址為 1000,那么每個(gè)一維數(shù)組的首地址如下圖所示:

為了更好的理解指針和二維數(shù)組的關(guān)系,我們先來(lái)定義一個(gè)指向 a 的指針變量 p:
int (*p)[4] = a;
括號(hào)中的*表明 p 是一個(gè)指針,它指向一個(gè)數(shù)組,數(shù)組的類(lèi)型為int [4],這正是 a 所包含的每個(gè)一維數(shù)組的類(lèi)型。
[ ]的優(yōu)先級(jí)高于*,( )是必須要加的,如果赤裸裸地寫(xiě)作int *p[4],那么應(yīng)該理解為int *(p[4]),p 就成了一個(gè)指針數(shù)組,而不是二維數(shù)組指針,這在《C語(yǔ)言指針數(shù)組》中已經(jīng)講到。
對(duì)指針進(jìn)行加法(減法)運(yùn)算時(shí),它前進(jìn)(后退)的步長(zhǎng)與它指向的數(shù)據(jù)類(lèi)型有關(guān),p 指向的數(shù)據(jù)類(lèi)型是int [4],那么p+1就前進(jìn) 4×4 = 16 個(gè)字節(jié),p-1就后退 16 個(gè)字節(jié),這正好是數(shù)組 a 所包含的每個(gè)一維數(shù)組的長(zhǎng)度。也就是說(shuō),p+1會(huì)使得指針指向二維數(shù)組的下一行,p-1會(huì)使得指針指向數(shù)組的上一行。
數(shù)組名 a 在表達(dá)式中也會(huì)被轉(zhuǎn)換為和 p 等價(jià)的指針!
下面我們就來(lái)探索一下如何使用指針 p 來(lái)訪(fǎng)問(wèn)二維數(shù)組中的每個(gè)元素。按照上面的定義:
1) p指向數(shù)組 a 的開(kāi)頭,也即第 0 行;p+1前進(jìn)一行,指向第 1 行。
2) *(p+1)表示取地址上的數(shù)據(jù),也就是整個(gè)第 1 行數(shù)據(jù)。注意是一行數(shù)據(jù),是多個(gè)數(shù)據(jù),不是第 1 行中的第 0 個(gè)元素,下面的運(yùn)行結(jié)果有力地證明了這一點(diǎn):
#include <stdio.h>
int main(){
int a[3][4] = { {0, 1, 2, 3}, {4, 5, 6, 7}, {8, 9, 10, 11} };
int (*p)[4] = a;
printf("%d\n", sizeof(*(p+1)));
return 0;
}
運(yùn)行結(jié)果:
16
3) *(p+1)+1表示第 1 行第 1 個(gè)元素的地址。如何理解呢?
*(p+1)單獨(dú)使用時(shí)表示的是第 1 行數(shù)據(jù),放在表達(dá)式中會(huì)被轉(zhuǎn)換為第 1 行數(shù)據(jù)的首地址,也就是第 1 行第 0 個(gè)元素的地址,因?yàn)槭褂谜袛?shù)據(jù)沒(méi)有實(shí)際的含義,編譯器遇到這種情況都會(huì)轉(zhuǎn)換為指向該行第 0 個(gè)元素的指針;就像一維數(shù)組的名字,在定義時(shí)或者和 sizeof、& 一起使用時(shí)才表示整個(gè)數(shù)組,出現(xiàn)在表達(dá)式中就會(huì)被轉(zhuǎn)換為指向數(shù)組第 0 個(gè)元素的指針。
4) *(*(p+1)+1)表示第 1 行第 1 個(gè)元素的值。很明顯,增加一個(gè) * 表示取地址上的數(shù)據(jù)。
根據(jù)上面的結(jié)論,可以很容易推出以下的等價(jià)關(guān)系:
a+i == p+i
a[i] == p[i] == *(a+i) == *(p+i)
a[i][j] == p[i][j] == *(a[i]+j) == *(p[i]+j) == *(*(a+i)+j) == *(*(p+i)+j)
【實(shí)例】使用指針遍歷二維數(shù)組。
#include <stdio.h>
int main(){
int a[3][4]={0,1,2,3,4,5,6,7,8,9,10,11};
int(*p)[4];
int i,j;
p=a;
for(i=0; i<3; i++){
for(j=0; j<4; j++) printf("%2d ",*(*(p+i)+j));
printf("\n");
}
return 0;
}
運(yùn)行結(jié)果:
0 1 2 3
4 5 6 7
8 9 10 11
指針數(shù)組和二維數(shù)組指針的區(qū)別
指針數(shù)組和二維數(shù)組指針在定義時(shí)非常相似,只是括號(hào)的位置不同:
int *(p1[5]); //指針數(shù)組,可以去掉括號(hào)直接寫(xiě)作 int *p1[5];
int (*p2)[5]; //二維數(shù)組指針,不能去掉括號(hào)
指針數(shù)組和二維數(shù)組指針有著本質(zhì)上的區(qū)別:指針數(shù)組是一個(gè)數(shù)組,只是每個(gè)元素保存的都是指針,以上面的 p1 為例,在32位環(huán)境下它占用 4×5 = 20 個(gè)字節(jié)的內(nèi)存。二維數(shù)組指針是一個(gè)指針,它指向一個(gè)二維數(shù)組,以上面的 p2 為例,它占用 4 個(gè)字節(jié)的內(nèi)存。
以上就是C語(yǔ)言 指針和二維數(shù)組的資料整理,后續(xù)繼續(xù)補(bǔ)充相關(guān)知識(shí),謝謝大家對(duì)本站的支持!
- 關(guān)于C語(yǔ)言一維數(shù)組算法問(wèn)題詳解
- C語(yǔ)言一維數(shù)組初步學(xué)習(xí)筆記
- 關(guān)于C語(yǔ)言中的指針與二維數(shù)組
- C語(yǔ)言二維數(shù)組指針的概念及使用
- C語(yǔ)言與java語(yǔ)言中關(guān)于二維數(shù)組的區(qū)別
- 從頭學(xué)習(xí)C語(yǔ)言之二維數(shù)組
- C語(yǔ)言面試C++二維數(shù)組中的查找示例
- C語(yǔ)言二維數(shù)組幾種常用的表示方法
- C語(yǔ)言二維數(shù)組中的查找的實(shí)例
- 直觀(guān)理解C語(yǔ)言中指向一位數(shù)組與二維數(shù)組的指針
- C語(yǔ)言中的一維數(shù)組與二維數(shù)組的實(shí)現(xiàn)
相關(guān)文章
Linux下用Valgrind做檢查(防止內(nèi)存泄露)
Valgrind是一款基于模擬linux下的程序調(diào)試器和剖析器的軟件套件,可以運(yùn)行于x86, amd64和ppc32架構(gòu)上。valgrind包含一個(gè)核心,它提供一個(gè)虛擬的CPU運(yùn)行程序,還有一系列的工具,它們完成調(diào)試,剖析和一些類(lèi)似的任務(wù)2014-01-01
C++實(shí)現(xiàn)LeetCode(136.單獨(dú)的數(shù)字)
這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(136.單獨(dú)的數(shù)字),本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-07-07
C語(yǔ)言詳細(xì)分析講解關(guān)鍵字const與volatile的用法
在C語(yǔ)言中,我們經(jīng)常會(huì)見(jiàn)到const和volatile這兩個(gè)關(guān)鍵字,那么我們今天就來(lái)介紹下這兩個(gè)關(guān)鍵字,提起?const?關(guān)鍵字,我們可能首先想到的是經(jīng)過(guò)它修飾的變量便是常量了。其實(shí)我們這種想法是錯(cuò)誤的,其實(shí)?const?修飾的變量是只讀的,其本質(zhì)還是變量2022-04-04
C語(yǔ)言中指針 int *p=0;和int *p;*p=0;和”&“的關(guān)系和區(qū)別詳解
這篇文章主要介紹了C語(yǔ)言中指針 int *p=0;和int *p;*p=0;和”&“有什么關(guān)系和區(qū)別,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-02-02
淺析C語(yǔ)言中對(duì)于char*和char[]的理解
char * s 只是一個(gè)保存字符串首地址的指針變量,char a[]是許多連續(xù)的內(nèi)存單元,單元中的元素是char型,char * 和 char a[]具有相同的效果,源于字符串的本質(zhì),這篇文章主要介紹了C語(yǔ)言中對(duì)于char*和char[]的理解,需要的朋友可以參考下2023-02-02
C語(yǔ)言驅(qū)動(dòng)開(kāi)發(fā)之內(nèi)核使用IO/DPC定時(shí)器詳解
本章將繼續(xù)探索驅(qū)動(dòng)開(kāi)發(fā)中的基礎(chǔ)部分,定時(shí)器在內(nèi)核中同樣很常用,在內(nèi)核中定時(shí)器可以使用兩種,即IO定時(shí)器,以及DPC定時(shí)器,感興趣的可以了解一下2023-04-04
基于C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)單的掃雷小游戲
這篇文章主要為大家詳細(xì)介紹了基于C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)單的掃雷小游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-11-11

