詳解C++中的指針結(jié)構(gòu)體數(shù)組以及指向結(jié)構(gòu)體變量的指針
C++結(jié)構(gòu)體數(shù)組
一個(gè)結(jié)構(gòu)體變量中可以存放一組數(shù)據(jù)(如一個(gè)學(xué)生的學(xué)號(hào)、姓名、成績等數(shù)據(jù))。如果有10個(gè)學(xué)生的數(shù)據(jù)需要參加運(yùn)算,顯然應(yīng)該用數(shù)組,這就是結(jié)構(gòu)體數(shù)組。結(jié)構(gòu)體數(shù)組與以前介紹過的數(shù)值型數(shù)組的不同之處在于:每個(gè)數(shù)組元素都是一個(gè)結(jié)構(gòu)體類型的數(shù)據(jù),它們都分別包括各個(gè)成員項(xiàng)。
定義結(jié)構(gòu)體數(shù)組和定義結(jié)構(gòu)體變量的方法相仿,定義結(jié)構(gòu)體數(shù)組時(shí)只需聲明其為數(shù)組即可。如:
struct Student //聲明結(jié)構(gòu)體類型Student { int num; char name[20]; char sex; int age; float score; char addr[30]; }; Student stu[3]; //定義Student類型的數(shù)組stu
也可以直接定義一個(gè)結(jié)構(gòu)體數(shù)組,如:
struct Student { int num; char name[20]; char sex; int age; float score; char addr[30]; }stu[3];
或
struct { int num; char name[20]; char sex; int age; float score; char addr[30]; }stu[3];
結(jié)構(gòu)體數(shù)組的初始化與其他類型的數(shù)組一樣,對(duì)結(jié)構(gòu)體數(shù)組可以初始化。如:
struct Student { int num; char name[20]; char sex; int age; float score; char addr[30]; }stu[3]={ {10101,″Li Lin″, ′M′, 18,87.5, ″103 Beijing Road″}, {10102,″Zhang Fun″,′M′,19,99, ″130 Shanghai Road″}, {10104,″Wang Min″,′F′, 20,78.5, ″1010 Zhongshan Road″} };
定義數(shù)組stu時(shí),也可以不指定元素個(gè)數(shù),即寫成以下形式:
stu[ ]={{…},{…},{…}};
編譯時(shí),系統(tǒng)會(huì)根據(jù)給出初值的結(jié)構(gòu)體常量的個(gè)數(shù)來確定數(shù)組元素的個(gè)數(shù)。一個(gè)結(jié)構(gòu)體常量應(yīng)包括結(jié)構(gòu)體中全部成員的值。
當(dāng)然,數(shù)組的初始化也可以用以下形式:
Student stu[ ]={{…},{…},{…}}; //已事先聲明了結(jié)構(gòu)體類型Student
由上可以看到,結(jié)構(gòu)體數(shù)組初始化的一般形式是在所定義的數(shù)組名的后面加上 ={初值表列};
結(jié)構(gòu)體數(shù)組應(yīng)用舉例
下面舉一個(gè)簡單的例子來說明結(jié)構(gòu)體數(shù)組的定義和引用。
【例】對(duì)候選人得票的統(tǒng)計(jì)程序。設(shè)有3個(gè)候選人,最終只能有1人當(dāng)選為領(lǐng)導(dǎo)。今有10個(gè)人參加投票,從鍵盤先后輸入這10個(gè)人所投的候選人的名字,要求最后輸出這3個(gè)候選人的得票結(jié)果。
可以定義一個(gè)候選人結(jié)構(gòu)體數(shù)組,包括3個(gè)元素,在每個(gè)元素中存放有關(guān)的數(shù)據(jù)。程序如下:
#include <iostream> using namespace std; struct Person //聲明結(jié)構(gòu)體類型Person { char name[20]; int count; }; int main( ) { //定義Person類型的數(shù)組,內(nèi)容為3個(gè)候選人的姓名和當(dāng)前的得票數(shù) Person leader[3]={"Li",0,"Zhang",0,"Fun",0}; int i,j; char leader_name[20]; //leader_name為投票人所選的人的姓名 for(i=0;i<10;i++) { cin>>leader_name; //先后輸入10張票上所寫的姓名 for(j=0;j<3;j++) //將票上姓名與3個(gè)候選人的姓名比較 //如果與某一候選人的姓名相同,就給他加一票 if(strcmp(leader_name,leader[j].name)==0) leader[j].count++; } cout<<endl; for(i=0;i<3;i++) //輸出3個(gè)候選人的姓名與最后得票數(shù) { cout<<leader[i].name<<":"<<leader[i].count<<endl; } return 0; }
運(yùn)行情況如下:
Zhang↙ (每次輸入一個(gè)候選人的姓名) Li↙ Fun↙ Li↙ Zhang↙ Li↙ Zhang↙ Li↙ Fun↙ Wang↙ Li:4 (輸出3個(gè)候選人的姓名與最后得票數(shù)) Zhang:3 Fun:2
程序定義一個(gè)全局的結(jié)構(gòu)體數(shù)組leader,它有3個(gè)元素,每一元素包含兩個(gè)成員,即name(姓名)和count(得票數(shù))。在定義數(shù)組時(shí)使之初始化,使3位候選人的票數(shù)都先置零。
在這個(gè)例子中,也可以不用字符數(shù)組而用string方法的字符串變量來存放姓名數(shù)據(jù),程序可修改如下:
#include <iostream> #include <string> using namespace std; struct Person { string name;//成員name為字符串變量 int count; }; int main( ) { Person leader[3]={"Li",0,"Zhang",0,"Fun",0}; int i,j; string leader_name;// leader_name為字符串變量 for(i=0;i<10;i++) { cin>>leader_name; for(j=0;j<3;j++) if(leader_name==leader[j].name) leader[j].count++//用“==”進(jìn)行比較 } cout<<endl; for(i=0;i<3;i++) { cout<<leader[i].name<<":"<<leader[i].count<<endl; } return 0; }
運(yùn)行情況與前相同。顯然后一個(gè)程序節(jié)省內(nèi)存空間,使用更方便,易讀性更好。但是 有些C++系統(tǒng)不能對(duì)包含string成員的結(jié)構(gòu)體變量初始化,需要作一些修改才能運(yùn)行, 讀者可上機(jī)試一下。
C++指向結(jié)構(gòu)體變量的指針
一個(gè)結(jié)構(gòu)體變量的指針就是該變量所占據(jù)的內(nèi)存段的起始地址??梢栽O(shè)一個(gè)指針變量,用來指向一個(gè)結(jié)構(gòu)體變量,此時(shí)該指針變量的值是結(jié)構(gòu)體變量的起始地址。指針變量也可以用來指向結(jié)構(gòu)體數(shù)組中的元素。
通過指向結(jié)構(gòu)體變量的指針引用結(jié)構(gòu)體變量中的成員
下面通過一個(gè)簡單例子來說明指向結(jié)構(gòu)體變量的指針變量的應(yīng)用。
【例】指向結(jié)構(gòu)體變量的指針的應(yīng)用。
#include <iostream> #include <string> using namespace std; int main( ) { struct Student//聲明結(jié)構(gòu)體類型student { int num; string name; char sex; float score; }; Student stu;//定義Student類型的變量stu Student *p=&stu;//定義p為指向Student類型數(shù)據(jù)的指針變量并指向stu stu.num=10301;//對(duì)stu中的成員賦值 stu.name="Wang Fun";//對(duì)string變量可以直接賦值 stu.sex='f'; stu.score=89.5; cout<<stu. num<<" "<<stu.name<<" "<<stu.sex<<" "<< stu.score<<endl; cout<<p -> num<<" "<<(*p).name<<" "<<(*p).sex<<" "<<(*p).score<<endl; return 0; }
程序運(yùn)行結(jié)果如下:
10301 Wang Fun f 89.5 (通過結(jié)構(gòu)體變量名引用成員)
10301 Wang Fun f 89.5 (通過指針引用結(jié)構(gòu)體變量中的成員)
兩個(gè)cout語句輸出的結(jié)果是相同的。
為了使用方便和使之直觀,C++提供了指向結(jié)構(gòu)體變量的運(yùn)算符->,例如p->num表示指針p當(dāng)前指向的結(jié)構(gòu)體變量中的成員num。
p->num 和(*p).num等價(jià)。
同樣
p->name等價(jià)于(*p).name。
也就是說,以下3種形式等價(jià):
結(jié)構(gòu)體變量.成員名。如stu.num。
(*p).成員名。如(*p).num。
p->成員名。如p->num。
“->”稱為指向運(yùn)算符。
請(qǐng)分析以下幾種運(yùn)算:
- p->n 得到p指向的結(jié)構(gòu)體變量中的成員n的值。
- p->n++ 得到p指向的結(jié)構(gòu)體變量中的成員n的值,用完該值后使它加1。
- ++p->n 得到p指向的結(jié)構(gòu)體變量中的成員n的值,并使之加1,然后再使用它。
用結(jié)構(gòu)體變量和指向結(jié)構(gòu)體變量的指針構(gòu)成鏈表
鏈表是一種常見的重要的數(shù)據(jù)結(jié)構(gòu)。下圖表示最簡單的一種鏈表(單向鏈表)的結(jié)構(gòu)。
鏈表有一個(gè)“頭指針”變量,圖中以head表示,它存放一個(gè)地址。該地址指向一個(gè)元素。鏈表中的每一個(gè)元素稱為“結(jié)點(diǎn)”,每個(gè)結(jié)點(diǎn)都應(yīng)包括兩個(gè)部分:
一是用戶需要用的實(shí)際數(shù)據(jù),
二是下一個(gè)結(jié)點(diǎn)的地址。
可以看到鏈表中各元素在內(nèi)存中的存儲(chǔ)單元可以是不連續(xù)的。要找某一元素,可以先找到上一個(gè)元素,根據(jù)它提供的下一元素地址找到下一個(gè)元素。
可以看到,這種鏈表的數(shù)據(jù)結(jié)構(gòu),必須利用結(jié)構(gòu)體變量和指針才能實(shí)現(xiàn)。
可以聲明一個(gè)結(jié)構(gòu)體類型,包含兩種成員,一種是用戶需要用的實(shí)際數(shù)據(jù),另一種是用來存放下一結(jié)點(diǎn)地址的指針變量。
例如,可以設(shè)計(jì)這樣一個(gè)結(jié)構(gòu)體類型:
struct Student { int num; float score; Student *next; //next指向Student結(jié)構(gòu)體變量 };
其中成員num和score是用戶需要用到的數(shù)據(jù),相當(dāng)于圖7.8結(jié)點(diǎn)中的A, B, C, D。next是指針類型的成員,它指向Student類型數(shù)據(jù)(就是next所在的結(jié)構(gòu)體類型)。用這種方法就可以建立鏈表。見圖。
圖中每一個(gè)結(jié)點(diǎn)都屬于Student類型,在它的成員next中存放下一個(gè)結(jié)點(diǎn)的地址,程序設(shè)計(jì)者不必知道各結(jié)點(diǎn)的具體地址,只要保證能將下一個(gè)結(jié)點(diǎn)的地址放到前一結(jié)點(diǎn)的成員next中即可。
下面通過一個(gè)例子來說明如何建立和輸出一個(gè)簡單鏈表。
【例】建立一個(gè)如圖所示的簡單鏈表,它由3個(gè)學(xué)生數(shù)據(jù)的結(jié)點(diǎn)組成。輸出各結(jié)點(diǎn)中的數(shù)據(jù)。
#define NULL 0 #include <iostream> using namespace std; struct Student { long num; float score; struct Student *next; }; int main( ) { Student a,b,c,*head,*p; a. num=31001; a.score=89.5; //對(duì)結(jié)點(diǎn)a的num和score成員賦值 b. num=31003; b.score=90; //對(duì)結(jié)點(diǎn)b的num和score成員賦值 c. num=31007; c.score=85; //對(duì)結(jié)點(diǎn)c的num和score成員賦值 head=&a; //將結(jié)點(diǎn)a的起始地址賦給頭指針head a.next=&b; //將結(jié)點(diǎn)b的起始地址賦給a結(jié)點(diǎn)的next成員 b.next=&c; //將結(jié)點(diǎn)c的起始地址賦給b結(jié)點(diǎn)的next成員 c.next=NULL; //結(jié)點(diǎn)的next成員不存放其他結(jié)點(diǎn)地址 p=head; //使p指針指向a結(jié)點(diǎn) do { cout<<p->num<<" "<<p->score<<endl; //輸出p指向的結(jié)點(diǎn)的數(shù)據(jù) p=p->next; //使p指向下一個(gè)結(jié)點(diǎn) } while (p!=NULL); //輸出完c結(jié)點(diǎn)后p的值為NULL return 0; }
本例是比較簡單的,所有結(jié)點(diǎn)(結(jié)構(gòu)體變量)都是在程序中定義的,不是臨時(shí)開辟的,也不能用完后釋放,這種鏈表稱為靜態(tài)鏈表。對(duì)各結(jié)點(diǎn)既可以通過上一個(gè)結(jié)點(diǎn)的next指針去訪問,也可以直接通過結(jié)構(gòu)體變量名a, b, c去訪問。
動(dòng)態(tài)鏈表則是指各結(jié)點(diǎn)是可以隨時(shí)插入和刪除的,這些結(jié)點(diǎn)并沒有變量名,只能先找到上一個(gè)結(jié)點(diǎn),才能根據(jù)它提供的下一結(jié)點(diǎn)的地址找到下一個(gè)結(jié)點(diǎn)。只有提供第一個(gè)結(jié)點(diǎn)的地址,即頭指針head,才能訪問整個(gè)鏈表。如同一條鐵鏈一樣,一環(huán)扣一環(huán),中間是不能斷開的。
建立動(dòng)態(tài)鏈表,要用到后面介紹的動(dòng)態(tài)分配內(nèi)存的運(yùn)算符new和動(dòng)態(tài)撤銷內(nèi)存的運(yùn)算符delete。
相關(guān)文章
C++指針數(shù)組、數(shù)組指針、數(shù)組名及二維數(shù)組技巧匯總
這篇文章主要介紹了C++指針數(shù)組、數(shù)組指針、數(shù)組名及二維數(shù)組技巧匯總,對(duì)于深入理解C++數(shù)組與指針來說非常重要,需要的朋友可以參考下2014-08-08C++小利器之std::bind參數(shù)綁定包裝器的使用詳解
從 C++11 開始,標(biāo)準(zhǔn)庫提供了 std::bind 用于綁定函數(shù) f 和調(diào)用參數(shù),返回一個(gè)新可調(diào)用函數(shù)對(duì)象 fn,下面就跟隨小編一起深入了解一下std::bind的具體使用吧2023-12-12C++實(shí)現(xiàn)讀取特定路徑下文件夾及文件名的方法
這篇文章主要介紹了C++實(shí)現(xiàn)讀取特定路徑下文件夾及文件名的方法,需要的朋友可以參考下2014-07-07c與c++之間的相互調(diào)用及函數(shù)區(qū)別示例詳解
這篇文章主要為大家介紹了c與c++相互調(diào)用的使用示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-06-06