http://www.jb51.net/article/82709.htm
2016
一、關(guān)于結(jié)構(gòu)體的聲明
1、匿名聲明。如:
說明:
這段代碼的含義是,聲明一個無名(anonymous)的結(jié)構(gòu)體,并創(chuàng)建了一個結(jié)構(gòu)體變量point。如果這段聲明是放在全局域(在任意函數(shù)(比如main函數(shù))外)內(nèi),那么point內(nèi)的變量將被初始化為默認值,換句話說,以這種方式聲明結(jié)構(gòu)體變量時就已經(jīng)為它分配了內(nèi)存空間。
適用于該結(jié)構(gòu)體只需要產(chǎn)生一個變量!本例中,該匿名結(jié)構(gòu)體將有且僅有point這個結(jié)構(gòu)體變量!
不同的匿名結(jié)構(gòu)體變量,類型是不同的!如
如果將 p1=p2 ,則ok;如果將 p1=p3 ,則編譯器提示"incompatible types when assigning to type ‘struct <anonymous>' from type ‘struct <anonymous>'",兩者的實際類型是不一樣的。
2、顯式聲明一個結(jié)構(gòu)體
聲明了一個結(jié)構(gòu)體 struct node,如果需要聲明一個它的對象,則可以這樣:struct node n1;
可以聲明多個該結(jié)構(gòu)體的變量。
區(qū)別"C中的結(jié)構(gòu)體變量" 和 "Java中的類對象"。C中,"struct node n1;"創(chuàng)建了一個結(jié)構(gòu)體變量,并為它分配了內(nèi)存空間,不一定初始化!得看這個變量是否在全局域;而Java中,"Node n1;"只是聲明了一個類對象,也就是說是一個"空引用",可以想象成C中的空指針,當(dāng)"n1 = new Node();"時,n1才指向了該對象的內(nèi)存空間。因此,在Java中,可以通過"n1==null"來判斷對象是否為空;在C中,不能通過"n1==NULL"來判斷,因為"n1"并不是一個指針,而是一個類型變量的名字,就像"int a;"這種,顯然"a"不是指針!
3、用typedef來簡化結(jié)構(gòu)體的寫法
相當(dāng)于把代碼改名為Node了。以前需要這樣聲明"struct node n1;",現(xiàn)在只需要"Node n1;"。
這段代碼中,如果沒有typedef,代碼的意思是"聲明了一個匿名結(jié)構(gòu)體變量"!注意區(qū)別。
4、在結(jié)構(gòu)體中聲明結(jié)構(gòu)體變量。
這段代碼是錯誤的!
錯誤1:直接在結(jié)構(gòu)體中聲明另外一個結(jié)構(gòu)體,會出現(xiàn)死循環(huán),如A包括B,B又包括A,A又包括B……使得編譯器無法知道結(jié)構(gòu)體的空間大小,因此,無法通過編譯!
錯誤2:typedef還沒有將結(jié)構(gòu)體命名為Node,你就在結(jié)構(gòu)體中使用了Node,顯然,編譯器此時還不知到Node是什么!所以,無法通過編譯!
正確的使用方法如下:
二、關(guān)于結(jié)構(gòu)體的賦值
1、聲明一個變量后的默認值
如前面提到的,如果這個變量聲明是在全局,則"str.p等于NULL,str.i等于0,str.ch數(shù)組都是'\0'",為默認初始值;如果不在全局,則所有值都是"野值"。
2、手動初始化
此時,str2聲明時手動賦了初值。str2.p和str2.ch賦值時的行為是不一樣的!str2.p是一個字符指針,也就是將p指向常量字符串"abc"在內(nèi)存中的地址;而str2.ch是一個常量字符指針(無法操作指針),代表的是字符數(shù)組,也就是將常量字符串"def"逐字符copy到ch數(shù)組里,賦值結(jié)束后,ch的值是:'d','e','f','\0','\0'……
也可以像str3這樣初始化結(jié)構(gòu)體中的某些變量,值得注意的是str4和str5。對于數(shù)組(如 char a[size])來說,傳遞給常量字符指針,可以是"a",可以是"a[n]"(0<=n<size,編譯器會忽略掉n),不能是"a[size]"(編譯器會檢測,報"array index in initializer exceeds array bounds")。
3、賦值
或者
4、動態(tài)生成結(jié)構(gòu)體變量
注意,如果是動態(tài)生成的結(jié)構(gòu)體變量(用到了malloc),則必須在丟棄該變量前將他的內(nèi)存空間釋放掉(用free)。
如果結(jié)構(gòu)體內(nèi)部也存在動態(tài)生成的對象,在釋放結(jié)構(gòu)體之前要先釋放掉其內(nèi)部的內(nèi)存空間,如下
三、結(jié)構(gòu)體數(shù)組
我們知道基本數(shù)據(jù)類型的變量數(shù)組直接定義就可以分配空間了,結(jié)構(gòu)體可以看作一種新類型,它也是定義聲明變量之后就會自動分配空間的,結(jié)構(gòu)體的數(shù)組也是這樣。
這樣就定義了一個有10個book變量的數(shù)組,并且已經(jīng)分配了存儲空間。 結(jié)構(gòu)體的數(shù)組和普通數(shù)組索引方式是一樣的。
結(jié)構(gòu)體數(shù)組也可以使用字面量初始化方法,如下
是不是很方便了。要注意最外面的是 { ,不是 [ 哦。 對于成員是一個結(jié)構(gòu)的結(jié)構(gòu)體變量,同樣可以使用字面量初始化,記住,只是初始化,不能用于對結(jié)構(gòu)體變量的賦值哦。
四、指向結(jié)構(gòu)的指針
指向結(jié)構(gòu)體的指針是一個一直都沒有掌握好的點,希望這里能記錄好一點,加強理解。
對于指針有幾個好處,第一:就像指向數(shù)組的指針比數(shù)組本身更容易操作一樣,指向結(jié)構(gòu)的指針通常也更容易操作; 第二:在早期的C中參數(shù)傳遞只能使用結(jié)構(gòu)的指針;第三:很多奇妙的數(shù)據(jù)表示都是用了包含指向其他結(jié)構(gòu)的指針的結(jié)構(gòu)。
和數(shù)組不同,結(jié)構(gòu)的名字不是該結(jié)構(gòu)的地址(即單獨的結(jié)構(gòu)名并不是該結(jié)構(gòu)地址的同義詞),必須使用 & 運算符。聲明一個指針的方式與一個普通變量沒有什么區(qū)別:
假設(shè) lib 是一個 struct book 的數(shù)組,現(xiàn)在用結(jié)構(gòu)指針 cpp 指向 lib[0],那么根據(jù)指針的運算規(guī)則, cpp+1 會指向 lib[1]。雖然在一般的認識里面,結(jié)構(gòu)體中的元素在存儲器中是一次排列的,所以可以根據(jù)各個元素的大小來計算 cpp+1 與 cpp 之間的地址差多少。但是考慮到系統(tǒng)對存儲器的對齊要求,不同的系統(tǒng)對齊的方式可能不一樣,所以使用各個成員大小相加的方式計算結(jié)構(gòu)的存儲大小是不合適的。
五、訪問結(jié)構(gòu)的成員
這個比較簡單,注意結(jié)構(gòu)和指向機構(gòu)的指針訪問成員的方式不一樣,結(jié)構(gòu)本身使用 .運算符訪問,而指向結(jié)構(gòu)的指針則使用 -> 訪問。