0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

所有C語言數(shù)組和指針的知識(shí)都在這里了!|周立功手把手教你學(xué)C語言編程

AGk5_ZLG_zhiyua ? 來源:未知 ? 作者:電子大兵 ? 2017-09-01 09:28 ? 次閱讀

第一章為程序設(shè)計(jì)基礎(chǔ),本文為1.6.1數(shù)組與指針。

>>>1.6.1數(shù)組與指針

1.數(shù)組

(1)數(shù)組的聲明

我們知道,一個(gè)基本數(shù)據(jù)類型的變量只能存儲(chǔ)一個(gè)數(shù)據(jù),比如:

int data = 0x64;

如果需要存儲(chǔ)一組int型數(shù)據(jù)呢?比如,1、2、3,則至少需要3個(gè)變量data0、data1、data2。比如:

int data0 = 1, data1 = 2, data2 = 3;

由于數(shù)據(jù)的表現(xiàn)形式多種多樣,還有字符型和其它的數(shù)值類型,因此僅有基本數(shù)據(jù)類型是不夠的。是否可以通過基本數(shù)據(jù)類型的組合抽象構(gòu)造其它的數(shù)據(jù)類型呢?答案是可以的,構(gòu)造數(shù)據(jù)類型數(shù)組就是這樣產(chǎn)生的。

從概念的視角來看,int型整數(shù)1、2和3都是相同的數(shù)據(jù)類型,data0、data1和data2三個(gè)變量的共性是data,其差異性是下標(biāo)不一樣。因此可以將data0、data1和data2抽象為一個(gè)名字,然后用下標(biāo)區(qū)分這些變量的集合——data[0]、data[1]和data[2]。如果有以下聲明:

int data[3]; //解讀為data是int數(shù)組(元素個(gè)數(shù)3)

那么data[3]就成了存放3個(gè)int型數(shù)據(jù)1、2、3的data[0]、data[1]和data[2]所組成的數(shù)組,即可分別對(duì)data[0]、data[1]和data[2]賦值:

data[0] = 1, data[1] = 2, data[2] = 3;

當(dāng)然,也可以按照以下方式聲明一個(gè)數(shù)組并進(jìn)行初始化:

int data[3] = {1, 2, 3};

通常將data稱為數(shù)組(變量)名,data[0]、data[1]和data[2]被稱為變量。因而可以說,數(shù)組是將相同類型數(shù)據(jù)的若干變量按有序的形式組織起來,用一個(gè)名字命名,然后用下標(biāo)區(qū)分這些變量的集合。

由于數(shù)組是建立在其它類型的基礎(chǔ)上,因此C將數(shù)組看作構(gòu)造類型,在聲明數(shù)組時(shí)必須說明其元素的類型。比如,int類型的數(shù)組、float類型的數(shù)組或其它類型的數(shù)組。而其它類型也可以是數(shù)組類型,在這種情況下,創(chuàng)建的是數(shù)組類型的數(shù)組,簡(jiǎn)稱數(shù)組的數(shù)組。

(2)下標(biāo)與變量的值

在這里,定義了一個(gè)名為data的數(shù)組類型變量,它是由存放3個(gè)int型數(shù)據(jù)1、2、3的變量data[0]、data[1]和data[2]組成的。通常又將數(shù)組的各個(gè)變量稱為數(shù)組的元素,而數(shù)組的元素是按照順序編號(hào)的,這些元素的編號(hào)又稱為數(shù)組元素的下標(biāo)。

由于有了下標(biāo),因此數(shù)組元素在內(nèi)存中的位置就被唯一確定下來了。下標(biāo)總是從0開始的,最后一個(gè)元素的下標(biāo)為元素的個(gè)數(shù)減1(即2),data[0]叫第1個(gè)元素,data[1]叫第2個(gè)元素,data[2]叫第3個(gè)元素,也就意味著所有的元素在內(nèi)存中都是連續(xù)存儲(chǔ)的。

直觀上,數(shù)組是由下標(biāo)(或稱為索引)和值所組成的序?qū)?index, value>集合,對(duì)于每個(gè)有定義的下標(biāo)都存在一個(gè)與其關(guān)聯(lián)的值,在數(shù)學(xué)上稱為映射。除了創(chuàng)建新數(shù)組外,大多數(shù)語言對(duì)數(shù)組只提供兩種標(biāo)準(zhǔn)操作:一個(gè)操作是檢索一個(gè)值,另一個(gè)操作是存儲(chǔ)一個(gè)值。

函數(shù)Create(data, size)創(chuàng)建一個(gè)新的具有適當(dāng)大小的空數(shù)組,初始時(shí)數(shù)組的每一項(xiàng)都沒有定義。Retrieve操作接受一個(gè)數(shù)組data和一個(gè)下標(biāo)index,如果下標(biāo)合法,則該操作返回與下標(biāo)關(guān)聯(lián)的值,否則產(chǎn)生一個(gè)錯(cuò)誤。Store操作接受一個(gè)數(shù)組data、一個(gè)下標(biāo)index和一個(gè)項(xiàng)item的集合,即項(xiàng)是value值的集合,有時(shí)也將值(value)稱為項(xiàng)(item),返回在原來數(shù)組中增加新的序?qū)?index, value>后的數(shù)組。

顯然,int系的任何常量表達(dá)式都可以作為數(shù)組元素的下標(biāo)。比如:

int array[3+5]; //合法

int array['a']; //表示int array[97];

上述定義之所以合法,因?yàn)楸硎驹貍€(gè)數(shù)的常量表達(dá)式在編譯時(shí)就具有確定的意義,與變量的定義一樣明確地分配了固定大小的空間。

雖然使用符號(hào)常量增強(qiáng)了數(shù)組的靈活性,但如果定義采用了以下的形式:

int n = 5;

int array[n]; //非法

因?yàn)闃?biāo)準(zhǔn)C認(rèn)為數(shù)組元素的個(gè)數(shù)n不是常量,雖然編譯器似乎已經(jīng)“看到”了n的值,但int array[n]要在運(yùn)行時(shí)才能讀取變量n的值,所以在編譯期無法確定其空間大小。使用符號(hào)常量定義數(shù)組長(zhǎng)度的正確形式如下:

#define N 10

int array[N];

即可根據(jù)實(shí)際的需要修改常量N的值。

由于數(shù)組元素下標(biāo)的有效范圍為0~N-1,因此data[N]是不存在的,但C語言并不檢查下標(biāo)是否越界。如果訪問了數(shù)組末端之后的元素,訪問的就是與數(shù)組不相關(guān)的內(nèi)存。它不是數(shù)組的一部分,使用它肯定會(huì)出問題。C為何允許這種情況發(fā)生呢?這要?dú)w功于C信任程序員,因?yàn)椴粰z查越界可以使運(yùn)行速度更快,所以編譯器沒有必要檢查所有的下標(biāo)錯(cuò)誤。因?yàn)樵诔绦蜻\(yùn)行之前,數(shù)組的下標(biāo)可能尚未確定,所以為了安全起見,編譯器必須在運(yùn)行時(shí)添加額外代碼檢查數(shù)組的每個(gè)下標(biāo)值,但這樣會(huì)降低程序的運(yùn)行速度。C相信程序員能編寫正確的代碼,這樣的程序運(yùn)行速度更快。但并不是所有的程序員都能做到這一點(diǎn),越界恰恰是初學(xué)者最容易犯的錯(cuò)誤,因此要特別注意下標(biāo)的范圍不能超出合理的界限。

(3)變量的地址與類型

當(dāng)將變量data[0]、data[1]和data[2]作為&的操作數(shù)時(shí),&data[0]是指向變量data[0]的指針,&data[1]是指向變量data[1]的指針,&data[2]是指向變量data[2]的指針。data[0]、data[1]和data[2]變量的類型為int,&data[0]、&data[1]和&data[2]指針的類型為int *const,即指向常量的指針,簡(jiǎn)稱常量指針,其指向的值不可修改。比如:

int a;

int * const ptr = &a;

ptr = NULL; //試圖修改,則編譯報(bào)警

&a = NULL; //試圖修改,則編譯報(bào)警

同理,&data是指向變量data的指針,那么data是什么類型?

按照聲明變量的規(guī)約,將標(biāo)識(shí)符data取出后,剩下的“int [3]”就是data的類型,通常將其解釋為由3個(gè)int組成的數(shù)組類型,簡(jiǎn)稱數(shù)組類型。其目的是告訴編譯器需要分配多少內(nèi)存?3個(gè)元素的整數(shù)數(shù)組,data類型測(cè)試程序詳見程序清單 1.20。

程序清單1.20data類型測(cè)試程序

1 #include

2

3 void f(int x);

4 int main(int argc, char *argv[])

5 {

6 int data[3];

7 f(data);

8 return 0;

9 }

通過編譯器提示的警告,“funtion: 'int' differ in levels of indirection from 'int [3]'”,說明數(shù)組變量data的類型為不是int而是int [3]數(shù)組類型。由于在設(shè)計(jì)C語言時(shí),過多地考慮了開發(fā)編譯器的便利。雖然設(shè)計(jì)編譯器更方便了,卻因?yàn)楦拍畹哪:o初學(xué)者造成了理解上的困難。實(shí)際上數(shù)組應(yīng)該這樣定義:

int [3] data;

即int是與[3]結(jié)合的。&data到底是什么類型?

當(dāng)data作為&的操作數(shù)時(shí),則&data是指向data的指針。由于data的類型為int [3],因此&data是指向“int [3]數(shù)組類型”變量data的指針,簡(jiǎn)稱數(shù)組指針。其類型為int (*)[3],即指向int [2]的指針類型。為何要用“()”將“*”括起來?

如果不用括號(hào)將星號(hào)括起來,那么“int (*)[3]”就變成了“int *[3]”,而int *[3]類型名為指向int的指針的數(shù)組(元素個(gè)數(shù)3)類型,這是設(shè)計(jì)編譯器時(shí)約定的語法規(guī)則。

&data的類型到底是不是“int (*)[3]”?其驗(yàn)證程序范例詳見程序清單 1.21。

程序清單1.21&data類型測(cè)試程序

1 #include

2 int main(int argc, char *argv[])

3 {

4 int data[3];

5 int b = &data;

6 return 0;

7 }

通過編譯器提示的警告,“'int' differ in levels of indirection from 'int (*)[3]'”,說明&data的類型為int (*)[3]。

(4)sizeof(data)

當(dāng)data作為sizeof的操作數(shù)時(shí),其返回的是整個(gè)數(shù)組的長(zhǎng)度。在這里,sizeof(data)的大小為12,即3個(gè)元素占用的字節(jié)數(shù)為4×3=12,系統(tǒng)會(huì)認(rèn)為&data+1中的“1”,偏移了一個(gè)數(shù)組的大小,因此&data +1是下一個(gè)未知的存儲(chǔ)空間的地址(即越界)。在小端模式下,數(shù)組在內(nèi)存中的存儲(chǔ)方式詳見圖 1.10。

1.10數(shù)組的存儲(chǔ)

將如何尋找相應(yīng)的數(shù)組元素呢?常用的方法是通過“數(shù)組的基地址+偏移量”算出數(shù)組元素的地址。在這里,第一個(gè)元素&data[0]的地址稱為基地址,其偏移量就是下標(biāo)值和每個(gè)元素的大小sizeof(int)相乘。假設(shè)數(shù)組元素&data[0] 的地址為A,且在內(nèi)存中的實(shí)際地址為0x22FF74,那么&data[1]的值為:

A + 1×sizeof(int) = (unsigned int)data + 4 = 0x22FF74 + 4 = 0x22FF78

&data[2]的值為:

A + 2×sizeof(int) = (unsigned int)data + 8 = 0x22FF74 + 8 = 0x22FF7C

實(shí)際上,當(dāng)在C語言中書寫data[i]時(shí),C將它翻譯為一個(gè)指向int的指針。Data是指向data[0]的指針,data+i是指向data[i]的,因此不管data數(shù)組是什么類型,總有data+i等于data[i],于是*(data+i)等于data[i],其相應(yīng)的測(cè)試范例程序詳見程序清單 1.22。

程序清單1.22變量的地址測(cè)試程序

1 #include

2 int main(int argc, char *argv[])

3 {

4 int data[3]= {1, 2, 3};

5 printf("%x, %x, %x, %x, %x", &data[0], &data[1], &data[2], &data, &data+1);

6 return 0;

7 }

實(shí)踐證明,雖然&data[0]與&data的類型不一樣,但它們的值相等。同時(shí)也可以看出,數(shù)組的元素是連續(xù)存儲(chǔ)的。如果將數(shù)組變量占用內(nèi)存的大小除以數(shù)組變量中一個(gè)元素所占用空間的大小,便可得到數(shù)組元素的個(gè)數(shù)。即:

int numData = sizeof(data) / sizeof(data[0]);

當(dāng)然,也可以使用宏定義計(jì)算數(shù)組元素的個(gè)數(shù):

#define NELEMS(data) (sizeof(data) /sizeof(data[0])

當(dāng)數(shù)組作為函數(shù)的參數(shù)時(shí),C語言函數(shù)的所有參數(shù)必須在函數(shù)內(nèi)部聲明。但是,由于在函數(shù)內(nèi)部并沒有給數(shù)組分配新的存儲(chǔ)空間,因此一維數(shù)組的容量只在主程序中定義。顯然,如果函數(shù)需要得到一維數(shù)組的大小,則必須將它以函數(shù)參數(shù)的形式傳入函數(shù)中,或?qū)⑺鳛槿肿兞吭L問。

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 指針
    +關(guān)注

    關(guān)注

    1

    文章

    480

    瀏覽量

    70576
  • C語言編程
    +關(guān)注

    關(guān)注

    6

    文章

    90

    瀏覽量

    21110
  • 周立功
    +關(guān)注

    關(guān)注

    38

    文章

    130

    瀏覽量

    37660
  • 數(shù)組
    +關(guān)注

    關(guān)注

    1

    文章

    417

    瀏覽量

    25968

原文標(biāo)題:周立功:C語言數(shù)組與指針詳解

文章出處:【微信號(hào):ZLG_zhiyuan,微信公眾號(hào):ZLG致遠(yuǎn)電子】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    手把手教你學(xué)單片機(jī)之AVR入門視頻教程

    第02講 AVR硬件電路設(shè)計(jì)教程_手把手教你學(xué)單片機(jī)之AVR入門篇第03講 AVR開發(fā)基礎(chǔ)知識(shí)_手把手
    發(fā)表于 03-02 11:04

    手把手教你學(xué)單片機(jī)c語言》視頻教程

    `《手把手教你學(xué)單片機(jī)C語言》視頻教程電驢、優(yōu)酷、迅雷等下載地址《手把手
    發(fā)表于 10-26 14:14

    力天手把手教你學(xué)單片機(jī)視頻全集下載

    下--力天手把手教你學(xué)單片機(jī)之入門篇.rarhttp://115.com/file/an9yqmx3#07.第四講.C語言基礎(chǔ)
    發(fā)表于 02-14 17:06

    手把手教你學(xué)單片機(jī)c語言 郭天祥

    手把手教你學(xué)單片機(jī)c語言》視頻教程 https://bbs.elecfans.com/forum.php?mod=viewthread&t
    發(fā)表于 06-01 22:46

    期待宋老師的《手把手教你學(xué)51單片機(jī)-C語言版》

    看過宋老師的《手把手教你學(xué)51單片機(jī)-C語言版》,感覺挺好的,如果有書的教材那就更完美了!
    發(fā)表于 07-17 15:43

    今天收到關(guān)于《手把手教你學(xué)51單片機(jī)-C語言版》搶樓送書

    ``今天收到宋老師《手把手教你學(xué)51單片機(jī)-C語言版》這本書,感覺書還不錯(cuò)啊,{:12:}最近
    發(fā)表于 08-01 15:14

    手把手教你學(xué)51單片機(jī)-C語言版》公布獲獎(jiǎng)樓層

    學(xué)過其他人的教程,也可以通過這個(gè)教程來提高一下眼界,收益肯定不會(huì)少。小編推薦:《手把手教你學(xué)51單片機(jī)(C語言版)》的創(chuàng)作,目的就是改變當(dāng)前
    發(fā)表于 08-04 16:51

    手把手教你學(xué)CPLD/FPGA與單片機(jī)聯(lián)合設(shè)計(jì)》-興華

    。此外,為了幫助讀者掌握單片機(jī)與CPLD/FPGA的聯(lián)合設(shè)計(jì),還介紹51單片機(jī)的基本知識(shí)及單片機(jī)c語言編程的基礎(chǔ)
    發(fā)表于 12-29 17:10

    手把手教你學(xué)CPLD/FPGA與單片機(jī)聯(lián)合設(shè)計(jì)》-興華

    。此外,為了幫助讀者掌握單片機(jī)與CPLD/FPGA的聯(lián)合設(shè)計(jì),還介紹51單片機(jī)的基本知識(shí)及單片機(jī)c語言編程的基礎(chǔ)
    發(fā)表于 01-06 17:21

    【下載】手把手教你學(xué)DSP:基于TMS320C55x

    過程,所有代碼都標(biāo)注有詳細(xì)的中文注釋,為讀者快速熟悉并掌握DSP的開發(fā)方法和技巧提供方便?!?b class='flag-5'>手把手教你學(xué)DSP:基于TMS320X281x
    發(fā)表于 06-02 16:40

    手把手教你學(xué)PIC單片機(jī)C語言

    手把手教你學(xué)PIC單片機(jī)C語言教程第1課- PICHL-K18- PICHL-K18慧凈電子--做人人都買得起的PPIICC單片機(jī)開發(fā)板真誠(chéng)
    發(fā)表于 07-15 09:23

    指針數(shù)組

    手把手教你C語言難點(diǎn)編程,很好的C語言
    發(fā)表于 03-25 15:30 ?1次下載

    手把手教你如何開始DSP編程

    手把手教你如何開始DSP編程
    發(fā)表于 04-09 11:54 ?12次下載
    <b class='flag-5'>手把手</b><b class='flag-5'>教你</b>如何開始DSP<b class='flag-5'>編程</b>

    手把手教你學(xué)LabVIEW視覺設(shè)計(jì)

    手把手教你學(xué)LabVIEW視覺設(shè)計(jì)手把手教你學(xué)LabVIEW視覺設(shè)計(jì)
    發(fā)表于 03-06 01:41 ?3143次閱讀

    手把手教你學(xué)FPGA仿真

    電子發(fā)燒友網(wǎng)站提供《手把手教你學(xué)FPGA仿真.pdf》資料免費(fèi)下載
    發(fā)表于 10-19 09:17 ?2次下載
    <b class='flag-5'>手把手</b><b class='flag-5'>教你</b><b class='flag-5'>學(xué)</b>FPGA仿真