內(nèi) 容 簡(jiǎn) 介
本書融入了游戲設(shè)計(jì)思想,通過游戲攻關(guān)的方式,介紹各種算法的原理和應(yīng)用。全書共分8章,具體包括排序算法、窮舉算法、遞歸算法、回溯算法、貪心算法、分治算法,棧、隊(duì)列、樹三種數(shù)據(jù)結(jié)構(gòu),動(dòng)態(tài)規(guī)劃算法,圖論相關(guān)算法等內(nèi)容。
數(shù)據(jù)結(jié)構(gòu)與算法是計(jì)算機(jī)專業(yè)的一門必修基礎(chǔ)課,作為程序員的底層能力,理解并掌握這些知識(shí)不僅可以培養(yǎng)羅輯思維能力,更能深入理解計(jì)算機(jī)系統(tǒng),從而才能寫出更高效的代碼。本書作者曾獲得2007年全國(guó)青少年信息學(xué)奧林匹克聯(lián)賽提高組福建省賽區(qū)一等獎(jiǎng),作者結(jié)合自己的學(xué)習(xí)經(jīng)歷和實(shí)戰(zhàn)經(jīng)驗(yàn),深入淺出地講解了各種常用算法和數(shù)據(jù)結(jié)構(gòu)和原理和應(yīng)用。書中將游戲與算法結(jié)合,通過游戲攻關(guān)的形式將知識(shí)點(diǎn)層層遞進(jìn),將學(xué)習(xí)枯燥的算法變成了一項(xiàng)孩子們喜歡做的事情。用淺顯易懂的語言講解分析每種算法,并給出完整的代碼,使學(xué)習(xí)變得輕松有趣。
前言
近些年來,隨著人工智能、大數(shù)據(jù)等技術(shù)的迅猛發(fā)展,計(jì)算機(jī)編程開始逐漸進(jìn)入大眾的視野,特別是教育行業(yè),各種少兒編程班如雨后春筍般層出不窮。人們開始漸漸意識(shí)到,培養(yǎng)孩子的邏輯和計(jì)算思維能力,已經(jīng)成為一項(xiàng)必不可少的教育方式。而這些在我的童年時(shí)期,是一項(xiàng)遙不可及的事物,甚至在我小學(xué)和初中期間,計(jì)算機(jī)還仍未被普及。那時(shí)候人們對(duì)計(jì)算機(jī)的認(rèn)識(shí),還只限于QQ和各種網(wǎng)頁(yè)小游戲。計(jì)算機(jī)科學(xué)發(fā)展到現(xiàn)在,我們?cè)诟袊@科技發(fā)展迅速的同時(shí),也必須感謝這個(gè)美好的時(shí)代。
我第一次接觸計(jì)算機(jī)編程是在初三保送高中階段,其他同學(xué)還在埋頭準(zhǔn)備中考時(shí),我有幸遇到了我的恩師——NOI金牌教練董永建。恰逢董老師在保送生中選拔信息學(xué)奧賽學(xué)生,中考期間和整個(gè)暑假,我們42名保送生提前開始了高中生活,跟隨董老師從零開始接觸計(jì)算機(jī)、學(xué)習(xí)算法,直到暑假結(jié)束僅剩4名同學(xué)堅(jiān)持了下來。后來,我們4個(gè)人從機(jī)房搬進(jìn)了
“小黑屋”,認(rèn)識(shí)了各位師兄,開始了真正的“封閉式訓(xùn)練”。高中三年我們沒有周末、沒有節(jié)假日、沒有寒暑假,把所有的課余時(shí)間全貢獻(xiàn)給“小黑屋”,董老師帶著我們參加各種培訓(xùn)、參加高校ACM比賽,為了鍛煉體能,帶著我們爬山,偷玩游戲被發(fā)現(xiàn)了,罰我們操場(chǎng)跑圈……和師兄、師弟、師妹們?cè)凇靶『谖荨崩锏娜陼r(shí)光,是我至今回想都覺得快樂的時(shí)光,也是影響我一生的時(shí)光。在接觸編程后半年,我第一次參加了NOIP(National Olympiad in Informatics in Provinces,信息學(xué)奧林匹克聯(lián)賽),第一年初出茅廬,只是讓我們見識(shí)了世面,師兄們才是當(dāng)年的主力。我們充當(dāng)了一年陪考,直到第二年,經(jīng)過一年多的積累,我們成了主力,在2017年NOIP競(jìng)賽,我獲得了省一等獎(jiǎng)。所有光鮮的背后,都隱藏著熬過無數(shù)個(gè)不為人知的黑夜。
非常感謝董永建老師的悉心教導(dǎo),讓我在初中時(shí)期開始,對(duì)計(jì)算機(jī)編程和算法有了初步的認(rèn)識(shí)。也緣于NOIP競(jìng)賽,讓我取得了保送武漢大學(xué)的資格,并因此決定了我畢業(yè)以后的工作和生活。人生選擇一條路,一直走到盡頭,不退讓,不更改,是件幸事。
NOIP競(jìng)賽對(duì)于很多孩子和家長(zhǎng)來說,還是一個(gè)比較陌生的事物。它由中國(guó)計(jì)算機(jī)學(xué)會(huì)(CCF)統(tǒng)一組織,每年在同一時(shí)間、不同地點(diǎn)以各省市為單位由特派員組織。這個(gè)競(jìng)賽最初設(shè)立的目的是向中學(xué)階段的青少年普及計(jì)算機(jī)科學(xué)知識(shí),帶來新的學(xué)習(xí)思路,并借此選拔人才。得益于近年來計(jì)算機(jī)知識(shí)的普及,許多對(duì)算法思想感興趣的學(xué)生和家長(zhǎng)已經(jīng)開始把NOIP當(dāng)成一種新的學(xué)習(xí)思路和方法。本書所講的算法便是基于NOIP競(jìng)賽內(nèi)容進(jìn)行闡述的,從基礎(chǔ)的排序算法到圖論算法,逐步深入了解算法思路。
為了便于各位讀者對(duì)枯燥算法內(nèi)容的理解,本書融入了游戲設(shè)計(jì)思想,通過游戲攻關(guān)的方式,介紹各種算法的原理和應(yīng)用。當(dāng)你通讀完本書后,會(huì)發(fā)現(xiàn)原來我們平常玩的各種手游,比如“吃雞”“王者”,它們的設(shè)計(jì)思路就是采用了各種算法,這時(shí)候你就已經(jīng)開始領(lǐng)略到算法的奧妙,并開始進(jìn)入算法的魔法大門了。
為什么寫這本書?
寫這本書的原因,是來自朋友和妻子的啟發(fā)。得益于NOIP競(jìng)賽,我大學(xué)的專業(yè)學(xué)的是軟件工程,畢業(yè)后從事的也是軟件開發(fā)工作。在快要邁入而立之年的某一天,我的妻子問了我一個(gè)問題:“你做了這么多年軟件開發(fā),早年也參加過NOIP,是不是現(xiàn)在得有所沉淀了?”這時(shí)候我們的女兒剛出生6個(gè)月,我因此開始思考教育的問題。這時(shí)候我的一個(gè)朋友告訴
我:“你可以嘗試寫一本書,將你在NOIP競(jìng)賽所學(xué)及多年工作積累記錄下來,說不定以后還能用來給你女兒當(dāng)教材。”于是,便有了這本《哇,編程!——跟小明一起學(xué)算法》。
■ 本書導(dǎo)讀
全書共分8章:
第1章主要介紹排序算法,介紹了經(jīng)典的三種排序方法,同時(shí)介紹了時(shí)間復(fù)雜度、空間復(fù)雜度。
第2章主要介紹了五種基礎(chǔ)的算法,是信息學(xué)奧賽中常見的五種算法,一道算法題往往會(huì)包含多種基礎(chǔ)算法,本章也為后續(xù)篇章的算法奠定基礎(chǔ)。
第3、4章主要介紹了三種數(shù)據(jù)結(jié)構(gòu)——棧、隊(duì)列、樹,數(shù)據(jù)結(jié)構(gòu)是一門語言的基礎(chǔ),本章在介紹算法的同時(shí)也強(qiáng)化了C++語言,補(bǔ)充了一些C++基礎(chǔ)學(xué)習(xí)中未涉及的內(nèi)容。
第5章主要介紹了動(dòng)態(tài)規(guī)劃算法,動(dòng)態(tài)規(guī)劃算法往往是NOIP(普及組、提高組)的壓軸題,也是能區(qū)分競(jìng)賽選手差距的題目,本章由最經(jīng)典的背包問題開始講解動(dòng)態(tài)規(guī)劃算法,理解本章才算是開始理解“算法的精髓”。
第6、7、8章主要介紹圖論相關(guān)算法,包括深度優(yōu)先搜索、廣度優(yōu)先搜索、圖的遍歷、最小生成樹、并查集、最短路徑等。這三章所講的算法屬于進(jìn)階算法,圖論相關(guān)算法其本質(zhì)依然是前面章節(jié)所講的基礎(chǔ)算法,在介紹圖論算法時(shí),也繼續(xù)幫助讀者鞏固前面部分的基礎(chǔ)算法。
閱讀本書的讀者,需要具備基本的C++語言基礎(chǔ),本書算法盡量采用通俗易懂的例子給讀者講解。學(xué)習(xí)算法并不是一件難事,算法本身也并不高深,難在理解算法后怎么再用通俗的話語講解出來。本書所涉及的算法,也是我從初中就開始接觸并學(xué)習(xí)的,我相信初中程度的讀者也能夠理解并使用本書所寫的算法。
■ 致謝
在開始寫這本書之前,我一直猶豫要不要寫,因?yàn)閯偝錾呐畠簬缀跽加昧宋宜兄苣┖涂臻e時(shí)間,是妻子的鼓勵(lì),讓我下定決心要寫完這本書,作為送給女兒的禮物。在我寫書的這半年里,所有的節(jié)假日我都躲在圖書館度過,感謝我的妻子這半年里辛勞的付出,感謝她在背后的支持。
感謝我的大學(xué)室友——JazyWoo同學(xué)提供了給我寫書的機(jī)會(huì),幫我聯(lián)絡(luò)各方資源,也感謝他在寫書期間,與我共同探討,共同完成此書。
還要再一次感謝董永建老師,是他的諄諄教導(dǎo)才能讓我有幸寫了這本算法書。
游明偉
2019年4月前言
近些年來,隨著人工智能、大數(shù)據(jù)等技術(shù)的迅猛發(fā)展,計(jì)算機(jī)編程開始逐漸進(jìn)入大眾的視野,特別是教育行業(yè),各種少兒編程班如雨后春筍般層出不窮。人們開始漸漸意識(shí)到,培養(yǎng)孩子的邏輯和計(jì)算思維能力,已經(jīng)成為一項(xiàng)必不可少的教育方式。而這些在我的童年時(shí)期,是一項(xiàng)遙不可及的事物,甚至在我小學(xué)和初中期間,計(jì)算機(jī)還仍未被普及。那時(shí)候人們對(duì)計(jì)算機(jī)的認(rèn)識(shí),還只限于QQ和各種網(wǎng)頁(yè)小游戲。計(jì)算機(jī)科學(xué)發(fā)展到現(xiàn)在,我們?cè)诟袊@科技發(fā)展迅速的同時(shí),也必須感謝這個(gè)美好的時(shí)代。
我第一次接觸計(jì)算機(jī)編程是在初三保送高中階段,其他同學(xué)還在埋頭準(zhǔn)備中考時(shí),我有幸遇到了我的恩師——NOI金牌教練董永建。恰逢董老師在保送生中選拔信息學(xué)奧賽學(xué)生,中考期間和整個(gè)暑假,我們42名保送生提前開始了高中生活,跟隨董老師從零開始接觸計(jì)算機(jī)、學(xué)習(xí)算法,直到暑假結(jié)束僅剩4名同學(xué)堅(jiān)持了下來。后來,我們4個(gè)人從機(jī)房搬進(jìn)了
“小黑屋”,認(rèn)識(shí)了各位師兄,開始了真正的“封閉式訓(xùn)練”。高中三年我們沒有周末、沒有節(jié)假日、沒有寒暑假,把所有的課余時(shí)間全貢獻(xiàn)給“小黑屋”,董老師帶著我們參加各種培訓(xùn)、參加高校ACM比賽,為了鍛煉體能,帶著我們爬山,偷玩游戲被發(fā)現(xiàn)了,罰我們操場(chǎng)跑圈……和師兄、師弟、師妹們?cè)凇靶『谖荨崩锏娜陼r(shí)光,是我至今回想都覺得快樂的時(shí)光,也是影響我一生的時(shí)光。在接觸編程后半年,我第一次參加了NOIP(National Olympiad in Informatics in Provinces,信息學(xué)奧林匹克聯(lián)賽),第一年初出茅廬,只是讓我們見識(shí)了世面,師兄們才是當(dāng)年的主力。我們充當(dāng)了一年陪考,直到第二年,經(jīng)過一年多的積累,我們成了主力,在2017年NOIP競(jìng)賽,我獲得了省一等獎(jiǎng)。所有光鮮的背后,都隱藏著熬過無數(shù)個(gè)不為人知的黑夜。
非常感謝董永建老師的悉心教導(dǎo),讓我在初中時(shí)期開始,對(duì)計(jì)算機(jī)編程和算法有了初步的認(rèn)識(shí)。也緣于NOIP競(jìng)賽,讓我取得了保送武漢大學(xué)的資格,并因此決定了我畢業(yè)以后的工作和生活。人生選擇一條路,一直走到盡頭,不退讓,不更改,是件幸事。
NOIP競(jìng)賽對(duì)于很多孩子和家長(zhǎng)來說,還是一個(gè)比較陌生的事物。它由中國(guó)計(jì)算機(jī)學(xué)會(huì)(CCF)統(tǒng)一組織,每年在同一時(shí)間、不同地點(diǎn)以各省市為單位由特派員組織。這個(gè)競(jìng)賽最初設(shè)立的目的是向中學(xué)階段的青少年普及計(jì)算機(jī)科學(xué)知識(shí),帶來新的學(xué)習(xí)思路,并借此選拔人才。得益于近年來計(jì)算機(jī)知識(shí)的普及,許多對(duì)算法思想感興趣的學(xué)生和家長(zhǎng)已經(jīng)開始把NOIP當(dāng)成一種新的學(xué)習(xí)思路和方法。本書所講的算法便是基于NOIP競(jìng)賽內(nèi)容進(jìn)行闡述的,從基礎(chǔ)的排序算法到圖論算法,逐步深入了解算法思路。
為了便于各位讀者對(duì)枯燥算法內(nèi)容的理解,本書融入了游戲設(shè)計(jì)思想,通過游戲攻關(guān)的方式,介紹各種算法的原理和應(yīng)用。當(dāng)你通讀完本書后,會(huì)發(fā)現(xiàn)原來我們平常玩的各種手游,比如“吃雞”“王者”,它們的設(shè)計(jì)思路就是采用了各種算法,這時(shí)候你就已經(jīng)開始領(lǐng)略到算法的奧妙,并開始進(jìn)入算法的魔法大門了。
為什么寫這本書?
寫這本書的原因,是來自朋友和妻子的啟發(fā)。得益于NOIP競(jìng)賽,我大學(xué)的專業(yè)學(xué)的是軟件工程,畢業(yè)后從事的也是軟件開發(fā)工作。在快要邁入而立之年的某一天,我的妻子問了我一個(gè)問題:“你做了這么多年軟件開發(fā),早年也參加過NOIP,是不是現(xiàn)在得有所沉淀了?”這時(shí)候我們的女兒剛出生6個(gè)月,我因此開始思考教育的問題。這時(shí)候我的一個(gè)朋友告訴
我:“你可以嘗試寫一本書,將你在NOIP競(jìng)賽所學(xué)及多年工作積累記錄下來,說不定以后還能用來給你女兒當(dāng)教材!庇谑牵阌辛诉@本《哇,編程!——跟小明一起學(xué)算法》。
■ 本書導(dǎo)讀
全書共分8章:
第1章主要介紹排序算法,介紹了經(jīng)典的三種排序方法,同時(shí)介紹了時(shí)間復(fù)雜度、空間復(fù)雜度。
第2章主要介紹了五種基礎(chǔ)的算法,是信息學(xué)奧賽中常見的五種算法,一道算法題往往會(huì)包含多種基礎(chǔ)算法,本章也為后續(xù)篇章的算法奠定基礎(chǔ)。
第3、4章主要介紹了三種數(shù)據(jù)結(jié)構(gòu)——棧、隊(duì)列、樹,數(shù)據(jù)結(jié)構(gòu)是一門語言的基礎(chǔ),本章在介紹算法的同時(shí)也強(qiáng)化了C++語言,補(bǔ)充了一些C++基礎(chǔ)學(xué)習(xí)中未涉及的內(nèi)容。
第5章主要介紹了動(dòng)態(tài)規(guī)劃算法,動(dòng)態(tài)規(guī)劃算法往往是NOIP(普及組、提高組)的壓軸題,也是能區(qū)分競(jìng)賽選手差距的題目,本章由最經(jīng)典的背包問題開始講解動(dòng)態(tài)規(guī)劃算法,理解本章才算是開始理解“算法的精髓”。
第6、7、8章主要介紹圖論相關(guān)算法,包括深度優(yōu)先搜索、廣度優(yōu)先搜索、圖的遍歷、最小生成樹、并查集、最短路徑等。這三章所講的算法屬于進(jìn)階算法,圖論相關(guān)算法其本質(zhì)依然是前面章節(jié)所講的基礎(chǔ)算法,在介紹圖論算法時(shí),也繼續(xù)幫助讀者鞏固前面部分的基礎(chǔ)算法。
閱讀本書的讀者,需要具備基本的C++語言基礎(chǔ),本書算法盡量采用通俗易懂的例子給讀者講解。學(xué)習(xí)算法并不是一件難事,算法本身也并不高深,難在理解算法后怎么再用通俗的話語講解出來。本書所涉及的算法,也是我從初中就開始接觸并學(xué)習(xí)的,我相信初中程度的讀者也能夠理解并使用本書所寫的算法。
■ 致謝
在開始寫這本書之前,我一直猶豫要不要寫,因?yàn)閯偝錾呐畠簬缀跽加昧宋宜兄苣┖涂臻e時(shí)間,是妻子的鼓勵(lì),讓我下定決心要寫完這本書,作為送給女兒的禮物。在我寫書的這半年里,所有的節(jié)假日我都躲在圖書館度過,感謝我的妻子這半年里辛勞的付出,感謝她在背后的支持。
感謝我的大學(xué)室友——JazyWoo同學(xué)提供了給我寫書的機(jī)會(huì),幫我聯(lián)絡(luò)各方資源,也感謝他在寫書期間,與我共同探討,共同完成此書。
還要再一次感謝董永建老師,是他的諄諄教導(dǎo)才能讓我有幸寫了這本算法書。
游明偉
2019年4月
游明偉,神雞編程教育研究院副院長(zhǎng),畢業(yè)于武漢大學(xué),原平安集團(tuán)高級(jí)工程師。15歲開始接觸編程,師從NOI金牌教練董永建,獲NOIP2007(提高組)福建省賽區(qū)一等獎(jiǎng),獲武漢大學(xué)、廈門大學(xué)等多校保送資格,參與《信息學(xué)奧賽一本通(提高篇)》第一版編著。
吳健之。武漢大學(xué)計(jì)算機(jī)學(xué)院本科、碩士,師從武漢大學(xué)原常務(wù)副校長(zhǎng)、現(xiàn)深圳大學(xué)校長(zhǎng)李清泉教授。神雞編程聯(lián)合創(chuàng)始人兼首席技術(shù)官,神雞編程教育研究院常務(wù)副院長(zhǎng),騰訊QQ音樂原高級(jí)工程師,金牌講師。10年專業(yè)編程實(shí)踐經(jīng)驗(yàn),是中國(guó)青少兒編程教育界中的實(shí)戰(zhàn)派。加入騰訊前,曾任極驗(yàn)驗(yàn)證(IDG、紅衫投資)創(chuàng)業(yè)合伙人兼產(chǎn)品副總裁。曾經(jīng)參與工信部在線視頻教學(xué)系統(tǒng)的研發(fā)、中華書局“基于二十四史的大數(shù)據(jù)與知識(shí)圖譜語義分析”研發(fā)。
目錄
第1章 整理下背包 1
1.1 桶排序 2
1.2 冒泡排序 8
1.3 快速排序 15
1.4 時(shí)間和空間復(fù)雜度 20
第2章 開始闖關(guān)吧 22
2.1 忘記密碼了——窮舉算法 23
2.2 漢諾塔——遞歸算法 25
2.3 八皇后——回溯算法 31
2.4 分裝備——貪心算法 41
2.5 二分查找——分治算法 45
第3章 爆滿的服務(wù)器與背包 53
3.1 服務(wù)器爆滿——隊(duì)列 54
3.2 合成寶石——優(yōu)先隊(duì)列 61
3.3 背包里的道具——棧 65
3.4 十進(jìn)制轉(zhuǎn)任意進(jìn)制 74
第4章 點(diǎn)亮技能樹 77
4.1 樹 78
4.1.1 樹的定義 79
4.1.2 樹的相關(guān)術(shù)語 80
4.2 二叉樹 83
4.2.1 二叉樹性質(zhì) 84
4.2.2 特殊的二叉樹 85
4.2.3 二叉樹的遍歷 87
4.2.4 二叉樹的存儲(chǔ)結(jié)構(gòu) 105
4.3 堆 107
4.3.1 大根堆與小根堆 107
4.3.2 堆的操作 109
4.4 堆排序 132
第5章 爆裝備啦,快來?yè)?139
5.1 撿到完美的海螺——遞推算法 140
5.2 01背包——?jiǎng)右?guī)算法 143
5.3 完全背包——?jiǎng)右?guī)算法 148
5.4 多重背包——?jiǎng)右?guī)算法 152
第6章 迷宮 156
6.1 圖的概念 157
6.1.1 圖的定義 158
6.1.2 圖的存儲(chǔ)結(jié)構(gòu) 162
6.2 圖的遍歷 167
6.2.1 深度優(yōu)先搜索法 168
6.2.2 廣度優(yōu)先搜索法 172
6.3 并查集 176
6.3.1 分析 177
6.3.2 并查集的原理 179
6.3.3 并查集的操作 180
6.4 最小生成樹 186
6.4.1 Prim算法 187
6.4.2 Kruskal算法 192
第7章 探索地圖每個(gè)角落 197
7.1 深度優(yōu)先搜索 198
7.2 廣度優(yōu)先搜索 211
第8章 快逃命去吧 229
8.1 拓?fù)渑判?230
8.2 最短路徑 240
8.2.1 Floyd算法 240
8.2.2 Dijkstra算法 250
8.2.3 Bellman-Ford算法 255
8.2.4 SPFA算法 261