版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進(jìn)行舉報或認(rèn)領(lǐng)
文檔簡介
數(shù)據(jù)結(jié)構(gòu)DATASTRUCTURE——Java描述1.1數(shù)據(jù)結(jié)構(gòu)的地位1.2基本概念和術(shù)語
1.2.1基本概念1.數(shù)據(jù)(data):是對客觀事物的符號表示,所有能輸入到計算機(jī)中,被計算機(jī)程序識別和處理的符號的總稱。2.數(shù)據(jù)元素(dataelement):數(shù)據(jù)元素是表示現(xiàn)實(shí)世界中一個實(shí)體的一組數(shù)據(jù),是數(shù)據(jù)結(jié)構(gòu)的基本組成單位。一個數(shù)據(jù)元素可由一個或若干個數(shù)據(jù)項(xiàng)(dataItem)組成。3.數(shù)據(jù)對象(DataObject):是性質(zhì)相同的數(shù)據(jù)元素的集合。4.數(shù)據(jù)結(jié)構(gòu)(Data_Structure)相互之間存在一種或多種特定關(guān)系的數(shù)據(jù)元素的集合。1.2.2數(shù)據(jù)結(jié)構(gòu)的種類1、集合結(jié)構(gòu)。在集合結(jié)構(gòu)中,數(shù)據(jù)元素間的關(guān)系除了“屬于同一個集合”外,別無任何關(guān)系。
1.2.2數(shù)據(jù)結(jié)構(gòu)的種類2、線性結(jié)構(gòu)。該結(jié)構(gòu)的數(shù)據(jù)元素之間存在著一對一的關(guān)系。
1.2.2數(shù)據(jù)結(jié)構(gòu)的種類3、樹型結(jié)構(gòu)。該結(jié)構(gòu)的數(shù)據(jù)元素之間存在著一對多的關(guān)系。1.2.2數(shù)據(jù)結(jié)構(gòu)的種類4、圖形結(jié)構(gòu)。該結(jié)構(gòu)的數(shù)據(jù)元素之間存在著多對多的關(guān)系,圖形結(jié)構(gòu)也稱作網(wǎng)狀結(jié)構(gòu)。1.2.3數(shù)據(jù)結(jié)構(gòu)的數(shù)學(xué)定義數(shù)據(jù)結(jié)構(gòu)的數(shù)學(xué)定義為:數(shù)據(jù)結(jié)構(gòu)是一個二元組:
Data-Structure=(D,S)其中:D是數(shù)據(jù)元素的集合,S是數(shù)據(jù)元素之間關(guān)系的集合。例:復(fù)數(shù)a+bi的數(shù)據(jù)結(jié)構(gòu)定義如下:
Complex=(C,R)其中,D部分用C表示,C是含兩個實(shí)數(shù)的集合﹛a,b﹜;S部分用R表示,R={〈a,b〉},〈a,b〉表示a和b存在一種有序關(guān)系,a表示復(fù)數(shù)的實(shí)部,b表示復(fù)數(shù)的虛部,二者不能互換。1.2.4數(shù)據(jù)的存儲結(jié)構(gòu)
本課程是在高級語言Java的層次上討論數(shù)據(jù)結(jié)構(gòu)在計算機(jī)中內(nèi)存中的存儲,主要研究的是數(shù)據(jù)元素以及它們之間的關(guān)系在Java虛擬處理器中的表示,可以稱為虛擬存儲結(jié)構(gòu)。1、數(shù)據(jù)元素的表示單個數(shù)據(jù)項(xiàng)組成的數(shù)據(jù)元素可以利用java語言固有的基本數(shù)據(jù)類型(8種)來表示。
例如:復(fù)數(shù)數(shù)據(jù)結(jié)構(gòu)Complex=(C,R)中的C=﹛a,b﹜,數(shù)據(jù)元素a和b是實(shí)數(shù),可用Java語言中的固有的雙精度浮點(diǎn)型double表示。多個數(shù)據(jù)項(xiàng)組成的數(shù)據(jù)元素可以利用Java語言中的自己定義的類表示。例如:學(xué)生實(shí)體的數(shù)據(jù)元素可由學(xué)號、姓名、所在系、性別、出生日期、入學(xué)成績等數(shù)據(jù)項(xiàng)組成,在Java語言中用自己定義的類表示如下:classStudent{
StringNO;//String位于java.lang子類庫中
StringName;
StringDepartment;
charSex;
DateBirthDate;//Date位于java.util子類庫中
doubleScore;}1.2.4數(shù)據(jù)的存儲結(jié)構(gòu)2數(shù)據(jù)元素之間關(guān)系的表示數(shù)據(jù)元素之間的關(guān)系在計算機(jī)內(nèi)存中有兩種不同的表示方法:順序存儲和鏈?zhǔn)酱鎯?。順序存儲是把?shù)據(jù)元素按一定的規(guī)則存放在一組編號連續(xù)的存儲單元中,通過元素在內(nèi)存中的存儲位置來體現(xiàn)數(shù)據(jù)元素之間的邏輯關(guān)系,邏輯上相鄰的數(shù)據(jù)元素在內(nèi)存中的存儲位置也相鄰。abcd
例如:線性數(shù)據(jù)結(jié)構(gòu)(‘a(chǎn)’,‘b’,‘c’,‘d’)的順序存儲就是把數(shù)據(jù)元素‘a(chǎn)’、‘b’、‘c’、‘d’放在一組編號連續(xù)的存儲單元(1個存儲單元即1個字節(jié))中,通過元素的存儲地址是否相鄰來表示數(shù)據(jù)元素在邏輯上是否相鄰。存儲單元編號存儲內(nèi)容
…0030a0032b0034c0036d
…在高級語言Java的層次上討論時,線性數(shù)據(jù)結(jié)構(gòu)(‘a(chǎn)’,‘b’,‘c’,‘d’)的數(shù)據(jù)元素是char型,每個數(shù)據(jù)元素在占用2個字節(jié)的內(nèi)存空間。。在Java語言中,存儲單元的編號對用戶透明,在Java虛擬處理器中線性數(shù)據(jù)結(jié)構(gòu)的順序存儲就是線性數(shù)據(jù)結(jié)構(gòu)中的數(shù)據(jù)元素依次存放到一維數(shù)組中,因?yàn)镴ava語言中一維數(shù)組占用的就是一組編號連續(xù)的內(nèi)存單元。1.2.4數(shù)據(jù)的存儲結(jié)構(gòu)2數(shù)據(jù)元素之間關(guān)系的表示鏈?zhǔn)酱鎯Γ烘準(zhǔn)酱鎯r數(shù)據(jù)元素不必存放在一組編號連續(xù)的存儲單元中,把數(shù)據(jù)元素后附加一個地址域,通過指示相鄰元素的存儲地址來體現(xiàn)數(shù)據(jù)元素的邏輯關(guān)系,邏輯上相鄰的數(shù)據(jù)元素物理存儲位置不一定相鄰。線性數(shù)據(jù)結(jié)構(gòu)(‘a(chǎn)’,‘b’,‘c’,‘d’)的鏈?zhǔn)酱鎯κ窃诎褦?shù)據(jù)元素后面加一個地址域,通過指示相鄰元素的存儲地址來體現(xiàn)數(shù)據(jù)元素的邏輯關(guān)系,邏輯上相鄰的數(shù)據(jù)元素物理存儲位置上不一定相鄰。abcd
存儲單元編號存儲內(nèi)容
…0100b01020120……0120c01220160……0144a01460100……0160d0162NULL
…h(huán)eadd^cba前面已經(jīng)提及,在Java語言中,存儲單元的編號對用戶透明,在Java虛擬處理器中,把相鄰元素的存儲地址抽象為引用,在存儲結(jié)構(gòu)圖中用箭頭表示,整個存儲結(jié)構(gòu)像一條鏈條,鏈?zhǔn)酱鎯Y(jié)構(gòu)也是由此命名。1.2.5抽象數(shù)據(jù)類型抽象數(shù)據(jù)類型:數(shù)據(jù)的邏輯結(jié)構(gòu)及定義在該邏輯結(jié)構(gòu)上的一組操作。
ADT={D,S,P}為什麼還要研究操作呢?因?yàn)獒槍Σ煌臄?shù)據(jù)元素集合,定義于其上的操作也不一樣。以Java語言中的int,boolean數(shù)據(jù)類型為例:int數(shù)據(jù)類型,其數(shù)據(jù)元素集是-231~231-1區(qū)間的整數(shù),定義在其上的操作是為加、減、乘、除和取模等算術(shù)運(yùn)算和大于、不等于、等于、小于等關(guān)系運(yùn)算。boolean數(shù)據(jù)類型,其數(shù)據(jù)元素集是{true,false},定義在其上的操作是與、或、非等邏輯運(yùn)算。在后面的章節(jié)中,每研究一種數(shù)據(jù)結(jié)構(gòu),都是從抽象數(shù)據(jù)類型開始,即先研究邏輯結(jié)構(gòu)及定義于其上的操作,邏輯結(jié)構(gòu)嚴(yán)格來說屬于離散數(shù)學(xué)課程的學(xué)習(xí)內(nèi)容。1.2.5抽象數(shù)據(jù)類型對于使用數(shù)據(jù)類型的用戶而言,引入“數(shù)據(jù)類型”,實(shí)現(xiàn)了信息的隱蔽,即將一切用戶不必了解的細(xì)節(jié)都封裝在類型中。例如,Java編程用戶使用int數(shù)據(jù)類型時,既不需要了解int型整數(shù)在計算機(jī)的內(nèi)部是如何表示的,也不需要知道其操作是如何實(shí)現(xiàn)的,這些由Java語言的推出者關(guān)注實(shí)現(xiàn)。就int數(shù)據(jù)類型的“加”操作而言,Java編程用戶只需關(guān)注“加”的功能及它的數(shù)學(xué)意義,不必去關(guān)注其硬件的“位”操作如何進(jìn)行。
抽象數(shù)據(jù)類型和數(shù)據(jù)類型實(shí)質(zhì)上是一個概念,抽象的意義在于數(shù)據(jù)類型的數(shù)學(xué)抽象特性。另外抽象數(shù)據(jù)類型的范疇更廣,它不再局限于各高級語言處理器中已定義并實(shí)現(xiàn)的數(shù)據(jù)類型,還包括用戶開發(fā)軟件系統(tǒng)時自己定義的數(shù)據(jù)類型。
為了提高軟件的復(fù)用率,在近代程序設(shè)計方法學(xué)中指出,一個軟件系統(tǒng)的框架應(yīng)建立在數(shù)據(jù)之上,而不是建立在操作之上。在構(gòu)成軟件系統(tǒng)的每個相對獨(dú)立的模塊上,定義一組數(shù)據(jù)和施于這組數(shù)據(jù)上的一組操作,并在模塊內(nèi)部給出這些數(shù)據(jù)的表示及操作的細(xì)節(jié),而在模塊外部使用的只是抽象的操作。顯然,所定義的數(shù)據(jù)類型的抽象層次越高,含有該數(shù)據(jù)類型的軟件模塊的復(fù)用度也越高。1.3數(shù)學(xué)預(yù)備知識1.3.1集合1、集合的概念
集合是由一些確定的、彼此不同的成員或者元素構(gòu)成的一個整體。成員的個數(shù)稱為集合的基數(shù)。如果某集合的成員都屬于該集合,那麼某集合叫做該集合的子集
沒有元素的集合稱為空集,記作Φ。3是R的成員,記為:3∈R,6不是R的成員,記為:6?R。{3,4}是R的子集。2、集合的表示法
窮舉法:S={2,4,6,8,10};
描述法:S={x|x是偶數(shù),且0≤x≤10}。3、集合的特性確定性:任何一個對象都能被確切地判斷是集合中的元素或不是;互異性:集合中的元素不能重復(fù);無序性:集合中元素與順序無關(guān)。
1.3.2常用的數(shù)學(xué)術(shù)語計量單位(Unit):字節(jié)縮寫為“B”,位縮寫為“b”,千字節(jié)縮寫為“KB”,百萬字節(jié)縮寫為縮寫為“MB”,十億字節(jié)縮寫為縮寫為“GB”,萬億字節(jié)縮寫為“TB”,。階乘函數(shù)(FactorialFunction):階乘函數(shù)n!是指從1到n之間所有整數(shù)的連乘,其中n為大于0的整數(shù)。因此,5!=1?2?3?4?5=120。特別地,0!=1。取下整和取上整(FloorandCeiling):實(shí)數(shù)x的取下整函數(shù)(Floor),返回不超過x的最大整數(shù),例如Floor(5.5)=5;實(shí)數(shù)x的取上整函數(shù)(Ceiling),返回不小于x的最小整數(shù),Ceiling(
5.5)=6
。取模操作符(Modulus):取模函數(shù)返回整除后的余數(shù),有時稱為求余,記為%,例如:5%3=2。1.3.3對數(shù)一般地,如果a(a>0,a≠1)的b次冪等于N,就是ab=N,那么數(shù)b叫做以a為底N的對數(shù),記作logaN=b,其中a叫做對數(shù)的底數(shù),N叫做真數(shù)。
負(fù)數(shù)和零沒有對數(shù)。1.3.4遞歸如果一個算法直接調(diào)用自己或間接地調(diào)用自己,就稱這個算法是遞歸的(Recursive)。根據(jù)調(diào)用方式的不同,它分為直接遞歸和間接遞歸。直接遞歸調(diào)用就是在函數(shù)a(或過程)中直接引用(調(diào)用)函數(shù)a本身。間接遞歸調(diào)用就是在函數(shù)a(或過程)中調(diào)用另外一個函數(shù)b,而該函數(shù)b又引用(調(diào)用)了函數(shù)a。
1.3.4遞歸一個遞歸算法必須有兩個部分:初始部分(Basecase)和遞歸部分。初始情況只處理可以直接解決而不需要再次遞歸調(diào)用的簡單輸入。遞歸部分包含對算法的一次或多次遞歸調(diào)用,每一次的調(diào)用參數(shù)都在某種程度上比原始調(diào)用參數(shù)更接近初始情況。
函數(shù)的遞歸調(diào)用可以理解為:通過一系列的自身調(diào)用,達(dá)到某一終止條件后,在按照調(diào)用路線逐步返回。遞歸是程序設(shè)計中強(qiáng)有力的工具,有很多數(shù)學(xué)函數(shù)時遞歸定義的。階乘函數(shù),對n!作如下定義:
階乘函數(shù)的java語言實(shí)現(xiàn)如下:
public
static
double
fun(intn){
if(n=0)
return1;
else
return
fun(n-1)*n;}1.4算法和算法分析1.4.1算法1、定義算法(Algorithm)是對特定問題求解步驟的一種描述,是指令的有限序列。其中每一條指令表示一個或多個操作。算法的描述工具有自然語言、框圖、PAD圖、偽碼、程序設(shè)計語言等等。在Java程序設(shè)計語言中,算法一般用類的一個方法進(jìn)行描述,方法體內(nèi)的一條條語句就是對問題求解步驟的一種描述。
1.4算法和算法分析1.4.1算法2、特性輸入
有輸入輸出有輸出(處理結(jié)果)確定性每步定義都是確切、無歧義的有窮性算法應(yīng)在執(zhí)行有窮步后結(jié)束有效性每一條運(yùn)算能有效的執(zhí)行
(1)有輸入。算法有一個或多個輸入數(shù)據(jù),輸入數(shù)據(jù)是算法的加工對象,是數(shù)據(jù)原材料。
在Java程序設(shè)計語言中,算法表現(xiàn)為類的方法,算法的輸入可以通過方法的形式參數(shù)接受,也可以在方法體內(nèi)由變量賦值語句指定,還可以在方法體內(nèi)通過輸入語句從鍵盤、外存等外部設(shè)備接受,算法的輸入數(shù)據(jù)亦可由方法所在類的成員變量的值提供。
例如求階乘的算法有一個輸入數(shù)據(jù),通過方法的形式參數(shù)接受,如下所示:
staticdoublefact(intn){ doubles=1; for(inti=1;i<=n;i++){ s=s*i; } returns;
}算法也可以有兩個或多個輸入數(shù)據(jù),例如求最大公約數(shù)的算法有兩個輸入數(shù)據(jù),需要輸入兩個整數(shù)的值。
求兩個非負(fù)整數(shù)數(shù)a和b(要求a>b)的最大公約數(shù)可以使用輾轉(zhuǎn)相除法,算法用自然語言描述為:
①a除以b得到的余數(shù)為c(0<=c<b);
②若c=0則算法結(jié)束,b為最大公約數(shù)。否則轉(zhuǎn)③;
③a=b,b=c,轉(zhuǎn)①;
static
int
gys1(int
a,int
b)//最大公約數(shù)
{
int
temp=0;
if(a<b){temp=a;a=b;b=temp;}
int
c=a%b;
while(c!=0){
a=b;
b=c;
c=a%b;}
return
b;}
輾轉(zhuǎn)相除法算法用Java語言描述如下,算法的兩個輸入數(shù)據(jù)均通過方法的形式參數(shù)接受:在Java語言中,輾轉(zhuǎn)相除算法的兩個輸入數(shù)據(jù)也可以在方法體內(nèi)通過輸入語句從外設(shè)鍵盤接受,如下所示:static
int
gys2()//最大公約數(shù)
{Scannerreader=newScanner(System.in);System.out.print("請輸入第一個數(shù):");inta=reader.nextInt();System.out.print("請輸入第二個數(shù):");intb=reader.nextInt();
int
temp=0;
if(a<b){temp=a;a=b;b=temp;}
int
c=a%b;
while(c!=0){
a=b;
b=c;
c=a%b;}
return
b;}
(2)有輸出。
算法有一個或多個輸出數(shù)據(jù),輸出數(shù)據(jù)是一組與輸入有確定關(guān)系的值,是算法對輸入數(shù)據(jù)進(jìn)行加工后得到的結(jié)果數(shù)據(jù),這種確定關(guān)系即為算法的功能。上面例子中,階乘、最大公約數(shù)就是算法的輸出。
在Java程序設(shè)計語言中,算法的輸出一般通過方法體內(nèi)的return語句返回,如上面兩個例子;還可以把結(jié)果數(shù)據(jù)輸出到屏幕、打印機(jī)、外存等外部設(shè)備,也可以傳到網(wǎng)絡(luò),還可以表現(xiàn)為方法所在類中成員變量值的改變。例如輾轉(zhuǎn)相除法求最大公約數(shù)可以把結(jié)果數(shù)據(jù)輸出到屏幕,如下所示:staticvoidgys3()//最大公約數(shù)
{Scannerreader=newScanner(System.in);System.out.print("請輸入第一個數(shù):");inta=reader.nextInt();System.out.print("請輸入第二個數(shù):");intb=reader.nextInt();
int
temp=0;
if(a<b){temp=a;a=b;b=temp;}
int
c=a%b;
while(c!=0){
a=b;
b=c;
c=a%b;}
System.out.print("最大公約數(shù)為:"+b);}
(3)確定性。每步定義都是確切、無歧義的,而不是含糊的,模棱兩可的。并且在任何條件下,算法只有惟一的一條執(zhí)行路徑,即對于相同的輸入只能得到相同的輸出。
在Java程序設(shè)計語言中,算法的操作步驟通過語句描述,所寫語句必須符合語法規(guī)定,不符合確定性要求的語句編譯不會通過,算法的確定性檢驗(yàn)主要由編譯器協(xié)助完成。(4)有窮性。對于任意一組合法的輸入值,算法在執(zhí)行有窮步驟后一定能結(jié)束。即算法的操作步驟為有限個,且每步都能在有限時間內(nèi)完成。
在Java程序設(shè)計語言中,死循環(huán)的算法是不符合設(shè)計要求的。事實(shí)上,“有窮性”是指合理的范圍內(nèi),計算機(jī)執(zhí)行100年才結(jié)束的算法,雖然是有窮的,但超過了合理的限度,也是不可行的。(5)有效性。算法每一步能有效的執(zhí)行,并能得到確定的結(jié)果。例如,在除法中,除數(shù)不為0才能被有效執(zhí)行,除數(shù)為0是不能被有效執(zhí)行的。
在Java程序設(shè)計語言中,不能被有效執(zhí)行的語句會中止執(zhí)行,并產(chǎn)生異常對象。1.4.2算法設(shè)計的要求
對于一個特定的問題,采用的數(shù)據(jù)結(jié)構(gòu)不同,其設(shè)計的算法一般也不同,即使在同一種數(shù)據(jù)結(jié)構(gòu)下,也可以采用不同的算法。那么,解決實(shí)際問題時,選擇哪一種算法比較合適,以及如何對現(xiàn)有的算法進(jìn)行改進(jìn),從而設(shè)計出更優(yōu)質(zhì)的算法,這就是算法設(shè)計要求的問題。評價一個算法優(yōu)劣的主要標(biāo)準(zhǔn)如下:
正確性可讀性健壯性時間效率與空間效率
1、正確性(correctness)
算法的正確性指的是對合法正確的輸入能夠獲取正確的輸出。比如求階乘的算法,對于合法正確的輸入數(shù)據(jù)4,算法能夠輸出正確的結(jié)果24。
對于程序設(shè)計語言描述的算法,“正確”大體分為以下四個層次:
(1)不含語法錯誤,編譯通過。
(2)幾組輸入數(shù)據(jù)能輸出正確的結(jié)果數(shù)據(jù)。
(3)精心選擇的幾組典型、苛刻、刁難性數(shù)據(jù)能得出正確的結(jié)果數(shù)據(jù)。
(4)一切合法輸入均能輸出正確的結(jié)果數(shù)據(jù)。
顯然,達(dá)到第四層次意義下的正確是極為困難的,一切合法輸入數(shù)據(jù)量大得驚人,逐一測試的方法是不現(xiàn)實(shí)的。對大型軟件需要進(jìn)行專業(yè)測試,一般情況下,通常以第三層次意義的正確性作為軟件驗(yàn)收標(biāo)準(zhǔn)1.4.2算法設(shè)計的要求
2、可讀性(readbility)算法的可讀性指的是算法的表達(dá)思路清晰,簡潔明了,易于理解。對于程序設(shè)計語言描述的算法,為了提高可讀性,通常做法加注釋。1.4.2算法設(shè)計的要求3、健壯性(robustness)又稱魯棒性,是robust音譯,也稱容錯性。當(dāng)用戶輸入不合法時,例如用戶輸入錯誤時,能夠反饋給用戶相應(yīng)的提示信息,而不是終止程序的執(zhí)行,或顯示調(diào)試時出現(xiàn)的一些錯誤信息。1.4.2算法設(shè)計的要求4、時間效率與空間效率要求算法執(zhí)行時間越短,算法的時間效率越高,算法執(zhí)行時占用的內(nèi)存空間越少,算法的空間效率越高。下面兩小節(jié)就要來講述怎樣度量算法的效率。1.4.2算法設(shè)計的要求1.4.3算法時間效率的度量1、算法的執(zhí)行時間算法時間效率可以利用算法執(zhí)行時間進(jìn)行度量,算法執(zhí)行時間就是依據(jù)該算法編制的程序在計算機(jī)上運(yùn)行所消耗的時間。兩個缺陷:
1)必須首先運(yùn)行依據(jù)算法編制的程序
2)所得時間的統(tǒng)計量依賴于計算機(jī)硬件、軟件的因素,容易掩蓋算法本身的優(yōu)劣。分析算法時間效率時,必須撇開這些與算法無關(guān)的軟硬件因素。通常的做法是:從算法中選取一種對于所研究的問題來說是基本操作的原操作,以該基本操作重復(fù)執(zhí)行的次數(shù)作為算法的時間度量。稱之為:2、漸進(jìn)時間復(fù)雜度
被稱作問題的基本操作的原操作多數(shù)情況下是最深層循環(huán)內(nèi)語句的原操作,它的執(zhí)行次數(shù)和包含它的語句頻度相同,語句的頻度指的是該語句重復(fù)執(zhí)行的次數(shù)。
例題1求階乘的算法把乘法操作看作基本操作,乘法操作包含在下面畫橫線的語句中,基本操作重復(fù)執(zhí)行了多少次?staticdoublefact(intn){ doubles=1; for(inti=1;i<=n;i++){
s=s*i; } returns;}
乘法操作執(zhí)行了n次。該算法的時間復(fù)雜度記為O(n)基本操作執(zhí)行次數(shù)與n成正比的即an+b均記為O(n)基本操作執(zhí)行次數(shù)與n2成正比的即cn2+an+b均記為O(n2)(重點(diǎn)關(guān)注增長最快的項(xiàng))在一些常用的項(xiàng)中到底哪個項(xiàng)增長快呢?有下面的不等式成立:c<log2n<n<nlog2n<n2<n3<2n<3n<n!例題2:求n行n列矩陣所有元素之和的算法可以把加法操作看作基本操作,加法操作包含在下面劃橫線的語句中,請分析該算法的時間復(fù)雜度。
基本操作執(zhí)行n2+n次,該算法的時間復(fù)雜度記為O(n2)
staticfloatexample(float[][]x,intn){ float[]sum=newfloat[n]; floattotalsum=0.0f; for(inti=0;i<n;i++){//x[][]中各行數(shù)據(jù)累加和
for(intj=0;j<n;j++)
sum[i]=sum[i]+x[i][j]; } for(inti=0;i<n;i++)//各行數(shù)據(jù)匯總相加
totalsum=totalsum+sum[i]; returntotalsum;
}有的情況下,算法中基本操作重復(fù)執(zhí)行的次數(shù)還隨問題的輸入數(shù)據(jù)集不同而不同。比如冒泡排序法,此算法的功能是對輸入數(shù)據(jù)序列進(jìn)行排序,算法執(zhí)行完畢后,序列中數(shù)據(jù)元素依次遞增。排序過程如下所示:冒泡排序的過程:假設(shè)輸入數(shù)據(jù)序列中元素個數(shù)是n,首先比較第一個和第二個數(shù)據(jù),將其中較小的數(shù)據(jù)放到第一個位置,較大的放到第二個位置;然后比較第二個和第三個數(shù)據(jù),仍將較大的放到后一個位置。依此類推,直到比較第n-1和第n個數(shù)據(jù)。這樣,就將待排序序列中的最大的一個放到了第n個位置(一趟排序冒出一個泡來),這個過程稱為第一趟排序。對前n-1個數(shù)據(jù)重復(fù)這個過程(不用考慮第n個位置數(shù)據(jù),因?yàn)樗呀?jīng)是最大的了),又將次大的數(shù)據(jù)放到了第n-1個位置。第三趟排序選出第三大的數(shù)據(jù),放到倒數(shù)第三個位置。重復(fù)這個過程,直到數(shù)據(jù)依次遞增為止(共n-1趟)。[初態(tài)]:4682405267312173對此輸入數(shù)據(jù)序列進(jìn)行冒泡排序:冒泡排序法示例:[初態(tài)]:4682405267312173第1趟:4640526731217382第2趟:4046523121677382第3趟:4046312152677382第4趟:4031214652677382第5趟:3121404652677382第6趟:2131404652677382第7趟:2131404652677382java語言源碼
static
voidbubble_sort(int[]a,int
n) {
for(int
i=n-1;i>0;i--)
for(int
j=0;j<i;++j)
if(a[j]>a[j+1])
{int
temp=a[j];a[j]=a[j+1];a[j+1]=temp;} }冒泡排序法中把數(shù)據(jù)交換操作看作基本操作,數(shù)據(jù)交換操作包含在上面畫橫線的語句中,設(shè)輸入數(shù)據(jù)序列中數(shù)據(jù)元素個數(shù)是n,則冒泡排序法中基本操作重復(fù)執(zhí)行的次數(shù)隨輸入數(shù)據(jù)狀態(tài)的不同而不同。最壞的情況:當(dāng)初始狀態(tài)完全逆序,即數(shù)據(jù)元素遞減有序,需要進(jìn)行多少次數(shù)據(jù)交換?(此例中數(shù)據(jù)元素個數(shù)n=8)
[初態(tài)]:82736752464031
21第1趟:736752464031
21
82
n-1第2趟:6752464031
2173
82n-2第3趟:5246403121677382n-3第4趟:4640312152677382.第5趟:4031214652677382.第6趟:31214046526773822第7趟:21314046526773821第1趟需要進(jìn)行n-1次數(shù)據(jù)交換,第2趟需要進(jìn)行n-2次數(shù)據(jù)交換,依此類推...,第n-1要進(jìn)行需要進(jìn)行1次數(shù)據(jù)交換,數(shù)據(jù)交換操作重復(fù)進(jìn)行的總次數(shù)=(n-1)+(n-2)+…+2+1=n(n-1)/2。
[初態(tài)]:2131404652677382次數(shù)第1趟:21314046526773820第2趟:21314046526773820第3趟:21314046526773820第4趟:21314046526773820第5趟:21314046526773820第6趟:21314046526773820第7趟:21314046526773820最好的情況:當(dāng)初始狀態(tài)已經(jīng)遞增有序時,第1趟不需要進(jìn)行數(shù)據(jù)交換,第2趟不需要進(jìn)行數(shù)據(jù)交換,依此類推...,第n-1趟也不需要進(jìn)行次數(shù)據(jù)交換,如下所示:(此例中數(shù)據(jù)元素個數(shù)n=8)
計算時間復(fù)雜度以最壞情況為準(zhǔn),為O(n2)。當(dāng)某一趟排序沒有數(shù)據(jù)交換時,表明數(shù)據(jù)元素已經(jīng)按遞增有序,接下來的排序過程可以不必進(jìn)行,但此算法依就會進(jìn)行完n-1排序過程才終止。怎樣解決此問題?具體做法是:設(shè)置一個變量change=true,用它來記錄一趟排序過程中是否有數(shù)據(jù)交換。在每趟排序之前置change的值為false,每當(dāng)產(chǎn)生交換數(shù)據(jù),change的值就改為true。每趟排序之后,判斷change的值,若為true,則繼續(xù)下一趟排序;若為false,表明這一趟沒有交換任何數(shù)據(jù),排序已經(jīng)完成。
改進(jìn)算法中,如果輸入數(shù)據(jù)集的初始狀態(tài)已經(jīng)呈現(xiàn)遞增有序,形如21、31、40、46、52、67、73、82,則if分支下的語句無執(zhí)行的機(jī)會,change的值無法改變?yōu)閠rue,始終保持false,內(nèi)層循環(huán)語句執(zhí)行結(jié)束后,i>0&&change邏輯表達(dá)式的運(yùn)算結(jié)果為false,不能再次進(jìn)入外層循環(huán),排序過程終止。
static
voidimprove_bubble_sort(int[]a,int
n){
int
i,jtemp;boolean
change=true;
for(i=n-1;i>0&&change;--i)
{
change=false;
for(
j=0;j<i;++j)
if(a[j]>a[j+1]){
temp=a[j];a[j]=a[j+1];a[j+1]=temp;
change=true;}}}1.4.4算法的空間效率分析算法的空間效率指的是包含算法的程序從運(yùn)行開始到結(jié)束所需的存儲空間。執(zhí)行一個算法所需的存儲空間包括三部分:(1)程序指令占用的存儲空間;(2)輸入數(shù)據(jù)占用的存儲空間;(3)實(shí)現(xiàn)數(shù)據(jù)處理任務(wù)所必須的輔助存儲空間。指令占用存儲空間一般與問題規(guī)模n無關(guān);輸入數(shù)據(jù)占用的存儲空間多數(shù)情況下與算法無關(guān);重點(diǎn)分析輔助存儲空間與問題規(guī)模n的函數(shù)關(guān)系,該函數(shù)記為f(n)。
請分析下面算法的空間復(fù)雜度。
static
voidimprove_bubble_sort(int[]a,int
n){
int
i,jtemp;boolean
change=true;
for(i=n-1;i>0&&change;--i)
{
change=false;
for(
j=0;j<i;++j)
if(a[j]>a[j+1]){
temp=a[j];a[j]=a[j+1];a[j+1]=temp;
change=true;}}}該冒泡排序算法除了程序指令和輸入數(shù)據(jù)占用的存儲空間,還必須用到i、j、change、temp四個輔助變量,i、j、temp是int型變量,分別占用4個字節(jié)的存儲空間,change是boolean型變量,占用2個字節(jié)的存儲空間,所以該冒泡排序算法輔助變量占用的總存儲空間14個字節(jié),即f(n)=14,算法的空間復(fù)雜度記為O(1),是常量階的。第2章線性表本章主要介紹下列內(nèi)容線性表的類型定義線性表的順序存儲結(jié)構(gòu)線性表的鏈?zhǔn)酱鎯Y(jié)構(gòu)線性表的應(yīng)用舉例2.1線性表的類型定義2.1.1線性表的定義線性表是由n(n≥0)個類型相同的數(shù)據(jù)元素組成的有限序列,通常表示成如下形式:
L=(a1,a2,...,ai-1,ai,ai+1,...,an)
其中,L為線性表名稱,習(xí)慣用大寫字母書寫;ai為組成該線性表的數(shù)據(jù)元素,習(xí)慣用小寫字母書寫。
線性表中數(shù)據(jù)元素的個數(shù)被稱為線性表的長度,當(dāng)線性表的長度為0時,線性表被稱為空線性表。
表中相鄰元素之間存在著前后次序關(guān)系,將ai-1稱為ai的直接前驅(qū),將ai+1稱為ai的直接后繼。
線性表舉例:
例如,
線性表LA=(34,89,765,12,90,-34,22)中,
線性表的長度為7,數(shù)據(jù)元素類型均為整型,765是12的直接前驅(qū),90是12的直接后繼。
線性表LB=(“Hello”,“World”,“China”,“Welcome”)中,
線性表的長度為4,數(shù)據(jù)元素類型均為字符串型,“World”是“China”的直接前驅(qū),“Welcome”是“China”的直接后繼。L=(a1,a2,...,ai-1,ai,ai+1,...,an)
2.1.2線性表的四個特點(diǎn):(1)存在唯一的一個被稱作“第一個”的數(shù)據(jù)元素。(2)存在唯一的一個被稱作“最后一個”的數(shù)據(jù)元素。(3)除第一個之外,集合中的每個數(shù)據(jù)元素均只有一個直接前驅(qū)。(4)除最后一個外,集合中的每個數(shù)據(jù)元素均只有一個直接后繼。2.1.3線性表的抽象數(shù)據(jù)類型定義
ADT
LIST{
數(shù)據(jù)對象:
D={ai|ai∈ElemSet,i=1,2,...,n,n≥0}
數(shù)據(jù)關(guān)系:
R={<ai-1,ai>|ai-1,ai
∈D,i=2,...,n}
基本操作:}
棧D是n個數(shù)據(jù)元素的集合,D是ElemSet的子集。ElemSet表示某個集合,集合中的數(shù)據(jù)元素類型相同,例如整數(shù)集、自然數(shù)集等。棧R是數(shù)據(jù)元素之間關(guān)系的集合,<ai-1,ai>表示ai-1和ai之間互為前驅(qū)和后繼的邏輯關(guān)系。
基本操作的定義如下。返回值類型
基本操作名(參數(shù)表)初始條件:<初始條件描述>操作結(jié)果:<操作結(jié)果描述>“初始條件”描述了基本操作執(zhí)行之前數(shù)據(jù)結(jié)構(gòu)和參數(shù)應(yīng)滿足的條件;若不滿足,則操作失敗,并返回相應(yīng)信息。若初始條件為空,即基本操作執(zhí)行之前數(shù)據(jù)結(jié)構(gòu)和參數(shù)不必滿足任何條件,則省略該部分-初始條件:<初始條件描述>?!安僮鹘Y(jié)果”說明了基本操作正常完成之后,數(shù)據(jù)結(jié)構(gòu)的變化狀況和應(yīng)返回的結(jié)果。關(guān)于ADT
LIST中基本操作的幾點(diǎn)附加說明:(1)線性表中的數(shù)據(jù)元素類型用T表示。(2)線性表基本操作是定義于邏輯結(jié)構(gòu)上的基本操作,是向使用者提供的使用說明?;静僮髦挥性诖鎯Y(jié)構(gòu)確定之后才能實(shí)現(xiàn)。如果線性表采用的存儲結(jié)構(gòu)不同,線性表基本操作的實(shí)現(xiàn)算法也不相同。(3)線性表基本操作的種類和數(shù)量可以根據(jù)實(shí)際需要決定。(4)線性表基本操作名稱,形式參數(shù)數(shù)量、名稱、類型,返回值類型由設(shè)計者決定,使用者根據(jù)設(shè)計者的設(shè)計規(guī)則使用基本操作。
基本操作P:(1)intCount()//求長度操作
操作結(jié)果:返回線性表中所有數(shù)據(jù)元素的個數(shù),如果線性表為空,返回0。(2)Clear()//清空
操作結(jié)果:從線性表中清除所有數(shù)據(jù)元素。(3)booleanIsEmpty()//判斷線性表是否為空
操作結(jié)果:判斷線性表當(dāng)前的狀態(tài),如果線性表為空返回true,否則返回false。
(4)Append(Titem)//附加操作初始條件:線性表未滿。操作結(jié)果:將值為item的新元素添加到表的末尾。(5)Insert(Titem,inti)//插入操作初始條件:線性表未滿且插入位置正確(1≤i≤n+1,n為插入前的表長)。操作結(jié)果:在線性表的第i個位置上插入一個值為item的新元素,這樣使得原序號為i,i+1,…,n的數(shù)據(jù)元素的序號變?yōu)閕+1,i+2,…,n+1,插入后表長=原表長+1。
(6)TDelete(inti)//刪除操作初始條件:線性表不為空且刪除位置正確(1≤i≤n,n為刪除前的表長)。操作結(jié)果:在線性表中刪除序號為i的數(shù)據(jù)元素,返回刪除后的數(shù)據(jù)元素。刪除后使原序號為i+1,i+2,…,n的數(shù)據(jù)元素的序號變?yōu)閕,i+1,…,n-1,刪除后表長=原表長-1。(7)TGetElem(inti)//取表元操作初始條件:線性表不為空且所取數(shù)據(jù)元素位置正確(1≤i≤n,n為線性表的表長)。操作結(jié)果:返回線性表中的第i個數(shù)據(jù)元素。。(8)intLocate(Tvalue)//按值查找操作
操作結(jié)果:在線性表中查找值為value的數(shù)據(jù)元素,其結(jié)果返回在線性表中首次出現(xiàn)的值為value的數(shù)據(jù)元素的序號,稱為查找成功;否則,在線性表中未找到值為value的數(shù)據(jù)元素,返回一個特殊值-1表示查找失敗。。interface
IListDS<T>{
int
Count();//求長度
void
Clear();//清空操作
boolean
IsEmpty();//判斷線性表是否為空
void
Append(Titem);//附加操作
void
Insert(Titem,int
i);//插入操作
TDelete(int
i);//刪除操作
TGetElem(int
i);//取表元
int
Locate(Tvalue);//按值查找}在Java程序設(shè)計語言中,用接口表示線性表的抽象數(shù)據(jù)類型如下。
2.2線性表的順序存儲及操作實(shí)現(xiàn)2.2.1順序存儲線性表的順序存儲是把線性表的數(shù)據(jù)元素放在一組地址連續(xù)的存儲單元中。(在這種存儲方式中,以元素在計算機(jī)內(nèi)存儲器內(nèi)的物理位置來體現(xiàn)互為前驅(qū)和后繼的邏輯關(guān)系,線性表中相鄰的元素在內(nèi)存儲器內(nèi)也相鄰)
假設(shè)每個元素占用l(小寫L)個存儲單元,線性表中第一個元素的存儲地址LOC(a1)=b,那麼線性表中第i個元素的存儲地址是多少?我們利用Java語言討論線性表的順序存儲,在Java虛擬處理器中,存儲單元的編號即存儲地址對用戶來說是透明的,不可見的。在java虛擬處理器中我們認(rèn)為線性表的順序存儲是把線性表的數(shù)據(jù)元素放在一維數(shù)組中,(數(shù)組占用的就是一組地址連續(xù)的存儲單元)數(shù)據(jù)元素在數(shù)組中相鄰就表示它們在線性表中互為前驅(qū)和后繼。在線性表抽象數(shù)據(jù)類型的定義中講過,定義于邏輯結(jié)構(gòu)上的基本操作只是向外界使用者提供的一個使用接口,存儲結(jié)構(gòu)確定了之后基本操才能實(shí)現(xiàn)。所以接下來討論在順序存儲結(jié)構(gòu)下基本操作的實(shí)現(xiàn),重點(diǎn)分析一下插入和刪除兩種基本操作的實(shí)現(xiàn)算法。ia1a2…an…ai+1
ai
a1a2…ani…ai+1ai
a1a2…ani…ai+1aiitemn+1n+1n+12.2.2順序存儲結(jié)構(gòu)下基本操作分析
1、
在線性表的第i個位置上插入數(shù)據(jù)元素item?分析插入數(shù)據(jù)元素到線性表(即數(shù)組)的第i個位置上所需移動元素次數(shù):答:從第i個數(shù)據(jù)元素到第n個都需移動,共需移動n-i+1次?每一個位置上都有可能插入數(shù)據(jù)元素,平均移動次數(shù)是多少?答:第一個位置上n次,第二個位置上n-1次,第三個位置上n-2次,依次類推,第n+1個位置上0次,平均移動次數(shù)為:[n+(n-1)+…+1]/n+1=n/2,即:
插入到每個位置的概率相等的情況下,平均移動次數(shù)是n/2,有些情況下插入到每個位置的概率不一定相等,怎麼算?假設(shè)插入到第一個位置的概率是1/2,其余位置是1/2n,所需移動元素次數(shù)的期望值(平均值)(用到概率中的數(shù)學(xué)期望概念):ia1a2…an…ai+2ai+1
iia1a2…
…ai+2ai+1
an
a1
a2
……ai+2ai+1ai
an2、
刪除線性表的第i個數(shù)據(jù)元素分析刪除線性表的第i個數(shù)據(jù)元素所需移動元素次數(shù)答:從第i+1個數(shù)據(jù)元素到第n個都需向上移動,共需移動n-i次每一個位置上的數(shù)據(jù)元素都有可能被刪除,平均移動次數(shù)是多少?答:刪除第一個數(shù)據(jù)元素需要n-1次,第二個數(shù)據(jù)元素需要n-2次,第三個位置上n-3次,依次類推,第n個位置上0次,平均移動次數(shù)為:[(n-1)+(n-2)+…+1]/n=(n-1)/2,即:刪除每個數(shù)據(jù)元素的概率相等的情況下即pi=1/n,平均移動次數(shù)是(n-1)/2,有些情況下刪除每個數(shù)據(jù)元素概率不一定相等,怎麼算?假設(shè)刪除第一個數(shù)據(jù)元素概率是1/2,其余位置是所需移動元素次數(shù)的期望值(平均值)(用到概率中的數(shù)學(xué)期望概念):2.2.3源碼實(shí)現(xiàn)1、幾點(diǎn)說明:
(1)前面定義的線性表接口IListDS<T>interface
IListDS<T>{
intCount();//求長度
voidClear();//清空操作
boolean
IsEmpty();//判斷線性表是否為空
void
Append(T
item);//附加操作
void
Insert(T
item,int
i);//插入操作
TDelete(int
i);//刪除操作
TGetElem(int
i);//取表元
int
Locate(T
value);//按值查找}(2)順序表類SeqList<T>,實(shí)現(xiàn)接口IListDS<T>。順序表類SeqList<T>的實(shí)現(xiàn)說明如下所示:
public
class
SeqList<T>implements
IListDS<T>
{T[]data;
//數(shù)組,用于存儲順序表中的數(shù)據(jù)元素
int
maxsize;//順序表的容量
int
last;
//指示順序表最后一個元素的位置
//接口成員方法的實(shí)現(xiàn);
}關(guān)于屬性成員的幾點(diǎn)說明:①Java語言中的數(shù)組在內(nèi)存中占用的存儲空間就是一組地址連續(xù)的存儲單元,所以,在Java虛擬處理器中考慮問題時,認(rèn)為線性表的順序存儲就是把線性表的數(shù)據(jù)元素存放到數(shù)組data中。②maxsize表示容量;容量可以用數(shù)組的length屬性即data.length來表示,但為了說明線性表的最大長度(容量),增強(qiáng)可讀性,在SeqList<T>類中用屬性成員maxsize來表示。③線性表中的元素由data[0]開始依次順序存放,由于線性表中的實(shí)際元素個數(shù)一般達(dá)不到maxsize,因此,在SeqList<T>類中需要一個屬性成員last表示線性表中最后一個數(shù)據(jù)元素在數(shù)組中的位置。last的值等于線性表中最后一個數(shù)據(jù)元素在數(shù)組中的下標(biāo),0≤last≤maxsize-1,last值為-1時表示線性表為空。
public
SeqList(int
size){
if
(size
>0){
data
=(T[])new
Object[size];
//建立一個長度為size,類型為T的數(shù)組
this.maxsize
=size;
//順序表的容量maxsize
=size
last
=-1;
//最后一個元素位置last=-1;表示順序表為空。
}else
throw
new
IllegalArgumentException("初始化容量必須大于0");
}(3)順序表類SeqList<T>的構(gòu)造方法在Java語言中,一個基本操作表現(xiàn)為類中的一個方法。SeqList<T>類除了實(shí)現(xiàn)接口IListDS<T>中的方法外,還可實(shí)現(xiàn)一些另外的成員方法,實(shí)現(xiàn)更復(fù)雜的操作。2基本操作實(shí)現(xiàn)(1)求長度
由于數(shù)組是0基數(shù)組,即數(shù)組的最小下標(biāo)為0,所以,長度就是最后一個元素在數(shù)組中的下標(biāo)last加1。
求長度的算法實(shí)現(xiàn)如下:
public
intCount(){
return
last+1;}2、清空操作
清除順序表中的數(shù)據(jù)元素是使順序表為空,此時,last等于-1。
public
voidClear(){
last=-1;}
3、判斷線性表是否為空如果順序表的last為-1,則順序表為空,返回true,否則返回false。
public
boolean
IsEmpty(){
if(last==-1){
return
true;}
else{
return
false;}}4、判斷順序表是否為滿如果順序表為滿,即last等于maxsize-1,則返回true,否則返回false。
public
boolean
IsFull(){
if(last==maxsize-1)//順序表已滿
{
return
true;}
else{
return
false;}}
5、附加操作
附加操作是在順序表未滿的情況下,在表的末端添加一個新元素,然后使順序表的last加1。
public
void
Append(T
item){
if(last==maxsize-1)//順序表已滿
{System.out.println("Listisfull");
//輸出順序表已滿信息提示
return;//返回
}
data[++last]=item;//在表的末端添加一個新元素,使順序表的last加1}
6、插入操作Insert(Titem,inti)
順序表的插入是指在順序表的第i個位置插入一個值為item的新元素。
插入后使原表長為n的表
(a1,a2,…,ai-1,ai,ai+1,…,an)成為表長為n+1的表
(a1,a2,…,ai-1,item,ai,ai+1,…,an)。i的取值范圍為1≤i≤n+1(n為順序表的表長=last+1)。i為n+1(即last+2)時,表示在順序表的末尾插入數(shù)據(jù)元素。
順序表上插入數(shù)據(jù)元素的步驟如下:判斷順序表是否已滿和插入的位置是否正確,表滿或插入的位置不正確不能插入;如果表未滿和插入的位置正確,則將an~ai依次向后移動,為新的數(shù)據(jù)元素空出位置。在算法中用循環(huán)來實(shí)現(xiàn);將新的數(shù)據(jù)元素插入到空出的第i個位置上;修改last(相當(dāng)于修改表長),使它仍指向順序表的最后一個數(shù)據(jù)元素。
插入操作的算法實(shí)現(xiàn)如下:
public
void
Insert(T
item,int
i){
//判斷順序表是否已滿
if(IsFull()){
System.out.println("Listisfull");
return;}//判斷插入的位置是否正確,//i小于1表示在第1個位置之前插入,//i大于last+2表示在最后一個元素后面的第2個位置插入。
if(i<1||i>last+2){
System.out.println("Positioniserror!");
return;}
//元素移動
for(int
j=last;j>=i-1;--j){
data[j+1]=data[j];}
//將新的數(shù)據(jù)元素插入到第i個位置上
data[i-1]=item;
//修改表長
++last;}算法的時間復(fù)雜度分析:順序表上的插入操作,時間主要消耗在數(shù)據(jù)的移動上;在第i個位置插入一個元素,從ai到an都要向后移動一個位置,共需要移動n-i+1個元素;i的取值范圍為1≤i≤n+1,當(dāng)i等于1時,需要移動的元素個數(shù)最多,為n個;當(dāng)i為n+1時,不需要移動元素。插入操作的時間復(fù)雜度為O(n)。7、刪除操作順序表的刪除操作是指將表中第i個數(shù)據(jù)元素從順序表中去掉。刪除后使原表長為n的表(a1,a2,…,ai-1,ai,ai+1,…,an)變?yōu)楸黹L為n-1的表(a1,a2,…,ai-1,ai+1,…,an)。i的取值范圍為1≤i≤n。i為n時,表示刪除末尾的數(shù)據(jù)元素。
順序表上刪除一個數(shù)據(jù)元素的步驟如下:(1)判斷順序表是否為空和刪除的位置是否正確,表空或刪除的位置不正確不能刪除;
(2)如果表未空和刪除的位置正確,則將ai+1~an依次向前移動。在算法中用循環(huán)來實(shí)現(xiàn);
(3)修改last(相當(dāng)于修改表長),使它仍指向順序表的最后一個元素。
publicTDelete(inti)刪除操作的算法實(shí)現(xiàn)如下:
publicTDelete(int
i){Ttmp;
//判斷表是否為空
if(IsEmpty()){
throw
new
RuntimeException("線性表為空,無法刪除");}
//判斷刪除的位置是否正確
//i小于1表示刪除第1個位置之前的元素,
//i大于last+1表示刪除最后一個元素后面的第1個位置的元素。
if(i<1||i>last+1){
throw
new
RuntimeException(“無效的下標(biāo):”+i);
//拋出異常}
tmp=data[i-1];
//tmp值等于順序表的第i個數(shù)據(jù)元素data[i-1]
for(int
j=i;j<=last;++j){
data[j-1]=data[j];}//元素移動
--last;//修改表長
return
tmp;//返回tmp值}
算法的時間復(fù)雜度分析在第i個位置刪除一個元素,從ai+1到an都要向前移動一個位置,共需要移動n-i個元素;而i的取值范圍為1≤i≤n;當(dāng)i等于1時,需要移動的元素個數(shù)最多,為n-1個;當(dāng)i為n時,不需要移動元素;刪除操作的時間復(fù)雜度為O(n)。8、取表元TGetElem(inti)取表元運(yùn)算是返回順序表中第i個數(shù)據(jù)元素。i的取值范圍是1≤i≤last+1步驟如下:判斷表是否為空和位置是否正確返回順序表的第i個數(shù)據(jù)元素
取表元運(yùn)算的算法實(shí)現(xiàn)如下:順序表取表元運(yùn)算的時間復(fù)雜度為O(1)。
publicTGetElem(int
i){//判斷表是否為空和位置是否正確
if(IsEmpty()){
throw
new
RuntimeException("線性表為空,無法刪除"); }
//判斷位置是否正確
//i小于1表示獲取第1個位置之前的元素,
//i大于last+1表示獲取最后一個元素后面的第1個位置的元素。
if(i<1||i>last+1) {
throw
new
RuntimeException("無效的下標(biāo):"+i);
//返回tmp }
return
data[i-1];//返回順序表的第i個數(shù)據(jù)元素
}
9、按值查找int
Locate(Tvalue)
順序表中的按值查找是指在表中查找滿足給定值的數(shù)據(jù)元素。從第一個元素起依次與給定值比較,如果找到,則返回在順序表中首次出現(xiàn)與給定值相等的數(shù)據(jù)元素的序號,稱為查找成功;否則,在順序表中沒有與給定值匹配的數(shù)據(jù)元素,返回一個特殊值-1表示查找失敗。
按值查找運(yùn)算的算法實(shí)現(xiàn)如下:
public
int
Locate(T
value){
//順序表為空
if(IsEmpty()){
System.out.println("listisEmpty");
return-1;}
int
i=0;
//循環(huán)處理順序表
for(i=0;i<=last;++i){
//順序表中存在與給定值相等的元素
if(data[i]==value){
break;}}
//順序表中不存在與給定值相等的元素
if(i>last){
return-1;}
return
i;}算法的時間復(fù)雜度分析順序表中的按值查找的主要運(yùn)算是比較,比較的次數(shù)與給定值在表中的位置和表長有關(guān)。當(dāng)給定值與第一個數(shù)據(jù)元素相等時,比較次數(shù)為1;而當(dāng)給定值與最后一個元素相等時,比較次數(shù)為n。平均比較次數(shù)為(n+1)/2,時間復(fù)雜度為O(n)。10、輸出線性表
2.2.4順序表中的復(fù)雜操作【例2-1】已知順序表L,寫一算法將其倒置。(a)
(b)
112336458060406640608045362311
算法思路把第一個元素與最后一個元素交換,把第二個元素與倒數(shù)第二個元素交換。也就是,把第i個元素與第n-i+1個元素交換。(n為線性表中元素的個數(shù))i的取值范圍是1到n/2。(包括n/2)public
static
void
ReversSeqList(SeqList<Integer>L)存儲整數(shù)的順序表的倒置的算法實(shí)現(xiàn)如下:
public
static
void
ReversSeqList(SeqList<Integer>L){//順序表數(shù)據(jù)類型是int
int
tmp=0;
int
n=L.Count();//n等于線性表的長度
//以下循環(huán)實(shí)現(xiàn)順序表的倒置
for(int
i=1;i<=n/2;++i){
//數(shù)組下標(biāo)從0開始,所以是下標(biāo)i-1的元素與n-i的元素交換
tmp=L.data[i-1];
L.data[i-1]=L.data[n-i];
L.data[n-i]=tmp;}}該算法只是對順序表中的數(shù)據(jù)元素順序掃描一遍即完成了倒置,所以時間復(fù)雜度為O(n)?!纠?】有數(shù)據(jù)類型為整型的順序表La和Lb,其數(shù)據(jù)元素均按從小到大的升序排列,編寫一個算法將它們合并成一個表Lc,要求Lc中數(shù)據(jù)元素也按升序排列。
算法實(shí)現(xiàn)思路是依次掃描La和Lb的數(shù)據(jù)元素,比較La和Lb當(dāng)前數(shù)據(jù)元素的值,將較小值的數(shù)據(jù)元素賦給Lc,如此直到一個順序表被掃描完,然后將未完的那個順序表中余下的數(shù)據(jù)元素賦給Lc即可。Lc的容量要能夠容納La和Lb兩個表相加的長度。線性表采用順序存儲方式時,缺點(diǎn):
1)占用一組編號連續(xù)的存儲單元。
2)刪除、插入操作需大量移動數(shù)據(jù)元素。
為什麼說占用一組編號連續(xù)的存儲單元是缺點(diǎn)?某一時刻計算機(jī)內(nèi)存狀態(tài)因?yàn)橛嬎銠C(jī)的內(nèi)存中存在很多零碎的小空間;采用順序存儲方式時,占用一組編號連續(xù)的存儲單元,這些零碎的小空間得不到充分利用.2.3.1基本概念1、線性表的鏈?zhǔn)酱鎯€性表的鏈?zhǔn)酱鎯κ侵赣靡唤M任意的存儲單元(可以不連續(xù))存儲線性表中的數(shù)據(jù)元素。
Le=(‘A’,‘B’,‘C’)數(shù)據(jù)類型是字符型,字符型數(shù)據(jù)占用二個存儲單元,即二個字節(jié)。
Le的順序存儲結(jié)構(gòu):線性表的順序存儲是指用一組連續(xù)的存儲單元存儲線性表中的數(shù)據(jù)元素。采用順序存儲方式時,數(shù)據(jù)的存儲是有規(guī)律的,計算機(jī)可以通過數(shù)據(jù)元素自身的地址計算出它的后繼元素地址,從而找到它的后繼元素。假設(shè)數(shù)據(jù)元素A的存儲地址是L,它的后繼元素的存儲地址是:L+2,L+2地址里面存放著B,表明A的后繼元素就是B數(shù)據(jù)元素B的存儲地址是L+2,它的后繼元素的存儲地址是L+4L+4地址里面存放著C,表明B的后繼元素就是C……B……A……C……864128Le=(‘A’,‘B’,‘C’)的鏈?zhǔn)酱鎯Y(jié)構(gòu):線性表采用鏈?zhǔn)酱鎯r是用一組任意的存儲單元存放它的數(shù)據(jù)元素,數(shù)據(jù)的存儲毫無規(guī)律可言,計算機(jī)通過數(shù)據(jù)元素自身的地址無法計算出它的后繼元素地址,從而無法找到它的后繼元素是誰?!?B……128A……NULLC……864128我們需要在它的后面附加一個指針(引用),指向它的后繼元素。查找元素后繼時,利用指針(引用)提供的后繼元素地址,從而找到它的后繼元素。例如:數(shù)據(jù)元素A,利用指針提供的后繼元素地址128,可以確認(rèn)它的后繼是B。數(shù)據(jù)元素B,利用指針提供的后繼元素地址8,可以確認(rèn)它的后繼是C。講到這兒,同學(xué)們能不能看出鏈?zhǔn)酱鎯Φ囊粋€優(yōu)點(diǎn)?
線性表的數(shù)據(jù)元素不必放在地址連續(xù)的存儲單元中,零碎空間能得到充分的利用。
ABCNULL在Java語言中,內(nèi)存單元的編號8,64,128對用戶屏蔽(也就是8,64,128不需要我們知道),線性表的鏈?zhǔn)酱鎯Y(jié)構(gòu)可以用如下形式表示:箭頭代表指針(引用),指向下一個元素的存儲位置一個藍(lán)色的方塊稱為一個結(jié)點(diǎn),它包含數(shù)據(jù)和指針(引用)兩部分。2、結(jié)點(diǎn):數(shù)據(jù)和引用(指針)的組合。上圖中一個藍(lán)色的方塊就代表一個結(jié)點(diǎn)。ABCnull3、鏈表:線性表的鏈?zhǔn)酱鎯Y(jié)構(gòu)。上圖三個方塊加兩個箭頭就是一個鏈表。
數(shù)據(jù)部分命名為data:用來存放線性表中數(shù)據(jù)元素。引用(指針)部分命名為next:存放下一個結(jié)點(diǎn)的地址。L=的鏈表(同學(xué)們自畫)a1a2an∧heada1a2an∧head
a1a2an
溫馨提示
- 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)用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
最新文檔
- 2025-2030年中國北斗應(yīng)急預(yù)警通信行業(yè)資本規(guī)劃與股權(quán)融資戰(zhàn)略制定與實(shí)施研究報告
- 2025-2030年中國電氣化鐵路接觸網(wǎng)行業(yè)資本規(guī)劃與股權(quán)融資戰(zhàn)略制定與實(shí)施研究報告
- 2025-2030年中國消費(fèi)性服務(wù)行業(yè)營銷創(chuàng)新戰(zhàn)略制定與實(shí)施研究報告
- 2025-2030年中國工藝品行業(yè)并購重組擴(kuò)張戰(zhàn)略制定與實(shí)施研究報告
- 自動售賣機(jī)創(chuàng)業(yè)計劃書
- 建設(shè)生態(tài)文明-推進(jìn)科學(xué)發(fā)展
- 新員工入職培訓(xùn)課件12
- 2024年幼兒園成長手冊寄語
- 狗狗護(hù)主知識培訓(xùn)課件
- 2025年中國頭孢拉定行業(yè)發(fā)展監(jiān)測及投資戰(zhàn)略研究報告
- 2024年萍鄉(xiāng)衛(wèi)生職業(yè)學(xué)院單招職業(yè)適應(yīng)性測試題庫參考答案
- 中國農(nóng)業(yè)銀行信用借款合同
- ISO 56001-2024《創(chuàng)新管理體系-要求》專業(yè)解讀與應(yīng)用實(shí)踐指導(dǎo)材料之9:“5領(lǐng)導(dǎo)作用-5.3創(chuàng)新戰(zhàn)略”(雷澤佳編制-2025B0)
- 江蘇省連云港市2023-2024學(xué)年八年級上學(xué)期期末數(shù)學(xué)試題(原卷版)
- 初中英語聽力高頻詞
- 2025年生活飲用水監(jiān)督檢查工作計劃
- Unit 3 My School Section B 1a-1d 教學(xué)實(shí)錄 2024-2025學(xué)年人教版七年級上冊英語
- 2024年度知識產(chǎn)權(quán)許可合同:萬達(dá)商業(yè)廣場商標(biāo)使用許可合同3篇
- 服務(wù)營銷課件-課件
- 一年級期末數(shù)學(xué)家長會課件
- 2024智能變電站新一代集控站設(shè)備監(jiān)控系統(tǒng)技術(shù)規(guī)范部分
評論
0/150
提交評論