欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C語言中結(jié)構(gòu)體struct編寫的一些要點解析

 更新時間:2016年04月20日 10:56:05   作者:whuslei  
這篇文章主要介紹了C語言中結(jié)構(gòu)體struct編寫的一些要點解析,談到了結(jié)構(gòu)體的聲明和指針指向等重要知識點,需要的朋友可以參考下

一、關(guān)于結(jié)構(gòu)體的聲明
1、匿名聲明。如:

struct {
  int i,j;
}point;

說明:
這段代碼的含義是,聲明一個無名(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)體變量,類型是不同的!如

struct {
  int i,j;
}p1,p2;

struct {
  int i,j;
}p3;

如果將 p1=p2 ,則ok;如果將 p1=p3 ,則編譯器提示"incompatible types when assigning to type ‘struct <anonymous>' from type ‘struct <anonymous>'",兩者的實際類型是不一樣的。

2、顯式聲明一個結(jié)構(gòu)體

struct node{
  int i,j;
};

聲明了一個結(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中的空指針,當"n1 = new Node();"時,n1才指向了該對象的內(nèi)存空間。因此,在Java中,可以通過"n1==null"來判斷對象是否為空;在C中,不能通過"n1==NULL"來判斷,因為"n1"并不是一個指針,而是一個類型變量的名字,就像"int a;"這種,顯然"a"不是指針!

3、用typedef來簡化結(jié)構(gòu)體的寫法

typedefstruct {
  int i,j;
}Node;

相當于把代碼改名為Node了。以前需要這樣聲明"struct node n1;",現(xiàn)在只需要"Node n1;"。
這段代碼中,如果沒有typedef,代碼的意思是"聲明了一個匿名結(jié)構(gòu)體變量"!注意區(qū)別。
4、在結(jié)構(gòu)體中聲明結(jié)構(gòu)體變量。

typedef struct {
  int i,j;
  Node n1;
}Node;

這段代碼是錯誤的!
錯誤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是什么!所以,無法通過編譯!
正確的使用方法如下:

typedef struct node{
  int i,j;
  struct node *n1;
}Node;

二、關(guān)于結(jié)構(gòu)體的賦值
1、聲明一個變量后的默認值

typedef struct {
  char *p;
  int i;
  char ch[256];
}mystr;
mystr str;//聲明一個變量,此時已為之分配了空間!

如前面提到的,如果這個變量聲明是在全局,則"str.p等于NULL,str.i等于0,str.ch數(shù)組都是'\0'",為默認初始值;如果不在全局,則所有值都是"野值"。

2、手動初始化

mystr str2={"abc",2,"def"};
mystr str3={.p="abc",.ch="def"};
mystr str4={.ch[256]="def"};//error!
mystr str5={.ch[10]="def"};//right!

此時,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、賦值

mystr str6;
str6.p = "abc";

或者

mystr * pstr = & str6;//得到這個結(jié)構(gòu)體變量的指針
pstr->p = "abc";

4、動態(tài)生成結(jié)構(gòu)體變量

mystr * pstr = (mystr*)malloc(sizeof(mystr));
pstr->p = "abc";

注意,如果是動態(tài)生成的結(jié)構(gòu)體變量(用到了malloc),則必須在丟棄該變量前將他的內(nèi)存空間釋放掉(用free)。
如果結(jié)構(gòu)體內(nèi)部也存在動態(tài)生成的對象,在釋放結(jié)構(gòu)體之前要先釋放掉其內(nèi)部的內(nèi)存空間,如下

pstr->p = (char*)malloc(sizeof(char)*256);
free(pstr->p);
free(pstr);

三、結(jié)構(gòu)體數(shù)組

我們知道基本數(shù)據(jù)類型的變量數(shù)組直接定義就可以分配空間了,結(jié)構(gòu)體可以看作一種新類型,它也是定義聲明變量之后就會自動分配空間的,結(jié)構(gòu)體的數(shù)組也是這樣。

struct book library[10];

這樣就定義了一個有10個book變量的數(shù)組,并且已經(jīng)分配了存儲空間。 結(jié)構(gòu)體的數(shù)組和普通數(shù)組索引方式是一樣的。

結(jié)構(gòu)體數(shù)組也可以使用字面量初始化方法,如下

struct book lib[2] = {
 {"apue", "stevens", 128.0},
 {"cpp", "prata", 60.0}
};

是不是很方便了。要注意最外面的是 { ,不是 [ 哦。 對于成員是一個結(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ū)別:

struct book *cpp;
struct book c = {
 "c primer plus",
 "prata",
 60.1
};

cpp = &c;

假設(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)的指針則使用 -> 訪問。

strcut book cpp, *pcpp;
...
char *title;
title = cpp.title;
// title = pcpp->title;
// title = (*pcpp).title; // 因為 . 的優(yōu)先級比 * 高,必須要有括號

相關(guān)文章

  • C++結(jié)構(gòu)體中變長數(shù)組的使用問題分解刨析

    C++結(jié)構(gòu)體中變長數(shù)組的使用問題分解刨析

    變長數(shù)組在C++中指的是集合(也叫容器)如vector就是C語言中,所有的數(shù)組都不定長,沒有下標越界的概念,數(shù)組實質(zhì)就是一個指針(由數(shù)組名充當)因此C語言中數(shù)組的長度沒有任何意義平常在C語言中講的不定長數(shù)組,其實就是指針
    2022-08-08
  • C語言詳細講解#error與#line如何使用

    C語言詳細講解#error與#line如何使用

    這篇文章主要介紹了C語言中#error與#line如何使用,#error與#line雖然在語言里面用的比較少,但是還是有必要了解一下
    2022-04-04
  • C語言運算符的優(yōu)先級和結(jié)合性實例詳解

    C語言運算符的優(yōu)先級和結(jié)合性實例詳解

    本文主要介紹C語言運算符的知識,這里對運算符的優(yōu)先級和結(jié)合性做出了詳解,并附實例代碼,希望能幫助有需要的小伙伴
    2016-07-07
  • C++基本用法實踐之移動語義詳解

    C++基本用法實踐之移動語義詳解

    移動(move)語義是C++引入了一種新的內(nèi)存優(yōu)化,以避免不必要的拷貝,下面小編就來和大家簡單聊聊C++中移動語義的相關(guān)使用吧,希望對大家有所幫助
    2023-07-07
  • C 語言基礎(chǔ)教程(我的C之旅開始了)[四]

    C 語言基礎(chǔ)教程(我的C之旅開始了)[四]

    C 語言基礎(chǔ)教程(我的C之旅開始了)[四]...
    2007-02-02
  • C++類型轉(zhuǎn)換詳解

    C++類型轉(zhuǎn)換詳解

    類型轉(zhuǎn)換有c風(fēng)格的,當然還有c++風(fēng)格的。c風(fēng)格的轉(zhuǎn)換的格式很簡單(TYPE)EXPRESSION,但是c風(fēng)格的類型轉(zhuǎn)換有不少的缺點,有的時候用c風(fēng)格的轉(zhuǎn)換是不合適的,因為它可以在任意類型之間轉(zhuǎn)換
    2021-10-10
  • C++實現(xiàn)簡易選課系統(tǒng)代碼分享

    C++實現(xiàn)簡易選課系統(tǒng)代碼分享

    這篇文章主要介紹了C++實現(xiàn)簡易選課系統(tǒng)及實現(xiàn)代碼的分享,具有一定的參考價值,需要的小伙伴可以參考一下,希望對你有所幫助
    2022-01-01
  • C/C++中可變參數(shù)的用法詳細解析

    C/C++中可變參數(shù)的用法詳細解析

    可變參數(shù)的使用方法遠遠不止以下介紹的幾種,不過在C,C++中使用可變參數(shù)時要小心,在使用printf()等函數(shù)時傳入的參數(shù)個數(shù)一定不能比前面的格式化字符串中的’%’符號個數(shù)少,否則會產(chǎn)生訪問越界,運氣不好的話還會導(dǎo)致程序崩潰
    2013-09-09
  • 如何配置?Sublime?Text4為?C++?編輯器

    如何配置?Sublime?Text4為?C++?編輯器

    這篇文章主要介紹了配置Sublime Text4為C++編輯器,本文給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2023-06-06
  • C語言詳解分析進程控制中進程終止的實現(xiàn)

    C語言詳解分析進程控制中進程終止的實現(xiàn)

    當進程完成執(zhí)行最后語句并且通過系統(tǒng)調(diào)用 exit() 請求操作系統(tǒng)刪除自身時,進程終止。這時,進程可以返回狀態(tài)值(通常為整數(shù))到父進程(通過系統(tǒng)調(diào)用 wait())。所有進程資源,如物理和虛擬內(nèi)存、打開文件和 I/O 緩沖區(qū)等,會由操作系統(tǒng)釋放
    2022-08-08

最新評論