西安電子科技大學(xué)數(shù)據(jù)結(jié)構(gòu)課件2 Linear lists_第1頁
西安電子科技大學(xué)數(shù)據(jù)結(jié)構(gòu)課件2 Linear lists_第2頁
西安電子科技大學(xué)數(shù)據(jù)結(jié)構(gòu)課件2 Linear lists_第3頁
西安電子科技大學(xué)數(shù)據(jù)結(jié)構(gòu)課件2 Linear lists_第4頁
西安電子科技大學(xué)數(shù)據(jù)結(jié)構(gòu)課件2 Linear lists_第5頁
已閱讀5頁,還剩90頁未讀 繼續(xù)免費(fèi)閱讀

下載本文檔

版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進(jìn)行舉報或認(rèn)領(lǐng)

文檔簡介

第二章線性表本章內(nèi)容2.1線性表的類型定義2.2線性表的順序表示和實現(xiàn)2.3線性表的鏈?zhǔn)奖硎竞蛯崿F(xiàn)2.3.1線性鏈表2.3.2循環(huán)鏈表2.3.3雙向鏈表2.4一元多項式的表示及相加12.1線性表的類型定義定義:一個線性表是有n個數(shù)據(jù)元素的有限序列:(a1,a2,…,ai,…,an)。線性表中元素之間的關(guān)系是線性關(guān)系:存在惟一的第一個元素;存在惟一的最后一個元素;除第一個元素之外,每個元素均只有一個直接前驅(qū);除最后一個元素之外,每個元素均只有一個直接后繼。線性表中的元素類型:原子類型:如整數(shù)、字符等。結(jié)構(gòu)類型:如表示一個學(xué)生信息的數(shù)據(jù)元素,包含學(xué)號、姓名、性別等數(shù)據(jù)項。一個線性表中元素的個數(shù)n(n≥0)定義為線性表的長度,n=0時稱為空表。非空線性表中的每個元素都有一個確定的位序。22.1線性表的類型定義線性表的抽象數(shù)據(jù)類型定義:ADTList{

數(shù)據(jù)對象:D={ai

|ai

ElemSet,i=1,2,…n,n0}

數(shù)據(jù)關(guān)系:R={<ai-1,ai>|ai-1,aiD,i=1,2,…n}

基本運(yùn)算:InitList(&L);DestroyList(&L);Length(L);GetElem(L,i,&e);LocateElem(L,e,compare());InsertElem(&L,i,e);DeleteElem(&L,i,&e);……}ADTList32.1線性表的類型定義抽象運(yùn)算(算法2.1)

例2-1利用兩個線性表LA和LB分別表示集合A和B,求一個新的集合A=A∪B。

算法思路:集合中的元素間是松散的關(guān)系,只需將在線性表LB中而不在LA中的數(shù)據(jù)元素追加到LA的尾部即可。voidunion(List&La,ListLb){

La_len=Length(La);Lb_len=Length(Lb);

for(i=1;i<=Lb_len;i++){

GetElem(Lb,i,e);if(!LocatElem(La,e,equal))

InsertElem(La,++La_len,e);}}//union4特點:存儲單元地址連續(xù)(需要一段連續(xù)空間)邏輯上相鄰的數(shù)據(jù)元素其物理位置也相鄰存儲密度大(100%)隨機(jī)存取元素序號與存儲位置存在如下關(guān)系:2.2線性表的順序表示和實現(xiàn)順序表---線性表的順序存儲 內(nèi)涵:

線性表的順序存儲指用一組地址連續(xù)的存儲單元依次存儲線性表的數(shù)據(jù)元素。這稱為順序表。a1a2...aian...}d

Loc(ai)=Loc(a1)+(i-1)*d(1≤i≤n)52.2線性表的順序表示和實現(xiàn)順序表存儲的C語言實現(xiàn) 由于C語言中的一維數(shù)組也是采用順序存儲表示,故可以用數(shù)組類型來描述順序表。下面我們用結(jié)構(gòu)類型來定義順序表類型。 //----------動態(tài)分配------------#defineLIST_INIT_SIZE100 //空間初始分配量#defineLISTINCREMENT10 //空間分配增量typedefstruct{

ElemType*elem; //存儲空間基址

intlength; //當(dāng)前長度

intlistsize; //當(dāng)前分配的存儲容量}SqList;62.2線性表的順序表示和實現(xiàn)創(chuàng)建一個空的順序表(算法2.3) StatusInitList_sq(SqList&L){

L.elem=(ElemType*)malloc(INIT_SIZE*sizeof(ElemType)); if(!L.elem)exit(OVERFLOW);

L.length=0;

L.listsize=INIT_SIZE; returnOK;}//InitList_sq72.2線性表的順序表示和實現(xiàn)線性表上的基本運(yùn)算插入運(yùn)算 含義: 將元素e插入到線性表:(a1,a2,…,ai-1,ai,…,an)中,構(gòu)成新的線性表(a1,a2,…,ai-1,e,ai,…,an),滿足ai-1≤e≤ai,(其中≤為比較關(guān)系),即不破壞原線性關(guān)系。表的長度為n+1將元素e插入到元素ai-1之后,ai-1的直接后繼和ai

的直接前驅(qū)就改變了,需要在順序表的存儲結(jié)構(gòu)上反映出來。82.2線性表的順序表示和實現(xiàn)順序表上插入運(yùn)算的實現(xiàn):a1a2......ai-1anai將元素ai~an(n-i+1個元素)依次向后移動一個位置,騰出要放置e的單元。a1a2......ai-1anaie將元素e放在騰出的第i個空位上。a1a2...aian...ai-1e在線性表L的第i個位置上插入元素e。e92.2線性表的順序表示和實現(xiàn)以7個元素的順序表為例進(jìn)行說明,將元素e插入a5之前。a1a2a5a7a4a3a6KK+1K+4K+6K+3K+2K+5K+7e(1)元素a7向后移動一個位置;(2)元素a6向后移動一個位置;a1a2a5a4a3a6a7KK+1K+4K+6K+3K+2K+5K+7a1a2a5a6a4a3a7KK+1K+4K+6K+3K+2K+5K+7(3)元素a5向后移動一個位置;a1a2a6a4a3a5a7KK+1K+4K+6K+3K+2K+5K+7(4)將元素e插入到空位上。

ee102.2線性表的順序表示和實現(xiàn)用類C語言實現(xiàn)順序表的插入StatusListInsert_sq(SqList&L,inti,ElemTypex){//將元素x插入在線性表L的第i個元素位置上if(i<1||i>L.length+1)returnERROR;if(L.length>=L.Listsize){

newbase=(ElemType*)realloc(L.elem, (L.listsize+LISTINCREMENT)*sizeof(ElemType));if(!newbase)exit(OVERFLOW);

L.elem=newbase;

L.listsize+=INCREMENT;}q=&(L.elem[i-1]);//q為插入位置,C語言數(shù)組下標(biāo)從0開始

for(p=&(L.elem[L.length-1]);p>=q;--p)*(p+1)=*p;*q=x;L.length++; returnOK;}//ListInsert_sq①判定輸入是否合法②判定空間是否夠用③修改順序表的參數(shù)④移動元素并插入,可改用數(shù)組下標(biāo)實現(xiàn)112.2線性表的順序表示和實現(xiàn)順序表上插入運(yùn)算效率分析:

時間復(fù)雜度 從算法流程上看,查找插入位置、插入元素的時間消耗是常數(shù)級,算法執(zhí)行時間主要消耗在插入前元素的移動上,而移動元素的個數(shù)與插入位置i有關(guān)。

最好情況:在表尾插入,不移動元素,T(n)=O(1)最壞情況:在表頭插入,移動n個元素,T(n)=O(n)平均復(fù)雜度 設(shè)pi為在第i個元素之前插入一個元素的概率,且P1=P2=…=Pi=…=Pn=1/(n+1),則平均移動次數(shù)為: 即T(n)=O(n)

??臻g復(fù)雜度不需要額外空間,S(n)=O(1)。122.2線性表的順序表示和實現(xiàn)線性表上的基本運(yùn)算刪除 含義: 將元素ai從線性表:(a1,a2,…,ai-1,ai,ai+1,…,an)中刪除,構(gòu)成新的線性表(a1,a2,…,ai-1,ai+1,…,an),刪除后不破壞原線性關(guān)系。表的長度為n-1將元素ai刪除之后,ai-1的直接后繼和ai+1

的直接前驅(qū)就改變了,需要在順序表的存儲結(jié)構(gòu)上反映出來。132.2線性表的順序表示和實現(xiàn)順序表上刪除運(yùn)算的實現(xiàn):a1a2...ai-1an...aiai+1a1a2...ai-1an...ai+1anStatusListDelete_Sq(SqList&L,inti,ElemType&e){第一步:判斷參數(shù)是否合法合理,否則出錯;第二步:在物理空間中找到刪除位置;第三步:刪除(將后續(xù)元素位置依次前移);第四步:刪除后的善后工作(修改表長)}//ListDelete_Sq請同學(xué)們課后自學(xué)順序表的刪除算法及復(fù)雜度分析。14內(nèi)容回顧線性表的邏輯定義元素的一個有限序列元素之間滿足線性關(guān)系線性表的抽象數(shù)據(jù)類型定義ADTList{

數(shù)據(jù)對象:D={ai

|ai

ElemSet,i=1,2,…n,n0}

數(shù)據(jù)關(guān)系:R={<ai-1,ai>|ai-1,aiD,i=1,2,…n}

基本運(yùn)算:……}ADTList線性表的順序存儲--順序表a1…a1a2...aian...…a2aian15內(nèi)容回顧順序表的優(yōu)缺點優(yōu)點:不需要額外的存儲空間來表示元素間的邏輯關(guān)系可以隨機(jī)地存取表中的任意一個元素缺點:插入和刪除元素時要移動大量的元素必須事先進(jìn)行空間分配,表的容量難以擴(kuò)充(新算法已解決)順序表的基本運(yùn)算的復(fù)雜度插入T(n)=O(n),S(n)=O(1)刪除T(n)=O(n),S(n)=O(1)16內(nèi)容回顧用類C語言實現(xiàn)順序表的插入StatusListInsert_sq(Sqlist&L,inti,ElemTypex){if(i<1||i>L.length+1)returnERROR;if(L.length>=L.Listsize){

newbase=(ElemType*)realloc(L.elem, (L.listsize+LISTINCREMENT)*sizeof(ElemType));if(!newbase)exit(OVERFLOW);

L.elem=newbase;

L.listsize+=INCREMENT;}q=&(L.elem[i-1]);//q為插入位置,C語言數(shù)組下標(biāo)從0開始for(p=&(L.elem[L.length-1]);p>=q;--p)*(p+1)=*p;*q=x;L.length++; returnOK;}//ListInsert_sq類C語言書寫的算法與C程序之間的差別:(1)算法中除形式參數(shù)外,變量不作定義,在C程序中必須定義;(2)算法中使用的元素類型(ElemType)沒有定義,C程序中必須定義,常量OK、ERROR、OVERFLOW等在第一章統(tǒng)一定義;(3)算法中的比較運(yùn)算符(equal、less)未作定義,C程序中必須定義;(4)必要的頭文件(用作輸入輸出的stdio.h及內(nèi)存申請的stdlib.h),在C程序中必須包含。172.3線性表的鏈?zhǔn)奖硎竞蛯崿F(xiàn)線性表的鏈?zhǔn)酱鎯λ悸穉1a2...aian...存儲結(jié)構(gòu)……a1a2aian邏輯結(jié)構(gòu)實現(xiàn)方式對每一個元素的存儲單位進(jìn)行擴(kuò)充,在包含元素值存儲的同時,在結(jié)點中存儲與當(dāng)前元素有直接關(guān)系的其他元素(直接前趨/直接后繼)存儲單位的地址(指針)。鏈?zhǔn)酱鎯挝唤Y(jié)構(gòu)示意圖:ai

ai

182.3線性表的鏈?zhǔn)奖硎竞蛯崿F(xiàn)鏈表---線性表的鏈?zhǔn)酱鎯?nèi)涵: 線性表的鏈?zhǔn)酱鎯χ赣萌我獾拇鎯卧娣啪€性表中的元素,每個元素與其直接前驅(qū)和(或)直接后繼之間的關(guān)系用指針來存儲。這稱為鏈表。術(shù)語結(jié)點:數(shù)據(jù)元素及與其有直接關(guān)系的元素的地址構(gòu)成的存儲單位。數(shù)據(jù)域指針域datanext(a)鏈表結(jié)點結(jié)構(gòu)示意圖datapriornextdata(b)(c)nextdata(d)prior192.3.1單鏈表的表示和實現(xiàn)單鏈表 鏈表中,如果每個結(jié)點中只包含一個指針域,則稱之為線性鏈表或單鏈表。單鏈表的存儲: 每個結(jié)點存儲當(dāng)前元素的直接后繼。 線性表(a1,a2,…,ai-1,ai,ai+1,…,an)

邏輯結(jié)構(gòu)………………110ai135………………130a2143135ai+1180………………160anNull165a1130170ai-1110………………單鏈表示意圖a1

a2

ai-1

…ai

ai+1

an^…Head單鏈表的內(nèi)存鏡像指向單鏈表中第一個結(jié)點的頭指針可惟一確定一個單鏈表。Head165202.3.1單鏈表的表示和實現(xiàn)單鏈表的C語言實現(xiàn)//----------用結(jié)構(gòu)指針描述------------typedef

struct

LNode{

ElemTypedata; //數(shù)據(jù)域

struct

LNode*next; //指針域}LNode,*LinkList;與之對應(yīng)的結(jié)點結(jié)構(gòu)示意圖單鏈表構(gòu)建示意圖(表頭插入法)Headpan^a1

Headppan-1pHead…ai

…ai

LinkList212.3.1單鏈表的表示和實現(xiàn)帶頭結(jié)點的單鏈表單鏈表示意圖a1

a2

ai-1

…ai

ai+1

an^…Head

頭結(jié)點除指針域外可以存儲鏈表的其他信息,如鏈表的長度、鏈表的說明等信息,也可以不存儲任何信息。帶頭結(jié)點的空鏈表Head

^頭結(jié)點222.3.1單鏈表的表示和實現(xiàn)單鏈表上的查找運(yùn)算在單鏈表中,必須從頭指針出發(fā)進(jìn)行查找:查找第i個元素查找指定的元素是否在表中...若找到,則返回該元素的值,否則返回ERROR。在單鏈表上查找第i個元素的示意圖pk=1單鏈表查找示意圖a1

a2

…ai

an^…Head

232.3.1單鏈表的表示和實現(xiàn)單鏈表上的查找運(yùn)算在單鏈表中,必須從頭指針出發(fā)進(jìn)行查找:查找第i個元素查找指定的元素是否在表中...若找到,則返回該元素的值,否則返回ERROR。在單鏈表上查找第i個元素的示意圖pk=2單鏈表查找示意圖a1

a2

…ai

an^…Head

242.3.1單鏈表的表示和實現(xiàn)單鏈表上的查找運(yùn)算在單鏈表中,必須從頭指針出發(fā)進(jìn)行查找:查找第i個元素查找指定的元素是否在表中...若找到,則返回該元素的值,否則返回ERROR。在單鏈表上查找第i個元素的示意圖pk=i單鏈表查找示意圖a1

a2

…ai

an^…Head

找到,返回P指向的結(jié)點的數(shù)據(jù)。問題:i<0或i>n呢?252.3.1單鏈表的表示和實現(xiàn)用類C語言實現(xiàn)單鏈表上的查詢(查找第i個元素):StatusGetElem_L(LinkListL,inti,ElemType&e){//L是帶頭結(jié)點的單鏈表的頭指針//當(dāng)?shù)趇個元素存在時,將該元素的值賦給e并返回OK,否則返回ERRORp=L->next;k=1;//初始化,p指向第一個結(jié)點,k為計數(shù)器while(p&&k<i){//順指針向后查找,直到p指向第i個元素或p為空

p=p->next;++k;} //循環(huán)的退出條件為k=i或到達(dá)單鏈表結(jié)尾(P為空)if(!p||k>i)returnERROR;//第i個元素不存在e=p->data;returnOK;}//GetElem_L思考:①本算法為何沒有考慮i值的合法性?

②查找指定元素是否在表中如何實現(xiàn)?

③查找算法的復(fù)雜度?a1

a2

…ai

an^…L

262.3.1單鏈表的表示和實現(xiàn)單鏈表上的插入運(yùn)算(第i個位置上插入新的結(jié)點)邏輯運(yùn)算(a1,a2,…,ai-1,ai,ai+1,…,an)(a1,a2,…,ai-1,e,ai,ai+1,…,an)在元素ai之前插入一個新元素e單鏈表上的實現(xiàn)示意圖……a1

ai

an^L

ai-1

ai+1

基本步驟:①找到第i-1個元素所在結(jié)點;②申請一個新結(jié)點s并填入e值;③

修改s的指針域指向p的下一結(jié)點;p①es②③272.3.1單鏈表的表示和實現(xiàn)單鏈表上的插入運(yùn)算(第i個位置上插入新的結(jié)點)邏輯運(yùn)算(a1,a2,…,ai-1,ai,ai+1,…,an)(a1,a2,…,ai-1,e,ai,ai+1,…,an)在元素ai之前插入一個新元素e單鏈表上的實現(xiàn)示意圖基本步驟:①找到第i-1個元素所在結(jié)點;p②申請一個結(jié)點s并填入e值;e③

修改s的指針域指向p的下一結(jié)點;s④修改p的指針域指向s。①②③思考:第③④步是否可交換?……a1

ai

an^L

ai-1

ai+1

④282.3.1單鏈表的表示和實現(xiàn)插入的類C語言實現(xiàn)StatusListInsert_L(LinkList&L,inti,ElemTypee){

//在帶頭結(jié)點的單鏈表L中第i個位置之前插入元素e

p=L;k=0;//初始化,p指向頭結(jié)點,k為計數(shù)器

while(p&&k<i-1){//逐步移動指針p,直到p指向第i-1個元素或p為空

p=p->next;++k;}//①if(!p||k>i-1)returnERROR;//無法插入

if(!(s=(LinkLinst)malloc(sizeof(LNode))))//申請元素e的結(jié)點空間

returnOVERFLOW;

//內(nèi)存無空閑單元,無法申請到空間

s->data=e;

//②

s->next=p->next;

//③插入元素e的結(jié)點

p->next=s; //④returnOK;}//LinstInsert_L復(fù)雜度:

①查找:T1(n)=O(n)

②插入:T2(n)=O(1)

T(n)=O(n)a1

a2

…ai

an^…L

292.3.1單鏈表的表示和實現(xiàn)單鏈表上的刪除運(yùn)算(把第i個位置的結(jié)點刪除)邏輯運(yùn)算(a1,a2,…,ai-1,ai,ai+1,…,an)(a1,a2,…,ai-1,ai+1,…,an)刪除結(jié)點ai單鏈表上的實現(xiàn)示意圖a1

…ai

an^…L

ai-1

ai+1

基本步驟:①找到第i-1個元素所在結(jié)點;p①②增設(shè)臨時指針q指向p的下一結(jié)點(待刪除的結(jié)點);q②302.3.1單鏈表的表示和實現(xiàn)單鏈表上的刪除運(yùn)算(把第i個位置的結(jié)點刪除)邏輯運(yùn)算(a1,a2,…,ai-1,ai,ai+1,…,an)(a1,a2,…,ai-1,ai+1,…,an)刪除結(jié)點ai單鏈表上的實現(xiàn)示意圖基本步驟:①找到第i-1個元素所在結(jié)點;③

修改結(jié)點p的指針域,指向第i-1個結(jié)點(p->next=p->next->next);②增設(shè)臨時指針q指向p的下一結(jié)點(待刪除的結(jié)點);④

刪除結(jié)點q,釋放內(nèi)存空間。a1

…ai

an^…L

ai-1

ai+1

③p①q②312.3.1單鏈表的表示和實現(xiàn)單鏈表上的刪除運(yùn)算(把第i個位置的結(jié)點刪除)邏輯運(yùn)算(a1,a2,…,ai-1,ai,ai+1,…,an)(a1,a2,…,ai-1,ai+1,…,an)刪除結(jié)點ai單鏈表上的實現(xiàn)示意圖基本步驟:①找到第i-1個元素所在結(jié)點;③

修改結(jié)點p的指針域,指向第i-1個結(jié)點(p->next=p->next->next);②增設(shè)臨時指針q指向p的下一結(jié)點(待刪除的結(jié)點);④

刪除結(jié)點q,釋放內(nèi)存空間。p①③④a1

…ai

an^…L

ai-1

ai+1

q②322.3.1單鏈表的表示和實現(xiàn)StatusListDelete_L(LinkList&L,inti,ElemType&e){//在帶頭結(jié)點的單鏈表L中,刪除第i個元素,并由e返回其值

p=L;k=0;//初始化,p指向頭結(jié)點,k為計數(shù)器

while(p->next&&k<i-1){

//逐一移動指針p,直到p指向第i-1個元素或p為空

p=p->next;++k;}if(!p->next||j>i-1)returnERROR;//無法刪除

q=p->next;

//* 令q指向待刪除的元素結(jié)點;

p->next=q->next;//*e=q->data;free(q);returnOK;}//ListDelete_L刪除的類C語言實現(xiàn)思考:①

while語句中的p->next可否用p替代?

②用*注釋的兩條語句可否位置交換?

③本算法的T(n)和S(n)?a1

…ai

an^…L

ai-1

ai+1

pq332.3.1單鏈表的表示和實現(xiàn)鏈表的優(yōu)缺點優(yōu)點:插入、刪除時無須移動元素,只需修改指針根據(jù)需要申請存儲空間,且不要求連續(xù)的存儲空間缺點:對表中的元素只能進(jìn)行順序訪問用指針指示元素之間的邏輯關(guān)系(直接前驅(qū)、后繼),存儲空間利用率低自學(xué)內(nèi)容:單鏈表的表頭插入構(gòu)造算法(算法2.11)34內(nèi)容回顧鏈表--線性表的鏈?zhǔn)酱鎯Α璦1a2aian邏輯結(jié)構(gòu)單鏈表示意圖a1

a2

ai-1

…ai

ai+1

an^…Heada1

a2

…ai

an^…Head

帶頭結(jié)點的單鏈表示意圖35內(nèi)容回顧單鏈表上的查找運(yùn)算(查找第i個元素)a1

a2

…ai

an^…Head

計數(shù)器kpk=2pk=i查找失敗的情形:(1)i<=0:此時k>i(2)i>n:此時p==NULLp(初始化)k=1計數(shù)器k找到,返回P指向的結(jié)點的數(shù)據(jù)。36內(nèi)容回顧單鏈表上的插入運(yùn)算(在第i個元素前插入新的元素)p①es②a1

…ai

an^…L

ai-1

ai+1

③②s=(LinkLinst)malloc(sizeof(LNode))基本操作對應(yīng)語句:

p=L;k=0;

while(p&&k<i-1){p=p->next;++k;}③

s->next=p->next;④

p->next=s;④37內(nèi)容回顧單鏈表上的刪除運(yùn)算(刪除第i個元素)③p①q②基本操作對應(yīng)語句:

p=L;k=0;

while(p&&k<i-1){p=p->next;++k;}②q=p->next;

p->next=q->next;

e=q->data;free(q);

特別注意:修改指針的語句次序一定要小心處理,否則會出現(xiàn)意想不到的錯誤。a1

…ai

an^…L

ai-1

ai+1

ai

④382.3.1單鏈表的表示和實現(xiàn)用表頭插入法建立帶頭結(jié)點的單鏈表基本步驟: (1)建立頭結(jié)點;L=(LinkList)malloc(sizeof(LNode));L->next=NULL;an

…a2

L^a1 (2)建立新結(jié)點p,填入元素;p=(LinkList)malloc(sizeof(LNode));P->data=ai (3)將新結(jié)點p插入到頭結(jié)點之后,修改頭結(jié)點指針域指向p;p->next=L->next;L->next=p;

^ (4)重復(fù)(2)(3)步直到插入所有結(jié)點,結(jié)束。注:表頭插入法建立的鏈表是輸入元素序列的逆序!392.3.1單鏈表的表示和實現(xiàn)用類C語言實現(xiàn)表頭插入算法:an

an-1

…ai

a1^…Head

voidCreateList_L(LinkLinst&L,intn){ //用表頭插入法逆序建立帶頭結(jié)點的單鏈表

L=(LinkList)malloc(sizeof(LNode)); //未做溢出判定,應(yīng)加入 L->next=NULL;//建立頭結(jié)點,假定與元素結(jié)點結(jié)構(gòu)相同

for(i=n;i>0;--i){ //從后向前輸入元素

p=(LinkList)malloc(sizeof(LNode));//生成新結(jié)點

scanf(&p->data);//從鍵盤輸入元素值,存入新結(jié)點中

p->next=L->next;L->next=p;//新結(jié)點插入到表頭 }}算法復(fù)雜度:T(n)=O(n),S(n)=O(1)402.3.1單鏈表的表示和實現(xiàn)用C語言實現(xiàn)表頭插入算法:

LinkList

CreateList_LH(intn){

inti;//聲明本函數(shù)的內(nèi)部變量類型

LinkListhead;//head為指向鏈表的指針類型

LNode*p; //p為指向結(jié)點的指針類型 head=NULL;for(i=n;i>0;--i){//從后向前輸入元素p=(LNode*)malloc(sizeof(LNode));//未做溢出判定,應(yīng)加

scanf(“%d”,&p–>data);//將鍵盤輸入置入p的數(shù)據(jù)域p–>next=head;head=p;}return(head);//返回指向鏈表的頭指針}//-------用結(jié)構(gòu)指針描述---------typedef

struct

LNode{

ElemTypedata; //數(shù)據(jù)域

struct

LNode*next;//指針域}LNode,*LinkList;412.3.1單鏈表的表示和實現(xiàn)用表尾插入法建立帶頭結(jié)點的單鏈表基本步驟: (1)建立頭結(jié)點,并附設(shè)一個指針r一直指向最后一個結(jié)點;

L=(LinkList)malloc(sizeof(LNode));L.next=NULL;r=L;L^ (2)申請一個新的結(jié)點空間,存入元素值;

p=(LinkList)malloc(sizeof(LNode)); p->data=ai;p->next=NULL; (3)將新結(jié)點插入鏈表尾部,r指向最后結(jié)點;a1^rrra2^r…an^rr

r->next=p;r=p; (4)重復(fù)(2)(3)直到最后一個結(jié)點,結(jié)束。r作業(yè):用類C語言寫出該算法。是否每次申請結(jié)點都要給其指針域賦空值?422.3.1單鏈表的表示和實現(xiàn)用表尾插入法建立不帶頭結(jié)點的單鏈表基本步驟: (1)初始化頭指針Head和指向最后結(jié)點的指針r;

head=NULL;r=NULL; (2)申請一個新的結(jié)點并令p指向該結(jié)點,存入元素值; (3)將新結(jié)點插入到表尾; r->next=p;r=p;?

①當(dāng)r指向非空結(jié)點時使用上面的語句

②當(dāng)r指向空地址時head=p;r=p; (4)重復(fù)(2)(3)直到最后一個結(jié)點,結(jié)束。L^ra1

Lra2

rr…an

rr^注:表尾插入法建立的鏈表與輸入元素順序一致!432.3.1單鏈表的表示和實現(xiàn)用表尾插入法建立不帶頭結(jié)點的單鏈表(元素為字符,C語言):LinkList

CreateList_LT(

){charch;LinkListhead;

LNode*p,*r;//p指向新申請的結(jié)點,r指向鏈表的尾結(jié)點head=NULL;r=NULL;

while((ch=getchar()!=‘\n’){p=(LNode*)malloc(sizeof(LNode));if(!p)returnNULL;p–>data=ch;

if(head==

NULL) //插入第一個結(jié)點 head=p;else

r–>next=p; //插入第二個及以后的結(jié)點r=p;}if(r!=NULL)r–>next=NULL;return(head);}442.3.1單鏈表的表示和實現(xiàn)說明:

第一個生成的結(jié)點(開始結(jié)點)要插入到鏈表尾部,此時指向最后一個結(jié)點的指針r指向的是空地址,插入時需要修改的是指針變量r,而隨后插入其余結(jié)點時,指向最后一個結(jié)點的指針r指向的是一個存在的結(jié)點,插入時需要修改該結(jié)點的指針域r->next。故在算法中應(yīng)加以區(qū)分。 對于表尾插入法,因為隨后的插入要修改前一結(jié)點的指針域,故可以在申請結(jié)點后不處理指針域,而在結(jié)束前對尾結(jié)點指針域賦空值。452.3.1單鏈表的表示和實現(xiàn)說明:

如果我們在鏈表的開始結(jié)點之前附加一個頭結(jié)點,那么會帶來以下兩個優(yōu)點:a.由于開始結(jié)點的位置被存放在頭結(jié)點的指針域中,所以在鏈表的第一個位置上的操作就和在表的其它位置上的操作一致,無需進(jìn)行特殊處理;b.無論鏈表是否為空,其頭指針是指向頭結(jié)點的非空指針(空表中頭結(jié)點的指針域為空),因此空表和非空表的處理也就統(tǒng)一了。462.3.1單鏈表的表示和實現(xiàn)兩個有序單鏈表的合并(Lc=La∪Lb)La

9

14

15

2740^Lb

7

14

20

^Lc

7

9

14

141520

2740^處理思路:(1)設(shè)置三個指針pa、pb、pc分別指向La、Lb、Lc的當(dāng)前結(jié)點;(2)比較pa、pb所指結(jié)點的元素值大小,將元素值小的結(jié)點鏈接到pc所指結(jié)點之后;(3)重復(fù)(2),直到pa、pb中某一指針為空,將另一非空指針鏈追加到pc之后,算法結(jié)束。472.3.1單鏈表的表示和實現(xiàn)用類C語言實現(xiàn)歸并算法:voidMergeList_L(LinkList&La,LinkList&Lb,LinkList&Lc){//歸并兩個非遞減單鏈表La、Lb到Lc中,且使Lc非遞減

pa=La->next;pb=Lb->next;

Lc=pc=La; while(pa&&pb){ if(pa->data<=Pb->data){ pc->next=pa;pc=pa;pa=pa->next; } else{ pc->next=pb;pc=pb;pb=pb->next; } }//while pc->next=pa?pa:pb; free(Lb);}//MergeList_L算法復(fù)雜度:?T(n)=O(na)+O(nb)=O(nLa+nLb)482.3.2循環(huán)單鏈表和雙向鏈表循環(huán)單鏈表

線性表:(a1,a2,…,ai,…,an)taila2a1anan-1…(a)循環(huán)單鏈表帶尾指針的循環(huán)單鏈表的特點:通過表尾指針可以一步得到表頭指針;對于簡單的兩個表的合并可以通過簡單操作完成(T(n)=O(1));492.3.2循環(huán)單鏈表和雙向鏈表循環(huán)單鏈表的合并tailbb2b1bmbn-1…(b)循環(huán)單鏈表Lbtailaa2a1anan-1…(a)循環(huán)單鏈表La502.3.2循環(huán)單鏈表和雙向鏈表tailbb2b1bmbn-1…(b)循環(huán)單鏈表Lb…tailaa2a1anan-1(a)循環(huán)單鏈表La循環(huán)單鏈表的合并pa①②512.3.2循環(huán)單鏈表和雙向鏈表tailbb2b1bmbn-1…(b)循環(huán)單鏈表Lb…tailaa2a1anan-1(a)循環(huán)單鏈表La循環(huán)單鏈表的合并pa①②③522.3.2循環(huán)單鏈表和雙向鏈表tailbb2b1bmbn-1…(b)循環(huán)單鏈表Lba2a1anan-1…循環(huán)單鏈表的合并②③(a)循環(huán)單鏈表La53內(nèi)容回顧單鏈表的建立方法表頭插入法建立單鏈表建立的鏈表是元素輸入次序的逆序表尾插入法建立單鏈表建立的鏈表與元素輸入次序一致帶頭結(jié)點的單鏈表的優(yōu)點 使得指向鏈表的起始結(jié)點(第一個元素所在的結(jié)點)的指針與指向其它元素結(jié)點的指針形式上相同(都是存儲在前一結(jié)點的指針域里),而可以在算法上一致處理。 與此對應(yīng),在不帶頭結(jié)點的單鏈表上,指向起始結(jié)點的指針是一個指針變量,而指向其他結(jié)點的指針是前一結(jié)點結(jié)構(gòu)的指針域,因此在遍歷、插入、刪除等操作時必須特殊處理起始結(jié)點。循環(huán)單鏈表的優(yōu)點 使得對鏈表的某些操作簡化542.3.2循環(huán)單鏈表和雙向鏈表雙向鏈表的結(jié)點結(jié)構(gòu):Datanextprior雙向鏈表的C語言描述:typedef

struct

DuLNode{

ElemTypedata;

struct

DuLNode*prior;

struct

DuLNode*next;}DuLNode,*DulinkList;雙向鏈表

線性表:(a1,a2,…,ai,…,an)heada1∧a2an-1anai…∧雙向鏈表示意圖…55head∧ai-1ai+1anai…∧…a12.3.2循環(huán)單鏈表和雙向鏈表雙向鏈表上的插入操作(將元素e插入到鏈表的第i個結(jié)點前)基本步驟:(1)定位指針p指向結(jié)點ai-1;pe

s(2)建立新結(jié)點s并賦e;(3)修改s的next指針域指向p下一結(jié)點:s->next=p->next;(5)修改p的next指針域指向s結(jié)點:p->next=s;(6)修改s下一結(jié)點的prior指針域指向s:s->next->prior=s;(4)修改s的prior指針域指向p結(jié)點:s->prior=p;問題:①修改指針的步驟是否可隨意?②不帶頭結(jié)點?562.3.2循環(huán)單鏈表和雙向鏈表雙向鏈表上的刪除操作(刪除第i個結(jié)點)head

∧ai-1ai+1anai…∧…a1∧基本步驟:(1)定位指針p指向結(jié)點ai;p(2)修改p的前一結(jié)點的next指針域指向p下一結(jié)點:p->prior->next=p->next;(3)修改p的下一結(jié)點的prior指針域指向p前一結(jié)點:p->next->prior=p->prior;(3)釋放結(jié)點p。572.3.2循環(huán)單鏈表和雙向鏈表雙向鏈表上的刪除操作(刪除第i個結(jié)點)p基本步驟:(1)定位指針p指向結(jié)點ai;(2)修改p的前一結(jié)點的next指針域指向p下一結(jié)點:p->prior->next=p->next;(3)修改p的下一結(jié)點的prior指針域指向p前一結(jié)點:p->next->prior=p->prior;(4)釋放結(jié)點p。head

∧ai-1ai+1an…∧…a1∧582.3.2循環(huán)單鏈表和雙向鏈表雙向循環(huán)鏈表

head帶頭結(jié)點的空雙向循環(huán)鏈表head

^a1an-1anai…^…帶頭結(jié)點的非空雙向循環(huán)鏈表592.4一元多項式的表示及相加一元多項式的形式:A(x)=p0+p1x+p2x2+…+pixi+…+pnxn存儲形式(a)順序存儲pip0p1...p2pn...pi……Head

(b)單鏈表方式存儲pip0

p1

pi

pn^問題:各存儲方式的優(yōu)缺點和使用場合?

Head(c)單鏈表方式存儲pi和ip1

1p5

5…pi

i…pn^n602.4一元多項式的表示及相加例A(x)=7+3x+9x8+

5x17-8x100B(x)=8x+22x7-9x8C(x)=A(x)+B(x)=7+11x+22x7+5x17-8x100存儲方式的選擇--以單鏈表存儲多項式系數(shù)和指數(shù)+headA703198517-8^100headB81227-9^8//-------多項式的項---------typedef

struct{floatcoef; //系數(shù)

int

expn;//指數(shù)}term,ElemType;//-------多項式結(jié)點---------typedef

struct

LNode{

ElemTypedata;//數(shù)據(jù)域

struct

LNode*next;//指針域}*polynomial;612.4一元多項式的表示及相加一元多項式的相加實現(xiàn)headA703198517-8^100headB81227-9^8headC^qaqbqc//-------多項式結(jié)點---------typedef

struct

LNode{

ElemTypedata;//數(shù)據(jù)域

struct

LNode*next;//指針域}*polynomial;622.4一元多項式的表示及相加一元多項式的相加實現(xiàn)headA703198517-8^100headB81227-9^8headC^qb7^0qcqcqaqa632.4一元多項式的表示及相加一元多項式的相加實現(xiàn)headB81227-9^8qbheadA703198517-8^100qaheadC^7^0qc11^1qaqbqc642.4一元多項式的表示及相加一元多項式的相加實現(xiàn)headA703198517-8^100qaheadB81227-9^8qbheadC^7^011^1qc22^7qbqc652.4一元多項式的表示及相加一元多項式的相加實現(xiàn)headA703198517-8^100qaheadB81227-9^8qbheadC^7^011^122^7qcqaqb662.4一元多項式的表示及相加一元多項式的相加實現(xiàn)headC^7^011^122^7qcheadA703198517-8^100qaheadB81227-9^8qb5^17qaqc672.4一元多項式的表示及相加一元多項式的相加實現(xiàn)headB81227-9^8qbheadA703198517-8^100qaheadC^7^011^122^75^17qc-8^100qaqc682.4一元多項式的表示及相加上機(jī)實現(xiàn)提示(1)編寫單鏈表創(chuàng)建算法(函數(shù));(2)編寫多項式加、減法算法;(3)編寫多項式表達(dá)式的輸出算法;(4)編寫主函數(shù),兩次調(diào)用建表函數(shù)建立單鏈表;調(diào)用加法函數(shù)創(chuàng)建結(jié)果鏈表;調(diào)用減法函數(shù)創(chuàng)建結(jié)果鏈表;調(diào)用輸出函數(shù)將結(jié)果鏈表轉(zhuǎn)為表達(dá)式輸出。69約瑟夫環(huán)(JosephCircle)問題描述: 編號為1,2,...,n的n個人按順時針方向圍坐一圈,每人持有一個密碼(正整數(shù))?,F(xiàn)在給定一個隨機(jī)數(shù)m>0,從編號為1的人開始,按順時針方向1開始順序報數(shù),報到m時停止。報m的人出圈,同時留下他的密碼作為新的m值,從他在順時針方向上的下一個人開始,重新從1開始報數(shù),如此下去,直至所有的人全部出列為止。 請編寫程序求出圈的順序。70約瑟夫環(huán)(JosephCircle)例:4596157318213224m:密碼k:計數(shù)出隊序列:startk=1571約瑟夫環(huán)(JosephCircle)例:4596157318213224m:密碼k:計數(shù)出隊序列:startk=2572約瑟夫環(huán)(JosephCircle)例:4596157318213224m:密碼k:計數(shù)出隊序列:startk=3573約瑟夫環(huán)(JosephCircle)例:4596157318213224m:密碼k:計數(shù)出隊序列:startk=4574約瑟夫環(huán)(JosephCircle)例:m:密碼k:計數(shù)出隊序列:start4596157318213224k=55存入密碼75約瑟夫環(huán)(JosephCircle)例:96157318213224m:密碼k:計數(shù)出隊序列:startk=14576約瑟夫環(huán)(JosephCircle)例:96157318213224m:密碼k:計數(shù)出隊序列:start5k=4477約瑟夫環(huán)(JosephCircle)例:961573113224m:密碼k:計數(shù)出隊序列:start5k=18278約瑟夫環(huán)(JosephCircle)例:961573113224m:密碼k:計數(shù)出隊序列:start52k=8879約瑟夫環(huán)(JosephCircle)例:1573113224m:密碼k:計數(shù)出隊序列:start52k=19680約瑟夫環(huán)(JosephCircle)例:1573113224m:密碼k:計數(shù)出隊序列:start52k=99681約瑟夫環(huán)(JosephCircle)例:3113224m:密碼k:計數(shù)出隊序列:start52k=1156782約瑟夫環(huán)(JosephCircle)例:3113224m:密碼k:計數(shù)出隊序列:start5267k=151583約瑟夫環(huán)(JosephCircle)例:3113m:密碼k:計數(shù)出隊序列:start52674k=12284約瑟夫環(huán)(JosephCircle)例:3113m:密碼k:計數(shù)出隊序列:start52674k=222285約瑟夫環(huán)(JosephCircle)例:31m:密碼k:計數(shù)出隊序列:start526743k=1186約瑟夫環(huán)(JosephCircle)例:m:密碼k:計數(shù)出隊序列:start526743k=13187約瑟夫環(huán)(JosephCircle)存儲結(jié)構(gòu)4596157318213224tailp需要的變量:k:計數(shù)p:指向當(dāng)前結(jié)點pre:指向p的前驅(qū)結(jié)點上機(jī)實現(xiàn)提示:①編寫建立循環(huán)鏈表算法②編寫結(jié)點輸出算法(帶輸入?yún)?shù)k)③編寫主程序pre88Joseph問題代碼#include<stdio.h>#include<stdlib.h>typedef

structnode{

intcode;/*游戲者的編號*/u

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
  • 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
  • 5. 人人文庫網(wǎng)僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負(fù)責(zé)。
  • 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時也不承擔(dān)用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

評論

0/150

提交評論