《數(shù)據(jù)結(jié)構(gòu):思想與方法》第二章線性表_第1頁
《數(shù)據(jù)結(jié)構(gòu):思想與方法》第二章線性表_第2頁
《數(shù)據(jù)結(jié)構(gòu):思想與方法》第二章線性表_第3頁
《數(shù)據(jù)結(jié)構(gòu):思想與方法》第二章線性表_第4頁
《數(shù)據(jù)結(jié)構(gòu):思想與方法》第二章線性表_第5頁
已閱讀5頁,還剩179頁未讀, 繼續(xù)免費閱讀

下載本文檔

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

文檔簡介

1表的概念表的實現(xiàn)線性表類的實現(xiàn)STL中表的實現(xiàn)第二章線性表2表的概念線性表是N個具有相同特征的結(jié)點A0,A1,…,AN-1構(gòu)成的集合。在這個集合中,除了A0

和AN-1

外,每個元素都有唯一的前趨和后繼。對于每個Ai,它的前驅(qū)是Ai-1,它的后繼是Ai+1。A0只有后繼沒有前驅(qū),AN-1只有前驅(qū)沒有后繼。表的術(shù)語:N為表的大小A0稱為首結(jié)點,AN-1稱為尾結(jié)點空表:元素個數(shù)為零的表。位置:元素Ai在表中的位置為i3表的操作創(chuàng)建一個線性表create():創(chuàng)建一個空的線性表;清除一個線性表clear():刪除線性表中的所有數(shù)據(jù)元素;求線性表的長度length():返回線性表的長度;在第i個位置插入一個元素insert(i,x):使線性表從(a0,a1,…ai-1,ai,…an-1)變成(a0,a1,…ai-1,x,ai,…an-1),參數(shù)i的合法取值范圍是0到n;4刪除第i個位置的元素remove(i):使線性表從(a0,a1,…ai-1,ai,ai+1

…an-1)變成(a0,a1,…ai-1,ai+1,…an-1),參數(shù)i的合法取值范圍是0到n-1;搜索某個元素在線性表中是否出現(xiàn)search(x):在線性表中搜索x是否出現(xiàn),并返回x的位置;訪問線性表的第i個元素visit(i):返回線性表中第i個數(shù)據(jù)元素的值;遍歷線性表運算traverse():按序訪問線性表的每一數(shù)據(jù)元素。5線性表表的概念表的實現(xiàn)線性表類的實現(xiàn)STL中表的實現(xiàn)6表的實現(xiàn)線性表的順序?qū)崿F(xiàn)線性表的鏈接實現(xiàn)7線性表的順序存儲結(jié)構(gòu)線性表中結(jié)點存放在存儲器上一塊連續(xù)的空間中。借助存儲空間的連續(xù)性,結(jié)點可以按照其邏輯順序依次存放。結(jié)點存放的物理位置和它的邏輯位置是一致的。

8線性表的順序存儲在程序設(shè)計語言中,一塊連續(xù)的存儲空間可以用一個數(shù)組實現(xiàn)。由于線性表中的元素個數(shù)是動態(tài)的,因此采用了動態(tài)數(shù)組。保存一個動態(tài)數(shù)組,需要三個變量:指向線性表元素類型的指針,數(shù)組規(guī)模(容量),數(shù)組中的元素個數(shù)(表長)。anan-1…a2a1a0maxSizelengthdata9線性表的運算實現(xiàn)length():只需要返回length的值visit(i):返回data[i]的值traverse():輸出數(shù)組data的前l(fā)ength個元素clear():只要將length置為0即可create(maxSize):申請一個maxSize大小的動態(tài)數(shù)組10search運算的實現(xiàn)intsearch(x){num=0;for(num=0;num<length;++num)if(data[num]==x)break;if(num==length)num=-1;returnnum;}11insert(i,x)運算的實現(xiàn)length在插入時,表長會增加。當(dāng)表長等于容量時,新增加的元素將無法存儲。此時有兩種解決方法:一種是不執(zhí)行插入,報告一個錯誤消息;另一種是擴大數(shù)組的容量a0a1…aiai+1…an-112voidinsert(i,x){if(length==maxSize)resize();for(j=n-1;j>=i;--j)data[j+1]=data[j];data[i]=x;++length;}13resize操作的實現(xiàn)resize操作按一定的比例擴大數(shù)組的空間,常用的比例是擴大一倍。數(shù)組空間在內(nèi)存中必須是連續(xù)的,因此,擴大數(shù)組空間的操作必須重新申請一個更大規(guī)模的動態(tài)數(shù)組,將原有數(shù)組的內(nèi)容拷貝到新數(shù)組中,釋放原有數(shù)組空間,將新數(shù)組作為存儲線性表的存儲區(qū)。14a0a1…an-1andatatmp15remove(i)運算的實現(xiàn)an…ai+1ai…a1a0voidremove(i){for(j=i;j<length-1;++j)data[j]=data[j+1];--length;}16順序?qū)崿F(xiàn)的算法分析length,visit和clear的實現(xiàn)與表中的元素個數(shù)無關(guān),因此它們的時間復(fù)雜度是O(1)。traverse()操作遍歷整個表中的所有元素,因此時間復(fù)雜度為O(n)。create操作需要申請一塊動態(tài)數(shù)組的空間,并設(shè)表為空。因此也是O(1)的時間復(fù)雜度。插入操作,需要移動結(jié)點。當(dāng)i等于n時,移動次數(shù)為0。當(dāng)i等于0時,移動次數(shù)為n。最好情況下的時間復(fù)雜度為O(1)最壞情況下的時間復(fù)雜度為O(n)平均的時間復(fù)雜度:如果在每個位置上的插入都是等概率的,則插入算法的平均時間復(fù)雜度為n/217線性表的順序?qū)崿F(xiàn)總結(jié)由于要保持邏輯次序和物理次序的一致性,順序表在插入刪除時需要移動大量的數(shù)據(jù),性能不太理想。由于邏輯次序和物理次序的一致性使得定位訪問的性能很好。順序表比較適合靜態(tài)的、經(jīng)常作定位訪問的線性表。18表的實現(xiàn)線性表的順序?qū)崿F(xiàn)線性表的鏈接實現(xiàn)19線性表的鏈接存儲結(jié)構(gòu)將每個結(jié)點放在一個獨立的存儲單元中,結(jié)點間的邏輯關(guān)系依靠存儲單元中附加的指針來給出。結(jié)點的存儲單元在物理位置上可以相鄰,也可以不相鄰。20線性表的鏈接存儲單鏈表雙鏈表循環(huán)鏈表21單鏈表每個結(jié)點附加了一個指針字段,如next,該指針指向它的直接后繼結(jié)點,最后一個結(jié)點的next字段為空。nilhead22insertai+1aixp23頭結(jié)點、頭指針、尾結(jié)點、尾指針為了消除特殊情況,通常在表頭額外增加一個相同類型的特殊結(jié)點,稱之為頭結(jié)點,在表尾也增加一個額外結(jié)點,稱為尾結(jié)點。它們不是線性表中的組成部分。頭尾結(jié)點的出現(xiàn),使得在表頭和表尾位置上進(jìn)行插入和刪除和在其它結(jié)點位置上是完全一致的,從而使得插入和刪除算法得到簡化。24帶頭結(jié)點的單鏈表a0a1an-1…∧head25Create函數(shù)的實現(xiàn)申請一塊存儲結(jié)點的空間,設(shè)結(jié)點的指針部分為空指針。將結(jié)點地址存入代表單鏈表的變量head?!膆ead26清除一個線性表clear()把所有結(jié)點的空間還給系統(tǒng),把頭結(jié)點的指針部分置為空指針voidclear(){P=頭結(jié)點的直接后繼;

While(p!=空指針){q=p;p=p的直接后繼地址;釋放q的空間;}

頭結(jié)點的后繼指針置為空指針;}27求表的長度length()方法1:從起始結(jié)點開始,沿著后繼指針鏈一個一個往后數(shù),數(shù)到一個結(jié)點,長度加1intlength(){len=0;p=頭結(jié)點的直接后繼;

While(p!=空指針){++len;p=p的直接后繼的地址;}}方法2:用空間換取時間的方法。在保存單鏈表的時候增加一個變量,保存表的長度

28在第i個位置插入一個元素insert(i,x)voidinsert(i,x){for(j=0,p=head;j<i;++j)p=p的直接后繼的地址;

tmp=new結(jié)點;

tmp指向的結(jié)點的數(shù)據(jù)部分=x;

tmp指向的結(jié)點的指針部分=p的直接后繼的地址;

p指向的結(jié)點的指針部分=tmp;}29刪除第i個位置的元素remove(i)voidremove(i){for(j=0,p=head;j<i;++j)p=p的直接后繼的地址;

tmp=p的直接后繼的地址;

p的指針部分=tmp的直接后繼的地址;

freetmp;}30搜索某個元素在線性表中是否出現(xiàn)search(x)intsearch(x){num=0;p=頭結(jié)點的直接后繼;

While(p!=空指針&&p的數(shù)據(jù)部分!=x){++num;p=p的直接后繼的地址;}if(p==空指針)num=-1;

returnnum;}31訪問線性表的第i個元素visit(i)dataTypevisit(i){for(j=0,p=head;j<i;++j)p=p的直接后繼的地址;

returnp指向的結(jié)點的數(shù)據(jù)部分;}32遍歷運算traverse()voidtraverse(){p=頭結(jié)點的直接后繼;

While(p!=空指針){cout<<p指向結(jié)點的數(shù)據(jù)部分;

p=p的直接后繼的地址;

}}33線性表的鏈接存儲單鏈表雙鏈表循環(huán)鏈表34雙鏈表每個結(jié)點附加了兩個指針字段,如prior和next,其中prior字段給出直接前驅(qū)結(jié)點的地址,next給出直接后繼結(jié)點的地址。頭結(jié)點中prior字段為空,它的next字段給出線性表中的首結(jié)點的地址,尾結(jié)點中next字段為空,這種形式的鏈表稱為雙鏈表。headtail∧∧35Create運算創(chuàng)建一個雙鏈表就是創(chuàng)建一個只有頭尾結(jié)點的鏈表,把頭結(jié)點的地址保存在head中,尾結(jié)點的地址保存在tail中head∧tail∧36insertxp37removexp38線性表的鏈接存儲單鏈表雙鏈表循環(huán)鏈表39單循環(huán)鏈表一般單循環(huán)鏈表不帶頭結(jié)點head40雙循環(huán)鏈表頭結(jié)點中prior字段給出尾結(jié)點的地址,尾結(jié)點中next字段給出頭結(jié)點的地址一般也不設(shè)頭尾結(jié)點41線性表表的概念表的實現(xiàn)線性表類的實現(xiàn)STL中表的實現(xiàn)42線性表類的實現(xiàn)線性表抽象類順序表類雙鏈表類43線性表的抽象類template<classelemType>classlist{public:virtualvoidclear()=0;virtualintlength()const=0;virtualvoidinsert(inti,constelemType&x)=0;virtualvoidremove(inti)=0;virtualintsearch(constelemType&x)const=0;virtualelemTypevisit(inti)const=0;virtualvoidtraverse()const=0virtual~list(){};};44線性表的抽象類線性表的抽象類是一個類模板抽象類不包括create函數(shù)。這個運算由每個線性表類的構(gòu)造函數(shù)完成增加了一個析構(gòu)函數(shù),以防內(nèi)存泄漏45線性表類的實現(xiàn)線性表抽象類順序表類雙鏈表類46順序表類的定義classOutOfBound{};classIllegalSize{};template<classelemType>classseqList:publiclist<elemType>{private:elemType*data;intcurrentLength;intmaxSize;voiddoubleSpace();47public:seqList(intinitSize=10);~seqList(){delete[]data;}voidclear(){currentLength=0;}intlength()const{returncurrentLength;}voidinsert(inti,constelemType&x);voidremove(inti);intsearch(constelemType&x)const;elemTypevisit(inti)const;voidtraverse()const;};48構(gòu)造函數(shù)template<classelemType>seqList<elemType>::seqList(intinitSize){if(initSize<=0)throwIllegalSize();data=newelemType[initSize];maxSize=initSize;currentLength=0;}maxSizelength49inserttemplate<classelemType>voidseqList<elemType>::insert(inti,constelemType&x){if(i<0||i>currentLength)throwOutOfBound();if(currentLength==maxSize)doubleSpace();for(intj=currentLength;j>i;j--)data[j]=data[j-1];data[i]=x;++currentLength;}50removetemplate<classelemType>voidseqList<elemType>::remove(inti){if(i<0||i>currentLength-1)throwOutOfBound();for(intj=i;j<currentLength-1;j++)data[j]=data[j+1];--currentLength;}51doubleSpacetemplate<classelemType>voidseqList<elemType>::doubleSpace(){elemType*tmp=data;maxSize*=2;data=newelemType[maxSize];for(inti=0;i<currentLength;++i)data[i]=tmp[i];

delete[]tmp;}52其他函數(shù)自己實現(xiàn)53線性表類的實現(xiàn)線性表抽象類順序表類雙鏈表類54鏈表類的設(shè)計以雙鏈表為例鏈表類的數(shù)據(jù)成員:頭指針、尾指針,以及為了提高時間性能而設(shè)置的數(shù)據(jù)成員currentLength鏈表類必須實現(xiàn)線性表的所有操作。為保證這一點,鏈表類從線性表的抽象類繼承。鏈表的插入、刪除操作都需要將指針移到被操作結(jié)點的前一結(jié)點。特設(shè)計函數(shù)move實現(xiàn)55結(jié)點及其組成鏈表中的結(jié)點包含三部分:數(shù)據(jù)字段、前趨指針和后繼指針字段。數(shù)據(jù)字段可以是任何類型的數(shù)據(jù),這里仍然用ElemType表示;指針字段用于存放其他相關(guān)結(jié)點的地址值。由于結(jié)點類型是鏈表專用的,因此可設(shè)為內(nèi)嵌類。由于鏈表的操作是通過對結(jié)點的操作實現(xiàn)的,于是把結(jié)點類定義成struct,以方便鏈表類訪問。56雙鏈表類的定義template<classelemType>classlinkList:publiclist<elemType>

{private:structnode{elemTypedata; node*prev,*next; node(constelemType&x,node*p=NULL,node*n=NULL)

{data=x;next=n;prev=p;} node():next(NULL),prev(NULL){} ~node(){} };node*head,*tail;intcurrentLength;node*move(inti)const;57public:linkList();~linkList(){clear();deletehead;deletetail;}voidclear();intlength()const{returncurrentLength;}voidinsert(inti,constelemType&x);voidremove(inti);intsearch(constelemType&x)const;elemTypevisit(inti)const;voidtraverse()const;};58構(gòu)造函數(shù)template<classelemType>linkList<elemType>::linkList(){head=newnode;head->next=tail=newnode;tail->prev=head;currentLength=0;}

ΛΛheadtail59cleartemplate<classelemType>voidlinkList<elemType>::clear(){node*p=head->next,*q;head->next=tail;tail->prev=head;while(p!=tail){q=p->next; deletep;p=q; }currentLength=0;}60template<classelemType>voidlinkList<elemType>::insert(inti,constelemType&x){node*pos,*tmp;pos=move(i);tmp=newnode(x,pos->prev,pos);pos->prev->next=tmp;pos->prev=tmp;++currentLength;}insertposxtmp61removetemplate<classelemType>voidlinkList<elemType>::remove(inti){node*pos;pos=move(i);pos->prev->next=pos->next;

pos->next->prev=pos->prev;deletepos;

--currentLength;}xpospos62searchtemplate<classelemType>intlinkList<elemType>::search(constelemType&x)const{node*p=head->next;inti=0;while(p!=tail&&p->data!=x){p=p->next;++i;}if(p==tail)return-1;elsereturni;}63visittemplate<classelemType>elemTypelinkList<elemType>::visit(inti)const{node*p=move(i);returnp->data;}64traversetemplate<classelemType>voidlinkList<elemType>::traverse()const{node*p=head->next;cout<<endl;while(p!=tail){ cout<<p->data<<""; p=p->next;}cout<<endl;} 65movetemplate<classelemType>linkList<elemType>::node*linkList<elemType>::move(inti)const{node*p=head->next;if(i<0||i>currentLength)throwOutOfBound();while(i--)p=p->next;returnp;}66缺點雙鏈表類僅實現(xiàn)了線性表的基本運算,并沒有用到雙鏈表的特點,如逆向查找等67線性表表的概念表的實現(xiàn)線性表類的實現(xiàn)STL中表的實現(xiàn)68STLSTL(標(biāo)準(zhǔn)模版庫)是C++中數(shù)據(jù)結(jié)構(gòu)的實現(xiàn)。這些數(shù)據(jù)結(jié)構(gòu)被稱為集合或容器。STL中線性表的實現(xiàn)有兩種:Vector:線性表的順序?qū)崿F(xiàn)List:線性表的雙鏈表的實現(xiàn)69Vector和list共同支持的操作所有的容器都支持的整體操作:求規(guī)模、清空和判空在表尾的插入和刪除以及取表頭表尾元素List特有的操作:表頭的插入和刪除Vector特有的操作:[]的重載,按下標(biāo)取元素,求容量,重新設(shè)置數(shù)組的大小。迭代器:兩者都能用迭代器訪問70STL中的迭代器操作獲取一個迭代器:Begin:返回一個指向第一個節(jié)點的迭代器End:返回一個指向最后一個節(jié)點的迭代器迭代器本身的方法:向后移動,取迭代器指向的元素值,判迭代器相等和不相等需要迭代器的容器操作:在指定位置上插入一元素刪除指定位置的元素刪除指定位置之間的所有元素71STL中線性表的實現(xiàn)Vector:線性表的順序?qū)崿F(xiàn),且大小可增長List:線性表的雙鏈表實現(xiàn)72Vector的特性Vector維護(hù)一個C的原始數(shù)組、容量及大小規(guī)模提供拷貝構(gòu)造函數(shù)和賦值運算符的重載函數(shù),實現(xiàn)了對數(shù)組的整體操作可以修改數(shù)組的容量提供[]運算符重載提供了一個內(nèi)嵌的迭代器頭文件:vector73Vector的定義template<classobject>classVector{private:inttheSize;inttheCapacity;object*objects;public:enum{SPARE_CAPACITY=16};74explicitVector(intinitSize=0);Vector(constVector&R);~Vector();constVector&operator=(constVector&R);voidresize(intnewsize);//重新設(shè)置表的大小voidreserve(intnewCapacity);//重新設(shè)置數(shù)組的容量

//[]重載object&operator[](intindex);constobject&operator[](intindex)const;不允許隱式轉(zhuǎn)換75boolempty()const;intsize()const;intcapacity()const;//對數(shù)據(jù)元素的操作

voidpush_back(constobject&x);//添加在表尾

voidpop_back();//刪除表尾元素

constobject&back()const;//返回表尾元素的值76//迭代器操作typedefobject*iterator;typedefconstobject*const_iterator;iteratorbegin();//返回元素0的地址const_iteratorbegin()const;//返回元素0的地址iteratorend();//指向最后一個元素的地址const_iteratorend()const;//指向最后一個元素的地址};77Vector的使用#include<iostream>#include<vector>usingnamespacestd;intmain(){vector<int>v;

cout<<"theinitialsizeofvis:"<<v.size()<<"\ntheinitialcapacityofvis:"<<v.capacity()<<endl;

78v.push_back(2);cout<<"\nafterpush2,capacityofvis:"<<v.capacity()<<endl;v.push_back(3);cout<<"\nafterpust3,capacityofvis:“<<v.capacity()<<endl;v.push_back(4);cout<<"\nafterpush4,capacityofvis:“<<v.capacity()<<endl;cout<<"\nthesizeofvis:"<<v.size()<<"\nthecapacityofvis:“<<v.capacity()<<endl;79cout<<"\ncontentsofvusing[]:";

for(inti=0;i<v.size();i++)cout<<v[i]<<"";cout<<endl;cout<<"\ncontentsofvusingiteratornotation:";vector<int>::const_iteratorp;for(p=v.begin();p!=v.end();p++)cout<<*p<<"";cout<<endl;}80執(zhí)行結(jié)果theinitialsizeofvis:0theinitialcapacityofvis:0Afterpush2,capacityofvis:1Afterpush3,capacityofvis:2Afterpush4,capacityofvis:4thesizeofvis:3thecapacityofvis:4contentsofvusing[]:234contentsofvusingiteratornotation:23481STL中表的實現(xiàn)Vector:線性表的順序?qū)崿F(xiàn),且大小可增長List:線性表的雙鏈表實現(xiàn)STL(標(biāo)準(zhǔn)模版庫)是C++中數(shù)據(jù)結(jié)構(gòu)的實現(xiàn)。82List的實現(xiàn)采用了帶頭節(jié)點的雙鏈表實現(xiàn)ΛΛheadtail83List的功能允許在任何位置插入和刪除支持雙向迭代器頭文件:list84List的定義Template<classobject>Classlist{private:structnode{…..};intthesize;node*head,*tail;

voidinit();85Public:classconst_iterator{…}classiterator:publicconst_iterator{…}

list();list(constlist&r);~list();constlist&operator=(constlist&r);86//迭代器操作iteratorbegin();const_iteratorbegin()const;iteratorend();const_iteratorend()const;iteratorinsert(iteratoritr,constObject&x);iteratorerase(iteratoritr);iteratorerase(iteratorstart,iteratorend);87Intsize()const;Boolempty()const;Voidclear();Object&front();constObject&front()const;Object&back();constObject&back()const;voidpush_front(constObject&x);voidpush_back(constObject&x);voidpop_front();voidpop_back();;88List的應(yīng)用#include<iostream>#include<list>usingnamespacestd;template<classobject>voidprintall(constlist<object>&v);intmain(){list<int>v1;

v1.push_front(1);v1.push_front(2);v1.push_back(4);v1.push_back(3);printall(v1);list<int>::iteratoritr=v1.begin(),itre=v1.end();for(inti=5;itr!=itre;++itr,++i)v1.insert(itr,i);printall(v1);return0;}contentsofvaluesis:2143contentsofvaluesis:5261748389template<classobject>voidprintall(constlist<object>&v){if(v.empty())cout<<"\nlistisempty!";else{ list<object>::const_iteratoritr,itre; itr=v.begin(); itre=v.end(); do{cout<<*itr<<''; ++itr; }while(itr!=itre); }cout<<endl;}90第三章棧棧的概念棧的順序?qū)崿F(xiàn)棧的鏈接實現(xiàn)棧類的實現(xiàn)STL中的棧棧的應(yīng)用91棧后進(jìn)先出(LIFO,LastInFirstOut)或先進(jìn)后出(FILO,FirstInLastOut)結(jié)構(gòu),最先(晚)到達(dá)棧的結(jié)點將最晚(先)被刪除。

乒乓球進(jìn)盒、出盒92相關(guān)概念an-1an-2…a1a0…出棧進(jìn)棧棧底棧頂93相關(guān)概念棧底(bottom):結(jié)構(gòu)的首部(結(jié)點最早到達(dá)的部分)棧頂(top):結(jié)構(gòu)的尾部(結(jié)點最晚到達(dá)的部分)出棧(Pop):結(jié)點從棧頂刪除進(jìn)棧(Push):結(jié)點在棧頂位置插入空棧:棧中結(jié)點個數(shù)為零時94棧的運算創(chuàng)建一個棧create():創(chuàng)建一個空的棧;進(jìn)棧push(x):將x插入棧中,使之成為棧頂元素;出棧pop():刪除棧頂元素并返回棧頂元素值;讀棧頂元素top():返回棧頂元素值但不刪除棧頂元素;判??読sEmpty():若棧為空,返回true,否則返回false。95棧棧的概念棧的順序?qū)崿F(xiàn)棧的鏈接實現(xiàn)棧類的實現(xiàn)STL中的棧棧的應(yīng)用96棧的順序?qū)崿F(xiàn)用連續(xù)的空間存儲棧中的結(jié)點,即數(shù)組進(jìn)棧和出??偸窃跅m斠欢诉M(jìn)行,不會引起類似順序表中的大量數(shù)據(jù)的移動。用數(shù)組的后端表示棧頂。an…a1a0top_p棧底maxSizeelem97順序存儲時的運算實現(xiàn)create():按照用戶估計的棧的規(guī)模申請一個動態(tài)數(shù)組,將數(shù)組地址保存在data中,數(shù)組規(guī)模保存在maxSize中,并設(shè)top_p的值為-1。push(x):將top_p加1,將x放入top_p指出的位置中。但要注意數(shù)組滿的情況。pop():返回top_p指出的位置中的值并將top_p減1。top():與pop類似,只是不需要將top_p減1。isEmpty():若top_p的值為-1,返回true,否則返回false。98順序?qū)崿F(xiàn)性能分析除了進(jìn)棧操作以外,所有運算實現(xiàn)的時間復(fù)雜度都是O(1)。進(jìn)棧運算在最壞情況下的時間復(fù)雜度是O(N)。但最壞情況在N次進(jìn)棧操作中至多出現(xiàn)一次。如果把擴展數(shù)組規(guī)模所需的時間均攤到每個插入操作,每個插入只多了一個拷貝操作,因此從平均的意義上講,插入運算還是常量的時間復(fù)雜度。這種分析方法稱為均攤分析法。99棧棧的概念棧的順序?qū)崿F(xiàn)棧的鏈接實現(xiàn)棧類的實現(xiàn)STL中的棧棧的應(yīng)用100棧的鏈接實現(xiàn)棧的操作都是在棧頂進(jìn)行的,不需要找某一元素的直接前驅(qū)的操作,因此不需要雙鏈表,用單鏈表就足夠了,而且不需要頭結(jié)點對棧來講,只需要考慮棧頂元素的插入刪除。從棧的基本運算的實現(xiàn)方便性考慮,可將單鏈表的頭指針指向棧頂。101棧的鏈接實現(xiàn)an-1an-2a0…∧top_p102鏈接存儲時的運算實現(xiàn)create():將top_p設(shè)為空指針。push(x):將x插入到單鏈表的表頭voidpush(x){p=new結(jié)點;

p指向的結(jié)點的數(shù)據(jù)部分=x;

p指向的結(jié)點的指針部分=top_p;

top_p=p;}103鏈接存儲時的運算實現(xiàn)pop():刪除表頭元素dataTypepop(){p=top_p;top_p=top_p指向的結(jié)點的指針部分;

x=p指向的結(jié)點的數(shù)據(jù)部分;

deletep;returnx;}104鏈接存儲時的運算實現(xiàn)top():返回top_p指向的結(jié)點的值。isEmpty():若top_p的值為空指針,返回true,否則返回false。105鏈接棧的性能分析由于所有的操作都是對棧頂?shù)牟僮?,郵展中的元素個數(shù)無關(guān)。所以,所有運算的時間復(fù)雜度都是O(1)106棧棧的概念棧的順序?qū)崿F(xiàn)棧的鏈接實現(xiàn)棧類的實現(xiàn)STL中的棧棧的應(yīng)用107棧的抽象類template<classelemType>classstack{public: virtualboolisEmpty()const=0; virtualvoidpush(constelemType&x)=0; virtualelemTypepop()=0;virtualelemTypetop()const=0; virtual~stack(){}};108順序棧類template<classelemType>classseqStack:publicstack<elemType>{private:elemType*elem; inttop_p; intmaxSize; voiddoubleSpace();109public:seqStack(intinitSize=10){ elem=newelemType[initSize]; maxSize=initSize;top_p=-1;}

~seqStack(){delete[]elem;}boolisEmpty()const{returntop_p==-1;}110voidpush(constelemType&x){ if(top_p==maxSize-1)doubleSpace(); elem[++top_p]=x;}

elemTypepop(){returnelem[top_p--];}

elemTypetop()const{returnelem[top_p];} };111doubleSpacetemplate<classelemType>voidseqStack<elemType>::doubleSpace(){

elemType*tmp=elem;

elem=newelemType[2*maxSize]; for(inti=0;i<maxSize;++i) elem[i]=tmp[i];

maxSize*=2; delete[]tmp;}112鏈接棧類template<classelemType>classlinkStack:publicstack<elemType>

{private:structnode{elemTypedata;

node*next; node(constelemType&x,node*N=NULL) {data=x;next=N;} node():next(NULL){} ~node(){} };node*elem;113public:linkStack(){elem=NULL;}

~linkStack();boolisEmpty()const{returnelem==NULL;}voidpush(constelemType&x){node*tmp=newnode(x,elem);

elem=tmp;}elemTypepop(){node*tmp=elem;elemTypex=tmp->data; elem=elem->next; deletetmp;

returnx;}elemTypetop()const{returnelem->data; } };114析構(gòu)函數(shù)template<classelemType>linkStack<elemType>::~linkStack(){ node*tmp; while(elem!=NULL){ tmp=elem; elem=elem->next; deletetmp; }}115棧棧的概念棧的順序?qū)崿F(xiàn)棧的鏈接實現(xiàn)棧類的實現(xiàn)STL中的棧棧的應(yīng)用116STL中的棧棧本質(zhì)上是一個線性表,在STL中棧類是借助于線性表類實現(xiàn)的。STL中的棧提供四個標(biāo)準(zhǔn)運算:push、pop、top和empty。在STL中,借助于其他容器存儲數(shù)據(jù)的容器稱為容器適配器。棧就是一個容器適配器117棧的類模板定義一個棧對象要指明棧中元素的類型以及借助于哪一個容器,因此棧的類模板有兩個模板參數(shù):棧元素類型和所借助的容器類型。??梢越柚娜萜饔衯ector,list和deque。棧的類模板的第二個參數(shù)帶有缺省值deque118STL棧的使用#include<iostream>#include<stack>usingnamespacestd;intmain(){stack<int,vector<int>>s1;stack<int,list<int>>s2;inti;for(i=0;i<10;++i)s1.push(i);while(!s1.empty()){cout<<s1.top()<<'';s1.pop();}cout<<endl;for(i=0;i<10;++i)s2.push(i);while(!s2.empty()){cout<<s2.top()<<'';s2.pop();}cout<<endl;return0;}輸出:9876543210119棧棧的概念棧的順序?qū)崿F(xiàn)棧的鏈接實現(xiàn)棧類的實現(xiàn)STL中的棧棧的應(yīng)用120棧的應(yīng)用遞歸函數(shù)的非遞歸實現(xiàn)符號平衡檢查表達(dá)式的計算121棧的應(yīng)用—遞歸函數(shù)的非遞歸實現(xiàn)遞歸的本質(zhì)是函數(shù)調(diào)用,而函數(shù)調(diào)用是通過棧實現(xiàn)的。因此,遞歸可以用棧消除。voidf1(){…r1:f2();r2:…}voidf2(){…t1:f3();t2:…}voidf3(){……}122函數(shù)執(zhí)行過程f1f2f3r1t1r2t2123函數(shù)調(diào)用的實現(xiàn)設(shè)置一個棧模擬函數(shù)調(diào)用。當(dāng)發(fā)生函數(shù)調(diào)用時,將當(dāng)前的函數(shù)的現(xiàn)場進(jìn)棧。調(diào)用f2前Top調(diào)用f2后Topr2調(diào)用f3后Topr2t2返回f2后Topr2返回f1后Top124遞歸遞歸是一種特殊的函數(shù)調(diào)用,是在一個函數(shù)中又調(diào)用了函數(shù)本身。遞歸程序的本質(zhì)是函數(shù)調(diào)用,而函數(shù)調(diào)用是要花費額外的時間和空間。在系統(tǒng)內(nèi)部,函數(shù)調(diào)用是用棧來實現(xiàn),如果程序員可以自己控制這個棧,就可以消除遞歸調(diào)用。125快速排序voidquicksort(inta[],intlow,inthigh){intmid;if(low>=high)return;mid=divide(a,low,high);quicksort(a,low,mid-1);quicksort(a,mid+1,high);}126快速排序的非遞歸實現(xiàn)設(shè)置一個棧,記錄要做的工作,即要排序的數(shù)據(jù)段先將整個數(shù)組進(jìn)棧當(dāng)排序區(qū)間被分成兩半時,把一半進(jìn)棧,先排序另一半。排完后再去檢查棧是否為空,為空,則排序結(jié)束,否則從棧中彈出一個區(qū)間,排序該區(qū)間。棧元素的格式:

structnode{intleft;intright;};12794183650941836startfinish5240986134mid20startfinish986134finishstart986134986431mid10128voidquicksort(inta[],intsize){seqStack<node>st;intmid,start,finish;nodes;if(size<=1)return;//排序整個數(shù)組

s.left=0;s.right=size-1;st.push(s);129while(!st.isEmpty()){s=st.pop();start=s.left;finish=s.right;mid=divide(a,start,finish);if(mid-start>1){s.left=start;s.right=mid-1;st.push(s);}if(finish-mid>1){s.left=mid+1;s.right=finish;st.push(s);}}}130棧的應(yīng)用遞歸函數(shù)的非遞歸實現(xiàn)符號平衡檢查表達(dá)式的計算131括號配對檢查編譯程序的任務(wù)之一,就是檢查括號是否配對。如:括號(、[和{后面必須依次跟隨相應(yīng)的}、]及),“后面必須有”。簡單地用開括號和閉括號的數(shù)量是否相等來判斷開括號與閉括號是否配對是不行的。例如,符號串[()]是正確的,而符號串([)]是不正確的。因為當(dāng)遇到)那樣的閉括號時,它與最近遇到開括號匹配。132使用棧解決符號配對使用棧,遇到左括號進(jìn)棧,見到右括號,則出棧,并比較兩者是否配對。如判斷(3+(5-2)*(6+4)/2),(3+((5-2)*(6+4)/2)

“abc”,‘n’

(a[3]+a[4]–‘a(chǎn)’)133算法首先創(chuàng)建一個空棧。從源程序中讀入符號。如果讀入的符號是開符號,那末就將其進(jìn)棧。如果讀入的符號是一個閉符號但棧是空的,出錯。否則,將棧中的符號出棧。如果出棧的符號和和讀入的閉符號不匹配,出錯。繼續(xù)從文件中讀入下一個符號,非空則轉(zhuǎn)向3,否則執(zhí)行7。如果棧非空,報告出錯,否則括號配對成功。134細(xì)節(jié)問題如果對C++的源程序使用此算法,需要考慮三種括號:圓括號、方括號和花括號。但有時又不需要考慮圓括號、花括號、方括號是否匹配的問題。例如,當(dāng)括號出現(xiàn)在注釋、字符串常量、字符常量中時,就不需要考慮它的匹配問題。在C++中有很多轉(zhuǎn)義字符,因此在讀入一個字符時,還必須考慮如何識別轉(zhuǎn)義字符。135要求設(shè)計一個類Balance:對象初始化時傳給它一個源文件名。這個類提供一個成員函數(shù)checkBalance檢查源文件中的符號是否配對。輸出所有不匹配的符號及所在的行號136類的定義classbalance{private: ifstreamfin;//待檢查的文件流

intcurrentLine;//正在處理行的的行號

intErrors;//已發(fā)現(xiàn)的錯誤數(shù)

structSymbol{charToken;intTheLine;};enumCommentType{SlashSlash,SlashStar};public: balance(constchar*s); intCheckBalance();137private:

boolCheckMatch(charSymb1,charSymb2,intLine1,intLine2);charGetNextSymbol();voidPutBackChar(charch);voidSkipComment(enumCommentTypetype);voidSkipQuote(chartype);charNextChar();};classnoFile{};138構(gòu)造函數(shù)balance::balance(constchar*s){fin.open(s);if(!fin)thrownoFile();currentLine=1;Errors=0;}139CheckBalance的實現(xiàn)檢查輸入流對象中的括號是否匹配,并返回錯誤數(shù)算法的實現(xiàn)需要用到一個棧,我們可以用本章中實現(xiàn)的棧類,如seqStack。采用逐步精細(xì)化的方法分解這個函數(shù)140自頂向下的分解初始化棧為空;While(lastChar=讀文件,直到讀入一括號)Switch(lastChar){case‘{‘,‘[‘,‘(‘:進(jìn)棧

case‘}’,’]’,’)’:

if(??眨┹敵瞿承心撤柌黄ヅ洌怀鲥e數(shù)加1;else{match=出棧的符號;檢查lastChar與match是否匹配;如不匹配,輸出出錯信息,出錯數(shù)加1;

}}if(棧非空)棧中元素均沒有找到匹配的閉符號,輸出這些錯誤

return出錯數(shù)141進(jìn)一步需要細(xì)化讀文件,直到讀入一括號;輸出某行某符號不匹配;出錯數(shù)加1;檢查lastChar與match是否匹配。如不匹配,輸出出錯信息,出錯數(shù)加1;棧中元素均沒有找到匹配的閉符號,輸出這些錯誤142進(jìn)一步抽取子函數(shù)第1項工作:GetNextSymbol功能:從文件的當(dāng)前位置開始,跳過所有的非括號,讀到第一個括號后返回。在遇到文件結(jié)束時,返回一個特定的符號,如NULL。函數(shù)原型:函數(shù)有一個字符型的返回值。執(zhí)行該函數(shù)必須有一個輸入流,在讀輸入流的過程中,當(dāng)前處理的行號會變,在讀的過程中也可能遇到異常情況,因此出錯數(shù)也會變。這三個信息:文件流對象,當(dāng)前處理的行號,出錯個數(shù)都是對象的數(shù)據(jù)成員。因此函數(shù)原型為:charGetNextSymbol()。第3項工作:CheckMatch功能:比較兩個指定位置的待比較的符號是否匹配函數(shù)原型:boolbalance::CheckMatch(charSymb1,charSymb2,intLine1,intLine2)143CheckBalance的定義intbalance::CheckBalance(){structSymbolnode;seqStack<Symbol>st;charLastChar,Match;//LastChar為讀入的符號,Match為棧頂?shù)姆朇heckBalance函數(shù)不需要參數(shù)。因為輸入流對象是類的數(shù)據(jù)成員。返回值是一個整型數(shù),表示出錯個數(shù)144while(LastChar=GetNextSymbol()){switch(LastChar){ case'(':case'[':case'{':node.Token=LastChar;node.TheLine=currentLine;st.push(node);break; case')':case']':case'}': if(st.isEmpty()){Errors++; cout<<"在第"<<currentLine<<"有一個多余的"<<LastChar<<endl; }else{node=st.pop();Match=node.Token; if(!CheckMatch(Match,LastChar,node.TheLine,currentLine))++Errors; } break;} }145while(!st.isEmpty()){ Errors++; node=st.pop(); cout<<"第"<<node.TheLine<<"行的符號"<<node.Token<<"不匹配\n"; }returnErrors;}146CheckMatchboolbalance::CheckMatch(charSymb1,charSymb2,intLine1,intLine2){if(Symb1=='('&&Symb2!=')'||Symb1=='['&&Symb2!=']'||Symb1=='{'&&Symb2!='}'){ cout<<"發(fā)現(xiàn)第"<<Line2<<"的符號"<<Symb2<<"與第"<<Line1<<"的符號"<<Symb1<<"不匹配\n"; returnfalse; }elsereturntrue;}147GetNextSymbol在讀文件時要跳過其它符號,提取出各類括號注釋中的括號不用考慮,字符串常量和字符常量中的括號也不用考慮。C++中的注釋又有兩種形式。一種是以“//”開始到本行結(jié)束。另一種是以“/*”開始到“*/”結(jié)束,可以跨行。不管是哪一種注釋,都要判斷兩個符號才能決定148GetNextSymbol的偽代碼CharGetNextSymbol(){while(ch=從文件中讀入下一字符){ if(Ch==‘/’){//可能是注釋的開始

if(ch=從文件中讀入下一字符) if(Ch==‘*’)跳過標(biāo)準(zhǔn)C的注釋;elseif(Ch==‘/’)跳過C++的注釋; else不是注釋,把ch放回輸入流; }elseif(Ch==‘\’’||Ch==‘”’)跳過字符常量或字符串常量;elseif(Ch==‘{’||Ch==’[’||Ch==‘(’||Ch==’)’||Ch==‘]’||Ch==’}’)returnCh;}return0;//文件結(jié)束。}149進(jìn)一步提取子函數(shù)從文件中讀入下一字符:

charNextChar();跳過的注釋:voidSkipComment(enumCommentTypetype);把ch放回輸入流:

voidPutBackChar(charch);跳過字符常量或字符串常量:

voidSkipQuote(chartype);150GetNextSymbol函數(shù)的實現(xiàn)charbalance::GetNextSymbol(){charch;while(ch=NextChar()){ if(ch=='/'){ ch=NextChar(); if(ch=='*')SkipComment(SlashStar);elseif(ch=='/')SkipComment(SlashSlash); elsePutBackChar(ch); } elseif(ch=='\''||ch=='"')SkipQuote(ch); elseif(ch=='{'||ch=='['||ch=='('||ch==')'||ch==']'||ch=='}')returnch; } returnNULL;//文件結(jié)束。}151NextChar函數(shù)的實現(xiàn)NextChar函數(shù)從輸入文件流中讀入一個字符,返回給調(diào)用程序,遇到文件結(jié)束符,返回NULL。如遇到換行符,則當(dāng)前處理行加1。charbalance::NextChar(){charch;if((ch=fin.get())==EOF)returnNULL;if(ch=='\n')currentLine++;returnch;}152PutBackChar函數(shù)的實現(xiàn)PutBackChar函數(shù)將傳入的字符放回輸入流,這是通過調(diào)用輸入流對象的成員函數(shù)putback實現(xiàn)的。如放回的字符是回車,當(dāng)前處理行號減1。voidbalance::PutBackChar(charch){fin.putback(ch);if(ch=='\n')currentLine--;}153SkipQuote函數(shù)的實現(xiàn)讀文件,直到讀到一個和參數(shù)值相同的符號。要注意幾個特殊情況:字符或字符串常量是不允許跨行的。也就是說,如果在讀文件的過程中遇到了回車,則表示字符或字符串常量的開始和結(jié)束符不匹配,輸出出錯信息。在字符或字符串常量中可能會包含單引號或雙引號,此時不能將這個單引號或雙引號作為結(jié)束符。如何知道這個單引號或雙引號代表的是普通的字符而不是結(jié)束符呢?C++采用了轉(zhuǎn)義字符來表示,因此當(dāng)讀到一個表示轉(zhuǎn)義字符開始的標(biāo)記(\)時,不管它后面是什么符號,都不用檢查。

溫馨提示

  • 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

提交評論