一、線(xiàn)性表的定義
線(xiàn)性表(List):由零個(gè)或多個(gè)數(shù)據(jù)元素組成的有限序列。
這里需要強(qiáng)調(diào)幾個(gè)關(guān)鍵的地方:
- 首先它是一個(gè)序列,也就是說(shuō)元素之間是有個(gè)先來(lái)后到的。
- 若元素存在多個(gè),則第一個(gè)元素?zé)o前驅(qū),而最后一個(gè)元素?zé)o后繼,而其他元素都有且只有一個(gè)前驅(qū)和后繼。
- 另外,線(xiàn)性表強(qiáng)調(diào)是有限的,事實(shí)上無(wú)論計(jì)算機(jī)發(fā)展多強(qiáng)大,它所處理的元素都是有限的。
二、數(shù)據(jù)類(lèi)型的定義
數(shù)據(jù)類(lèi)型:是指一組性質(zhì)相同的值的集合及定義在此集合上的一些操作的總稱(chēng)(整型、浮點(diǎn)型、字符型)。
三、線(xiàn)性表的抽象數(shù)據(jù)類(lèi)型
Operation
- InitList(*L):初始化操作,建立一個(gè)空的線(xiàn)性表。
- ListEmpty(L):判斷線(xiàn)性表是否為空表,若線(xiàn)性表為空,返回true,否則返回false。
- ClearList(*L):將線(xiàn)性表清空。
- GetElem(L,i,*e):將線(xiàn)性表L中的第i個(gè)位置元素值返回給e。
- LocateElem(L,e):在線(xiàn)性表L中查找與給定值e相等的元素,如果查找成功,返回該元素在表中序號(hào)表示成功。否則,返回0表示失敗。
- ListInsert(*L,i,e):在線(xiàn)性表L中第i個(gè)位置插入新元素e。
- ListDeleted(*l,i,*e):刪除線(xiàn)性表L中第i個(gè)位置元素,并用e返回其值。
- ListLength(L):返回線(xiàn)性表L的元素個(gè)數(shù)。
//AUB
//La表示A集合,Lb表示B集合
void unionL(List *La,List Lb){
int La_len ,Lb_len,i;
ElemType e;
La_len = ListLength(*La);
Lb_len = ListLength(Lb);
for(i = 1;i<=Lb_len;i++){
GetElem(Lb,i,&e)
if(!LocateElem(*La,e)){
ListInsert(La,++La_len,e);
}
}
}
四、線(xiàn)性表的順序存儲(chǔ)結(jié)構(gòu)
線(xiàn)性表的順序存儲(chǔ)結(jié)構(gòu),指的是用一段地址連續(xù)的存儲(chǔ)單元依次存儲(chǔ)線(xiàn)性表的數(shù)據(jù)元素。(像C語(yǔ)言的數(shù)組)
物理上的存儲(chǔ)方式實(shí)際上就是在內(nèi)存中找個(gè)初始地址,然后通過(guò)占位的形式,把一定的內(nèi)存空間給占了,然后把相同數(shù)據(jù)類(lèi)型的數(shù)據(jù)元素依次存放在這塊空地中。
順序存儲(chǔ)結(jié)構(gòu)的結(jié)構(gòu)代碼
#define MAXSIZE 20
typedef int ElemType;
typedef struct{
ElemType data[MAXSIZE];
int length;//線(xiàn)性表當(dāng)前長(zhǎng)度
} SqList;
大家看到了,這里我們封裝了一個(gè)結(jié)構(gòu),事實(shí)上就是對(duì)數(shù)組進(jìn)行封裝,增加了一個(gè)當(dāng)前長(zhǎng)度的變量罷了。
總結(jié)下,順序存儲(chǔ)結(jié)構(gòu)封裝需要三個(gè)屬性:
1.存儲(chǔ)空間的起始位置,數(shù)組data,它的存儲(chǔ)位置就是線(xiàn)性表存儲(chǔ)空間的存儲(chǔ)位置。
2.線(xiàn)性表的最大存儲(chǔ)容量:數(shù)組的長(zhǎng)度MAXSIZE
3.線(xiàn)性表的當(dāng)前長(zhǎng)度:length
地址計(jì)算方法
假設(shè)ElemType占用的是C個(gè)存儲(chǔ)單元(字節(jié)),那么線(xiàn)性表中第i+1個(gè)數(shù)據(jù)元素和第i個(gè)數(shù)據(jù)元素的存儲(chǔ)位置關(guān)系是(Loc表示獲得存儲(chǔ)位置的函數(shù)):LOC(ai+1) = LOC(ai) + c;
所以對(duì)于第i個(gè)數(shù)據(jù)元素ai的存儲(chǔ)位置可以由a1推算出:LOC(ai) = LOC(a1) + (i-1)*c;
通過(guò)這個(gè)公式,我們可以隨時(shí)計(jì)算出線(xiàn)性白哦中任意位置的地址,不管它是第一個(gè)還是最后一個(gè),都是相同的時(shí)間。那么它的存儲(chǔ)時(shí)間性能當(dāng)然就為O(1),我們通常稱(chēng)為隨機(jī)存儲(chǔ)結(jié)構(gòu)。
獲得元素操作
實(shí)現(xiàn)GetElem的具體操作,即將線(xiàn)性表L中的第i個(gè)位置元素值返回。就程序而言非常簡(jiǎn)單了,我們只需要把數(shù)組第i-1下標(biāo)的值返回即可。
GetElem.c
#define OK 1
#define ERROR 0
#define TRUE 1
#define FALSE 0
typedef int Status;
//Status是函數(shù)的類(lèi)型,其值是函數(shù)結(jié)果的狀態(tài)碼,入OK等。
//初始條件:順序線(xiàn)性表L已存在,1 <= i <= ListLength(L)
//操作結(jié)果:用e返回L中第i個(gè)數(shù)據(jù)元素的值
Status GetElem(SqList L,int i,ElemType *e){
if(L.length == 0 || i < 1 || i>L.length){
return ERROR;
}
*e = L.data[i-1];
return OK;
}
插入操作
插入算法的思路:
1.如果插入位置不合理,拋出異常;
2.如果線(xiàn)性表長(zhǎng)度大于等于數(shù)組長(zhǎng)度,則拋出異?;騽?dòng)態(tài)增加數(shù)組容量;
3.從最后一個(gè)元素開(kāi)始向前遍歷到第i個(gè)位置,分別將他們都向后移動(dòng)一個(gè)位置;
4.將要插入元素填入位置i處;
5.線(xiàn)性表長(zhǎng)度+1。
ListInsert.c
//初始條件:順序線(xiàn)性表L已存在,1<=i<=ListLength(L)
//操作結(jié)果:在L中第i個(gè)位置之前插入新的數(shù)據(jù)元素e,L長(zhǎng)度+1
Ststus ListInsert(SqList *L,int i;ElemTYpe e){
int k;
if(L->length == MAXSIZE){//順序線(xiàn)性表已經(jīng)滿(mǎn)了
return ERROR;
}
if(i<1 || i>L->length){//當(dāng)i不在范圍內(nèi)時(shí)
return ERROR;
}
if(i<= L->length){//若插入元素位置不再表尾
//要將插入位置后元素向后移動(dòng)一位
for(k=L->length-1;k>=i-1;k--){
L->data[k+1] = L->data[k];
}
}
L->data[i-1] = e;//將新元素插入
L->length++;
return OK;
}
刪除操作
刪除算法的思路
1.如果刪除位置不合理,拋出異常;
2.取出要被刪除的元素;
3.從刪除元素位置開(kāi)始遍歷到最后一個(gè)元素位置,分別將他們都向前移動(dòng)一個(gè)位置;
4.表長(zhǎng)-1。
ListDeleted.c
//初始條件:順序線(xiàn)性表L已存在,1<=i<=ListLength(L)
//操作結(jié)果:刪除L的第i個(gè)數(shù)據(jù)元素,并用e返回其值,L的長(zhǎng)度-1
Status ListDeleted(SqList *L,int i,ElemType *e){
int k;
if(L->length == 0){
return ERROR;
}
if(i<1 || i>L->length){
return ERROR;
}
*e = L->data[i-1];
if(i<L->length){
for(k=i;k<L->length;k++){
L->data[k-1] = L->data[k];
}
}
L->length--;
return OK;
}
性能分析
現(xiàn)在我們分析一下,插入和刪除的時(shí)間復(fù)雜度。
最好的情況:插入和刪除操作剛好要求在最后一個(gè)位置,因?yàn)椴恍枰苿?dòng)任何元素,所以此時(shí)的時(shí)間復(fù)雜度為O(1)。
最壞的情況:如果要插入和刪除的位置是第一個(gè)元素,那就意味著要移動(dòng)所有的元素像后或者向前,所以這個(gè)時(shí)間復(fù)雜度為O(n)。
至于平均情況,就去中間值O((n-1)/2) = O(n)。
線(xiàn)性表的優(yōu)缺點(diǎn)
線(xiàn)性表的順序存儲(chǔ)結(jié)構(gòu),在存、讀數(shù)據(jù)時(shí),不管是哪個(gè)位置,時(shí)間復(fù)雜度都是O(1)。而在插入或刪除時(shí),使勁復(fù)雜度都是O(n)。
這就說(shuō)明,他比較適合元素個(gè)數(shù)比較穩(wěn)定,不經(jīng)常插入和刪除,而更多的操作是存取數(shù)據(jù)的應(yīng)用。
優(yōu)點(diǎn):
1、無(wú)需為表示表中元素之間的邏輯關(guān)系而增加額外的存儲(chǔ)空間。
2、可以快速的存取表中的任意位置的元素。
缺點(diǎn):
1、插入和刪除操作需要移動(dòng)大量元素。
2、當(dāng)線(xiàn)性表長(zhǎng)度變化較大時(shí),難以確定存儲(chǔ)空間的容量。
3、容易造成存儲(chǔ)空間的“碎片”。
五、線(xiàn)性表的鏈?zhǔn)酱鎯?chǔ)結(jié)構(gòu)
線(xiàn)性表鏈?zhǔn)酱鎯?chǔ)結(jié)構(gòu)定義
線(xiàn)性表的鏈?zhǔn)酱鎯?chǔ)結(jié)構(gòu)的特點(diǎn)是用一組任意的存儲(chǔ)單元存儲(chǔ)線(xiàn)性表的數(shù)據(jù)元素,這組存儲(chǔ)單元可以存在內(nèi)存中未被占用的任意位置。
比起順序存儲(chǔ)結(jié)構(gòu)每個(gè)數(shù)據(jù)元素只需要存儲(chǔ)一個(gè)位置就可以了?,F(xiàn)在鏈?zhǔn)酱鎯?chǔ)結(jié)構(gòu)中,除了要存儲(chǔ)數(shù)據(jù)元素信息外,還要存儲(chǔ)它的后繼元素的存儲(chǔ)地址(指針)。
我們把存儲(chǔ)數(shù)據(jù)元素信息的域稱(chēng)為數(shù)據(jù)域,把存儲(chǔ)直接后繼位置的域稱(chēng)為執(zhí)指針域。指針域中存儲(chǔ)的信息稱(chēng)為指針或鏈。這兩部分信息組成數(shù)據(jù)元素稱(chēng)為存儲(chǔ)映像,稱(chēng)為節(jié)點(diǎn)(Node)。
因?yàn)榇随湵淼拿總€(gè)節(jié)點(diǎn)中包含一個(gè)指針域,所以叫做單鏈表。
對(duì)于線(xiàn)性表來(lái)說(shuō),總得有個(gè)頭有個(gè)尾,鏈表也不例外。我們把鏈表中的第一個(gè)節(jié)點(diǎn)的存儲(chǔ)位置叫做頭指針,最后以客觀(guān)節(jié)點(diǎn)指針為空(NULL)
頭指針與頭節(jié)點(diǎn)的異同
頭節(jié)點(diǎn)的數(shù)據(jù)域一般不存儲(chǔ)任何信息
頭指針:
頭指針是指鏈表指向第一個(gè)節(jié)點(diǎn)的指針,若鏈表有頭節(jié)點(diǎn),則是指向頭節(jié)點(diǎn)的指針。
頭指針具有標(biāo)識(shí)作用,所以常用頭指針冠以鏈表的名字(指針變量的名字)。
無(wú)論鏈表是否為空,頭指針均不為空。
頭指針是鏈表的必要元素。
頭節(jié)點(diǎn):
頭節(jié)點(diǎn)是為了操作的統(tǒng)一和方便而設(shè)立的,方在第一個(gè)元素的節(jié)點(diǎn)前,其數(shù)據(jù)域一般無(wú)意義(但也可以用來(lái)存放鏈表的長(zhǎng)度)。
有了頭節(jié)點(diǎn),對(duì)在第一個(gè)元素節(jié)點(diǎn)前插入節(jié)點(diǎn)和刪除第一個(gè)節(jié)點(diǎn)其操作與其他節(jié)點(diǎn)的操作就統(tǒng)一了。
頭節(jié)點(diǎn)不一定是鏈表的必須要素。
我們?cè)贑語(yǔ)言中可以用結(jié)構(gòu)指針來(lái)描述單鏈表
typedef struct Node{
ElemType data;//數(shù)據(jù)域
struct Node *Next;//指針域
}Node;
typedef struct Node *LinkList;
我們看到節(jié)點(diǎn)由存放數(shù)據(jù)元素的數(shù)據(jù)域和存放后繼節(jié)點(diǎn)地址的指針域組成。
單鏈表的讀取
在線(xiàn)性表的順序存儲(chǔ)結(jié)構(gòu)中,我們要計(jì)算任意一個(gè)元素的存儲(chǔ)位置是很容易的。
但是在單鏈表中,由于第i個(gè)元素到底在哪?我們壓根沒(méi)辦法一開(kāi)始就知道,必須得從第一個(gè)節(jié)點(diǎn)開(kāi)始挨個(gè)找。
因此,對(duì)于單鏈表實(shí)現(xiàn)獲取第i個(gè)元素的數(shù)據(jù)的操作GetElem,在算法上相對(duì)要麻煩一些。
獲得鏈表第i個(gè)數(shù)據(jù)的算法思路:
1、申明一個(gè)節(jié)點(diǎn)p指向鏈表的第一個(gè)節(jié)點(diǎn),初始化j從1開(kāi)始;
2、當(dāng)j<i時(shí),就遍歷鏈表,讓p的指針像后移動(dòng),不斷指向下一個(gè)節(jié)點(diǎn),j+1;
3、若到鏈表末尾p為空,則說(shuō)明第i個(gè)元素不存在;
4、若查找成功,返回節(jié)點(diǎn)p的數(shù)據(jù)。
GetElem.c
//初始條件:順序線(xiàn)性表L已存在,1<=i<=ListLength(L)
//操作結(jié)果:用e返回L中第i個(gè)數(shù)據(jù)元素的值
Status GetElem(LinkList L,int i,ElemType *e){
int j;
LinkList p;
p = L->next;
j = 1;
while(p && j<i){
p = p->next;
++j;
}
if(!p || j>i){
return ERROR;
}
*e = p->data;
return OK;
}
說(shuō)白了,就是從頭開(kāi)始找,知道第i個(gè)元素為止。
由于這個(gè)算法的時(shí)間復(fù)雜度取決于i的為止,當(dāng)i=1時(shí),則不需要遍歷,而i=n時(shí)則遍歷n-1次才可以。因此最壞情況的時(shí)間復(fù)雜度為O(n)。
由于單鏈表的結(jié)構(gòu)中沒(méi)有定義表長(zhǎng),所以不能實(shí)現(xiàn)知道要循環(huán)多少次,因此也就不方便使用for來(lái)控制循環(huán)。
其核心思想叫做“工作指針后移”,這其實(shí)也是很多算法的常用技術(shù)。
單鏈表的插入
假設(shè)存儲(chǔ)元素e的節(jié)點(diǎn)為s,要實(shí)現(xiàn)節(jié)點(diǎn)p、p->next和s之間邏輯關(guān)系
我們思考后發(fā)覺(jué)根本用不著驚動(dòng)其他節(jié)點(diǎn),只需要讓s->next和p->next的指針做一點(diǎn)變化。
s->next = p->next;
p->next = s;
那么我們考慮一下大部分初學(xué)者最容易搞壞腦子的問(wèn)題:這兩句代碼的順序可以可以交換過(guò)來(lái)?
p->next = s;
s->next = p->next;
如果先執(zhí)行p->next = s 的話(huà)會(huì)先被覆蓋為s的地址,那么s->next = p->next其實(shí)就等于s->next = s 了。
所以這兩句是無(wú)論如何不能弄反的,這點(diǎn)初學(xué)者一定要注意。
單鏈表第i個(gè)數(shù)據(jù)插入節(jié)點(diǎn)的算法思路:
1、聲明一節(jié)點(diǎn)p指向鏈表頭節(jié)點(diǎn),初始化j從1開(kāi)始;
2、當(dāng)j<i時(shí),就遍歷鏈表,讓p的指針向后移動(dòng),不斷指向下一節(jié)點(diǎn),j累加1;
3、若到鏈表末尾p為空,則說(shuō)明第i個(gè)元素不存在;
4、否則查找成功,在系統(tǒng)中生成一個(gè)空節(jié)點(diǎn)s;
5、將數(shù)據(jù)元素e復(fù)制給s->data;
6、單鏈表的插入剛才兩個(gè)標(biāo)準(zhǔn)語(yǔ)句;
7、返回成功。
//初始條件:鏈表L已經(jīng)存在,1<=i<=ListLength(L)
//操作結(jié)果:在L中第i個(gè)位置前插入新的數(shù)據(jù)元素e,L的長(zhǎng)度加1
Status ListInsert(LinkList *L,int i,ElemType e){
int j;
LinkList p,s;
p = *L;
j = 1;
while(p && j<i){
p = p->next;
j++;
}
if(!p || j>i){
return ERROR;
}
s = (LinkList)malloc(sizeof(Node));
s->data = e;
s->next = p->next;
p->next = s;
return OK;
}
單鏈表的刪除
假設(shè)元素a2的節(jié)點(diǎn)為q,要實(shí)現(xiàn)節(jié)點(diǎn)q刪除單鏈表的操作,其實(shí)就是將它的前繼節(jié)點(diǎn)的指針繞過(guò)指向后繼節(jié)點(diǎn)即可。
那我們要做的,實(shí)際上就是一步:
p->next = p->next->next;
或
q = p->next;
p->next = q->next;
單鏈表的刪除節(jié)點(diǎn)的算法思路:
1、申明節(jié)點(diǎn)p指向鏈表第一個(gè)節(jié)點(diǎn),初始化j = 1;
2、當(dāng)j<i時(shí),就遍歷鏈表,讓P的指針向后移動(dòng),不斷指向下一個(gè)節(jié)點(diǎn),j累加1;
3、若到鏈表末尾p為空,則說(shuō)明第i個(gè)元素不存在;
4、否則查找成功,將欲刪除節(jié)點(diǎn)p->next賦值給q;
5、單鏈表的刪除標(biāo)準(zhǔn)語(yǔ)句p->next = q->next;
6、將q節(jié)點(diǎn)中的數(shù)據(jù)賦值給e,作為返回;
7、釋放q節(jié)點(diǎn)。
Status ListDeleted(LinkList *L,int i,ElemType e){
int j;
LinkList p,q;
p = *L;
j = 1;
while(p && j<i){
p = p->next;
j++;
}
if(!(p->next) || j>i){
return ERROR;
}
q = p->next;
p->next = q->next;
*e = q->data;
free(q);
return OK;
}
效率PK
我們發(fā)現(xiàn)無(wú)論是單鏈表插入還是刪除算法,他們其實(shí)都是由兩部分組成:第一部分就是遍歷查找第i個(gè)元素,第二部分就是實(shí)現(xiàn)插入和刪除元素。
從整個(gè)算法來(lái)書(shū),我們很容易可以退出他們的時(shí)間復(fù)雜度都是O(n)。
在詳細(xì)點(diǎn)分析:如果在我們不知道第i個(gè)元素的指針位置,單鏈表數(shù)據(jù)結(jié)構(gòu)在插入和刪除操作上,與線(xiàn)性表的順序存儲(chǔ)結(jié)構(gòu)是沒(méi)有太大優(yōu)勢(shì)的。
但如果,我們希望從第i個(gè)位置開(kāi)始,插入連續(xù)10個(gè)元素,對(duì)于順序存儲(chǔ)結(jié)構(gòu)意味著,每一次插入都需要移動(dòng)n-1個(gè)位置,所以每次都是O(n)。
而單鏈表,我們只需要在第一次時(shí),找到第i個(gè)位置的指針,此時(shí)為O(n),接下來(lái)只是簡(jiǎn)單的通過(guò)復(fù)制移動(dòng)指針而已,時(shí)間復(fù)雜度都是O(1)。
顯然,對(duì)于插入或刪除數(shù)據(jù)越頻繁的擦歐總,單鏈表的效率優(yōu)勢(shì)就越是明顯。
單鏈表的正表創(chuàng)建
對(duì)于順序存儲(chǔ)結(jié)構(gòu)的線(xiàn)性表的整表創(chuàng)建,我們可以用數(shù)組的初始化來(lái)直觀(guān)理解。
而單鏈表和順序存儲(chǔ)結(jié)構(gòu)就不一樣了,它不像順序存儲(chǔ)結(jié)構(gòu)數(shù)據(jù)這么集中,它的數(shù)據(jù)可以是分散在內(nèi)存各個(gè)角落的,它的增長(zhǎng)也是動(dòng)態(tài)的。
對(duì)于每個(gè)鏈表來(lái)說(shuō),它所占用空間的大小和位置是不需要魚(yú)線(xiàn)分配劃定的,可以根據(jù)系統(tǒng)的情況和實(shí)際的需求即時(shí)生成。
創(chuàng)建單鏈表的過(guò)程是一個(gè)動(dòng)態(tài)生成鏈表的過(guò)程,從“空表”的初始狀態(tài)起,依次建立各個(gè)元素節(jié)點(diǎn)并逐個(gè)插入鏈表。
所以,單鏈表整表創(chuàng)建的算法思路如下:
1、聲明一個(gè)節(jié)點(diǎn)P和計(jì)數(shù)器變量i;
2、初始化一空鏈表L;
3、讓L的頭節(jié)點(diǎn)的指針指向NULL,即建立一個(gè)帶頭節(jié)點(diǎn)的單鏈表;
4、循環(huán)實(shí)現(xiàn)后繼節(jié)點(diǎn)的復(fù)制和插入。
頭插法建立單鏈表
頭插法從一個(gè)空表開(kāi)始,生成新節(jié)點(diǎn),讀取數(shù)據(jù)存放到新節(jié)點(diǎn)的數(shù)據(jù)域中,然后將新節(jié)點(diǎn)插入到當(dāng)前鏈表的表頭上,直到結(jié)束為止。
簡(jiǎn)單來(lái)說(shuō),就是把新加進(jìn)的元素放在表頭后的第一個(gè)位置:
先讓新節(jié)點(diǎn)的next指向頭節(jié)點(diǎn)之后
然后讓表頭的next指向新節(jié)點(diǎn)
void CreateListHead(LinkList *L,int n){
LinkList p;
int i;
srand(time(0));//初始化隨機(jī)數(shù)字
*L = (LinkList)malloc(sizeof(Node));
(*L)->next = NULL;
for(i=0;i<n;i++){
p = (LinkLIst)malloc(sizeof(Node));//生成新節(jié)點(diǎn)
p->data = rand()%100+1;
p->next = (*L)->next;
(*L)->next = p;
}
}
尾插法建立單鏈表
頭插法建立鏈表雖然算法簡(jiǎn)單,但生成的鏈表中節(jié)點(diǎn)的次序和輸入的順序相反。
void CreateListTail(LinkList *L,int n){
LinkList p,r;
int i;
srand(time(0));//初始化隨機(jī)數(shù)字
*L = (LinkList)malloc(sizeof(Node));
r = *L;
for(i=0;i<n;i++){
p = (LinkLIst)malloc(sizeof(Node));//生成新節(jié)點(diǎn)
p->data = rand()%100+1;
r->next = p;
r = p;//備注:初學(xué)者可能很難理解這句,重點(diǎn)解釋。
}
r->next = NULL;
}
單鏈表的整表刪除
單鏈表整表刪除的算法思路如下:
1、申明節(jié)點(diǎn)p和q;
2、將第一個(gè)節(jié)點(diǎn)賦值給p,下一節(jié)點(diǎn)賦值給q;
3、循環(huán)執(zhí)行釋放p和將q復(fù)制給p的操作;
Status ClearList(LinkList *L){
LinkList p,q;
p = (*L)->next;
while(p){
q = p->next;
free(p);
p = q;
}
(*L)->next = NULL;
return OK;
}
六、單鏈表結(jié)構(gòu)與順序存儲(chǔ)結(jié)構(gòu)優(yōu)缺點(diǎn)
我們分別從存儲(chǔ)分配方式、時(shí)間性能、空間性能三方面來(lái)做對(duì)比。
存儲(chǔ)分配方式
順序存儲(chǔ)結(jié)構(gòu)用一段連續(xù)的存儲(chǔ)單元依次存儲(chǔ)線(xiàn)性表的數(shù)據(jù)元素。
單鏈表采用鏈?zhǔn)酱鎯?chǔ)結(jié)構(gòu),用一組任意的存儲(chǔ)單元存放線(xiàn)性表的元素。
時(shí)間性能
- 1.查找
順序存儲(chǔ)結(jié)構(gòu)O(1)
單鏈表O(n)
- 2.插入和刪除
順序存儲(chǔ)結(jié)構(gòu)需要平均移動(dòng)表長(zhǎng)一般的元素,時(shí)間為O(n)
單鏈表在計(jì)算出某位置的指針后,插入和刪除時(shí)間僅為O(1)
空間性能
順序存儲(chǔ)結(jié)構(gòu)需要預(yù)分配存儲(chǔ)空間,分大了,容易造成空間浪費(fèi),分小了,容易發(fā)生溢出。
單鏈表不需要分配存儲(chǔ)空間,只要有就可以分配,元素個(gè)數(shù)也不受限制。
經(jīng)驗(yàn)性結(jié)論:
若線(xiàn)性表需要頻繁查找,很少進(jìn)行插入和刪除操作時(shí),宜采用順序存儲(chǔ)結(jié)構(gòu)。
若需要頻繁插入和刪除時(shí),宜采用單鏈表結(jié)構(gòu)。
七、靜態(tài)鏈表
在講解原理之前,先讓大家直到這種用數(shù)組描述的鏈表叫做靜態(tài)鏈表,這種描述方法叫做游標(biāo)實(shí)現(xiàn)法。
線(xiàn)性表的靜態(tài)鏈表存儲(chǔ)結(jié)構(gòu)
#define MAXSIZE 1000
typedef struct{
ElemType data;//數(shù)據(jù)
int cur;//游標(biāo)
}Component,StaticLinkList[MAXSIZE];
對(duì)靜態(tài)鏈表進(jìn)行初始化相當(dāng)于初始化數(shù)組:
Status InitList(StaticLinkList space){
int i;
for(i=0;i<MAXSIZE-1;i++){
space[i].cur = i+1;
}
space[MAXSIZE-1].cur = 0;
return OK;
}
我們對(duì)數(shù)組的第一個(gè)和最后一個(gè)元素做特俗處理,他們的data不存放數(shù)據(jù)。
我們通常把未使用的數(shù)組元素稱(chēng)為備用鏈表。
數(shù)組的第一個(gè)元素,即下標(biāo)為0的那個(gè)元素的cur就存放備用鏈表的第一個(gè)節(jié)點(diǎn)的下標(biāo)。
數(shù)組的最后一個(gè)元素,即下標(biāo)為MAXSIZE-1的cur則存放第一個(gè)有數(shù)值的元素的下標(biāo),相當(dāng)于單鏈表中的頭節(jié)點(diǎn)作用。
靜態(tài)鏈表的插入操作
首先是獲得空閑分量的下標(biāo):
int Malloc_SLL(StaticLinkList space){
int i = space[0].cur;
if(space[0].cur){
space[0].cur = space[i].cur;
//把它的下一個(gè)分量用來(lái)作為備用。
}
return i;
}
Status ListInsert(StaticLinkList L ,int i,ElemType e){
int j,k,l;
k = MAX_SIZE - 1;//數(shù)組的最后一個(gè)元素
if(i<1 || i>ListLength(L)+1){
return ERROR;
}
j = Malloc_SLL(L);
if(j){
L[j].data = e;
for(l=1;l<i-1;l++){
k = L[k].cur;
}
L[j].cur = L[k].cur;
L[k].cur = j;
return OK;
}
return ERROR;
}
靜態(tài)鏈表的刪除操作
void Free_SSL(StaticLinkList space,int k){
space[k].cur = space[0].cur;
space[0].cur = k;
}
Status ListDeleted(StaticLinkList L ,int i){
int j,k;
if(i<1 || j>ListLength(L)){
return ERROR;
}
k = MAX_SIZE-1;
for(j=1;j<=i-1;j++){
k = L[k].cur;
}
j - L[k].cur;
L[k].cur = L[j].cur;
Free_SLL(L,j);
return OK;
}
靜態(tài)鏈表優(yōu)缺點(diǎn)總結(jié)
- 1、優(yōu)點(diǎn)
在插入和刪除操作時(shí),只需要修改游標(biāo),不需要移動(dòng)元素,從而改進(jìn)了在順序存儲(chǔ)結(jié)構(gòu)中的插入和刪除操作需要移動(dòng)大量元素的缺點(diǎn)
- 2、缺點(diǎn)
沒(méi)有解決連續(xù)存儲(chǔ)分配(數(shù)組)帶來(lái)的表長(zhǎng)難以確定的問(wèn)題
失去了順序存儲(chǔ)結(jié)構(gòu)隨機(jī)存取的特性。
- 3、總結(jié)
總的來(lái)說(shuō),靜態(tài)鏈表其實(shí)是為了給沒(méi)有指針的編程語(yǔ)言設(shè)計(jì)的一種實(shí)現(xiàn)單鏈表功能的方法。
盡管我們可以用單鏈表就不用靜態(tài)鏈表了,但這樣的思考方式是非常巧妙的,應(yīng)該理解其思想,以備不時(shí)之需。
單鏈表小結(jié)騰訊面試題
題目:快速找到未知長(zhǎng)度單鏈表的中間節(jié)點(diǎn)。
既然是面試題就一定有普通方法和高級(jí)方法,而高級(jí)方法,而高級(jí)方法無(wú)疑會(huì)讓面試官大大加分!
普通方法很簡(jiǎn)單,首先遍歷一遍單鏈表以確定單鏈表的長(zhǎng)度L。然后再次從頭節(jié)點(diǎn)觸發(fā)循環(huán)L/2次找到單鏈表的中間節(jié)點(diǎn)。
算法的復(fù)雜度為:O(L+L/2) = O(3/2L)
能否再優(yōu)化一下這個(gè)時(shí)間復(fù)復(fù)雜度呢?
有一個(gè)很巧妙的方法:利用快慢指針!
利用快慢指針原理:設(shè)置兩個(gè)指針*search、*mid都指向單鏈表的頭節(jié)點(diǎn)。其中*search的移動(dòng)速度是*mid的2倍。當(dāng)*search指向末尾節(jié)點(diǎn)的時(shí)候,*mid正好就在中間了。這也是標(biāo)尺的思想。
GetMidNode.c
Status GetMidNode(LinkList L,ElemType *e){
LinkList search,mid;
mid = search = L;
while(search->next != NULL){
//search移動(dòng)的速度是mid的2倍
if(search->next->next !=NULL){
search = search->next->next;
mid = mid->next;
}else{
search = search->next;
}
}
*e = mid->data;
return OK;
}
八、循環(huán)鏈表
對(duì)于單鏈表,由于每個(gè)節(jié)點(diǎn)只存儲(chǔ)了向后的指針,到了尾部標(biāo)識(shí)就停止了向后鏈的操作。也就是說(shuō),按照這樣的方式,只能索引后繼節(jié)點(diǎn)不能索引前驅(qū)節(jié)點(diǎn)。
這會(huì)帶來(lái)什么問(wèn)題呢?
例如不從頭節(jié)點(diǎn)出發(fā),就無(wú)法訪(fǎng)問(wèn)到全部節(jié)點(diǎn)。
事實(shí)上要解決這個(gè)問(wèn)題也并不麻煩,只需要將單鏈表中終端節(jié)點(diǎn)的指針由空指針改為指向頭節(jié)點(diǎn),問(wèn)題就結(jié)了。
將單鏈表中終端節(jié)點(diǎn)的指針端由空指針改為指向頭節(jié)點(diǎn),就使整個(gè)單鏈表形成一個(gè)環(huán),這種頭尾相接的單鏈表成為單循環(huán)鏈表,簡(jiǎn)稱(chēng)循環(huán)鏈表。
初始化循環(huán)鏈表
//鏈表存儲(chǔ)結(jié)構(gòu)定義
typedef struct CLinkList{
int data;
struct ClinkList *next;
}
void ds_init(node **pNode){
int item;
node *temp;
node *target;
printf("輸入節(jié)點(diǎn)的值,輸入0完成初始化");
while(1){
scanf("%d",&item);
fflush(stdin);
if(item == 0){
return;
}
if((*pNode == NULL)){
//循環(huán)鏈表中只有一個(gè)節(jié)點(diǎn)
*pNode = (node*)malloc(sizeiof(struct ClinkList));
if(!(*pNode)){
exit(0);
}
(*pNode)->data = item;
(*pNode)->next = *pNode;
}else{
//找到next指向第一個(gè)節(jié)點(diǎn)的節(jié)點(diǎn)
for(target = (*pNode);target->next!=(*pNode);target = target->next){
}
//生成一個(gè)新的節(jié)點(diǎn)
temp = (node *)malloc(sizeof(struct CLinkList));
if(!temp){
exit(0);
}
temp->data = item;
temp->next = *pNode;
target->next = temp;
}
}
}
循環(huán)鏈表的插入
void ds_insert(node **pNode,int i){
node *temp;
node *target;
node *p;
int item;
int j;
printf("輸入要插入節(jié)點(diǎn)的值:");
scanf("%d",&item);
if(i == 1){
//新插入的節(jié)點(diǎn)作為第一個(gè)節(jié)點(diǎn)
temp = (node *)malloc(sizeof(struct CLinkList));
if(!temp){
exit(0);
}
temp->data = item;
//尋找到最后一個(gè)節(jié)點(diǎn)
for(target = (*pNode);target->next != (*pNode);target = target->next){}
temp->next = (*pNode);
target->next = temp;
*pNode = temp;
}else{
target = *pNode;
for(j = 1;j<(i-1);++j){
target = target->next;
}
temp = (node *)malloc(sizeof(struct CLinkList));
if(!temp){
exit(0);
}
temp->data = item;
p = target->next;
target->next = temp;
temp->next = p;
}
}
循環(huán)鏈表返回節(jié)點(diǎn)位置
int ds_search(node *pNode,int item){
node *8target;
int i;
for(target = pNode;target->data!=elem && target->next != pNode;++i){
target = target->next;
}
if(target->next == pNode){
return 0;
}else{
return i;
}
}
循環(huán)鏈表的特點(diǎn)
回顧一下,在單鏈表中,我們有了頭節(jié)點(diǎn)時(shí),我們可以用O(1)的時(shí)間訪(fǎng)問(wèn)第一個(gè)節(jié)點(diǎn),但對(duì)于要訪(fǎng)問(wèn)最后一個(gè)節(jié)點(diǎn),我們必須要挨個(gè)向下索引,所以需要O(n)的時(shí)間。
如果用上今天我們學(xué)的循環(huán)鏈表的特點(diǎn),用O(1)的時(shí)間就可以由鏈表指針訪(fǎng)問(wèn)到最后一個(gè)節(jié)點(diǎn)。
不過(guò)我們需要改造一下現(xiàn)有的循環(huán)鏈表,我們不用頭指針,而是用指向指端節(jié)點(diǎn)的尾指針來(lái)表示循環(huán)鏈表,此時(shí)查找開(kāi)始節(jié)點(diǎn)和終端節(jié)點(diǎn)都很方便了。
判斷單鏈表中是否有環(huán)
又環(huán)的定義是,鏈表的尾部節(jié)點(diǎn)指向了鏈表中的某個(gè)節(jié)點(diǎn)(不一定是第一個(gè)節(jié)點(diǎn))。
那么要判斷單鏈表中是否有環(huán),主要有一下兩種方法。
方法一:使用p、q兩個(gè)指針,p總是向前走,但q每次都從頭開(kāi)始走,對(duì)于每個(gè)節(jié)點(diǎn),看p走的步數(shù)是否和q一樣。如圖,當(dāng)p從6走到3時(shí)候,用了6步,此時(shí)若q從head出發(fā)則只需要兩部就到3,因?yàn)椴綌?shù)不等,出現(xiàn)矛盾,存在環(huán)。
方法二:使用p、q兩個(gè)指針,p每次向前走一步,q每次向前走兩步,若在某個(gè)時(shí)候p==q,則存在環(huán)。
循環(huán)鏈表的應(yīng)用-約瑟夫問(wèn)題
據(jù)說(shuō)著名猶太歷史學(xué)家Josephus有過(guò)以下的故事:
在羅馬人占領(lǐng)橋塔帕特后,39個(gè)猶太人與Josephus及它的朋友躲到一個(gè)洞中,39個(gè)猶太人決定寧愿死也不要被敵人抓到,于是決定了一個(gè)自殺方式,41個(gè)人,排成一個(gè)圓圈,由第一個(gè)人開(kāi)始報(bào)數(shù),沒(méi)報(bào)數(shù)到第三個(gè)人就必須自殺,然后再由下一個(gè)重新報(bào)數(shù),直到所有人都自殺身亡為止。然而Josephus和他的朋友并不想遵從,Josephus要他的朋友先假裝遵從,它將朋友與自己安排在第16個(gè)與第31個(gè)位置,于是逃過(guò)了這場(chǎng)游戲。
問(wèn)題:用循環(huán)鏈表模擬約瑟夫問(wèn)題,把41個(gè)人的順序編號(hào)輸出。
#include <stdio.h>
#include <stdlib.h>
typedef struct node{
int data;
struct node *next;
}node;
node *create(int n){
node *p = NULL;
node *head;
p = head;
node *s;
int i =1;
if(0 != n){
while(i<=n){
s = (node *)malloc(sizeof(node));
s->data = i++;
p->next = s;
p = s;
}
s->next = head->next;
}
//free(head);
return s->next;
}
int main(){
int n = 41;
int m = 3;
int i;
node *p = create(n);
node *temp;
m %= n;//m = 2
while(p != p->next){
for(i = 1;i<m-1;i++){
p = p->next;
}
printf("%d->",p->next->data);
temp = p->next;
p->next = temp->next;
free(temp);
p = p->next;
}
printf("%d ",p->data);
return 0;
}
循環(huán)鏈表的應(yīng)用-魔術(shù)師發(fā)牌問(wèn)題
問(wèn)題描述:魔術(shù)師利用一副牌中的13張黑牌,預(yù)先將他們排好后疊放在一起,牌面朝下。對(duì)著觀(guān)眾說(shuō):“我不看牌,只數(shù)數(shù)就可以猜到每張牌是什么”
#include <stdio.h>
#include <stdlib.h>
#define CardNumber 13
typedef struct node{
int data;
struct node *next;
}sqlist,*linklist;
linklist CreateLinkList(){
linklist head = NULL;
linklist s,r;
int i;
r = head;
for(i = 1;i<=CardNumber;i++){
s = (linklist)malloc(sizeof(sqlist));
s->data = 0;
if(head == NULL){
head = s;
}else{
r->next = s;
}
r = s;
}
r->next = head;
return head;
}
//發(fā)牌順序計(jì)算
void Magician(linklist head){
linklist p;
int j;
int Countnumber =2;
p = head;
p->data = 1;//第一張牌數(shù)1
while(1){
for(j = 0;j<Countnumber;j++){
p = p->next;
if(p->data != 0){//該位置有牌的話(huà),則下一個(gè)位置
//p->next;
j--;
}
}
if(p->data == 0){
p->data = Countnumber;
Countnumber++;
if(Countnumber == 14){
break;
}
}
}
}
int main(){
linklist head;
head = CreateLinkList();
Magician(head);
int i;
for(i = 0;i<13;i++){
printf("%d-->",head->data);
head = head->next;
}
return 0;
}
循環(huán)鏈表的應(yīng)用-拉丁方陣問(wèn)題
拉丁方陣是一種n*n的方陣,方陣中恰有n種不同的元素,每種元素恰有n個(gè),并且每種元素在一行和一列中恰好出現(xiàn)一次。著名數(shù)學(xué)家個(gè)物理學(xué)家歐拉使用拉丁字母來(lái)作為拉丁方陣?yán)镌氐姆?hào),拉丁方陣因此而得名。
1 2 3
2 3 1
3 1 2
#include<stdio.h>
#include<stdlib.h>
typedef struct node{
int data;
struct node *next;
}node;
node * makeList(int n){
node *head = NULL;
node *temp;
node *s;
int i;
temp = head;
for(i=1;i<=n;i++){
s = (node *)malloc(sizeof(node));
s->data = i;
if(head == NULL){
head = s;
}else{
temp->next = s;
}
temp = s;
}
temp->next = head;
return head;
}
int main(){
int n,i,j,k;
printf("請(qǐng)輸入方陣的大?。?);
scanf("%d",&n);
node * head;
node * temp;
head = makeList(n);
temp = head;
for(i = 1;i<=n;i++){
for(j = 1;j<=n;j++){
printf("%d ",temp->data);
temp = temp->next;
}
temp = head;
for(k=1;k<=i;k++){
temp = temp->next;
}
printf(" ");
}
return 0;
}
九、雙向鏈表
雙向鏈表節(jié)點(diǎn)結(jié)構(gòu)
typedef struct DualNode{
ElemType data;
struct DualNode *prior;//前驅(qū)節(jié)點(diǎn)
struct DualNode *next;//后繼節(jié)點(diǎn)
}DualNode,*DualList;
既然單鏈表可以有循環(huán)鏈表,那么雙向鏈表當(dāng)然也可以有
雙向鏈表的插入
插入操作其實(shí)并不復(fù)雜,不過(guò)順序很重要,千萬(wàn)不能寫(xiě)反了。
s->next = p;
s->prior = p->prior;
p->prior->next = s;
p->prior = s;
關(guān)鍵在于交換的過(guò)程中不要出現(xiàn)矛盾,例如第四步先被執(zhí)行了,那么p->prior就會(huì)提前變?yōu)閟,使得插入的工作出錯(cuò)。
雙向鏈表的刪除操作
如果剛才的插入操作理解了,那么再來(lái)理解接下來(lái)的刪除操作就容易多了。
p->prior->next = p->next;
p->next->prior = p->proor;
free(p);
最后總結(jié)一下,雙向鏈表相對(duì)于單鏈表來(lái)說(shuō),是要更復(fù)雜一點(diǎn),買(mǎi)個(gè)節(jié)點(diǎn)多了一個(gè)prior指針,對(duì)于插入和刪除操作的順序大家要格外小心。
不過(guò),雙向鏈表可以有效提高算法的時(shí)間性能,說(shuō)白了就是用空間來(lái)?yè)Q取時(shí)間。
????
本文摘自 :https://blog.51cto.com/u