greenplum 存儲過程_揭秘!Greenplum并行執行引擎到底是如何工作的?

65b9f8c4745a75eab208186f175edd57.gif

《深入淺出Greenplum內核》系列直播以每月一場的速度持續推出中。在第一場《架構解讀》直播里,我們了解了Greenplum的整體架構、存儲管理、索引、查詢執行、事務與日志等內容。今天(5月22日),第二場《Greenplum內核揭秘之執行引擎》也順利播出啦!現在,我們來回顧一下直播演講內容吧!

看完別忘了前往askGP做一下小測試(ask.greenplum.cn/exam)鞏固一下所學的知識點哦!b0bd152a2bcac26ccb72991c68d34fd0.png

感謝大家參加Greenplum的直播活動!在《深入淺出Greenplum內核》系列活動的第二場直播中,我們為大家詳細介紹了Greenplum執行器內容,以及Greenplum為了完成執行計劃而設計的兩個特殊模塊Dispatcher和Interconnect等內容。

執行器

首先我們先來了解一下什么是執行器。簡單來講,執行器是處理一個由執行計劃節點組成的樹,并返回查詢結果。那么什么是執行計劃節點呢?從本質上講,一個執行計劃節點,實際上就是一個數據處理節點。從下圖可看到,在數據輸入后,執行節點會對數據進行數據處理,然后返回數據作為輸出。這些執行節點會被組織成樹的形式。

5c75474049032f8aba07a0f3ecb253fa.png

下圖是一個SELECT查詢的執行計劃樹。通過優化器優化后,就會生成這樣的樹狀結構,我們可以看到里面有四個執行節點,包括HashJoin節點,Hash節點,順序掃描節點,所有的節點通過樹的方式組織在一起,來表示各節點之間的數據流動或者順序關系。? 每一個計劃節點包含足夠多的元數據信息提供給執行器。

圖中的Seq Scan被稱為原發性的掃描節點,原發性的掃描節點是指,節點本身可以自己產生數據,而不依賴于其他節點;反之,非原發性掃描節點是需要子節點來為其提供數據,圖中的Hash Join和Hash就是非原發性掃描節點。了解了原發性掃描節點和非原發性掃描節點的不同,就可以更好的理解后面的執行模型。

344df8baca4a902d866d4364a1eb88c1.png

那么執行器是怎么執行生成的執行計劃樹呢?就需要利用執行模型了。面對這樣的執行計劃樹時,處理方式其實很多,我們會根據包括每一個節點內的數據輸入是怎么樣的規定,輸出有什么樣的特點等不同的信息,會選擇不同的執行模型。現在我們來介紹一下幾種常見的執行模型。

執行模型

第一種是迭代模型,也被稱為流式模型,或者是抽拉式模型。它的定義非常簡單,每一個執行節點本質上就是一個next函數,我們會從一個樹節點的根節點一直往下執行這個next 函數。next 函數的實現會遵循這樣的特點:
  • 從輸出角度看,next 函數的每一次調用,執行節點返回一個tuple,沒有更多tuple的時候返回一個NULL。

  • 從輸入的角度看,執行節點實現一個循環,每次調用子執行節點的next函數來獲取它們的輸出,并處理它們直到能返回一個tuple或者NULL。

  • 執行控制流方向是自上往下,不斷抽拉的方式,由上層節點直接驅動下層節點來進行數據的驅動。而從數據流的角度來看,還是由上層節點往下層節點傳輸來完成。

這種執行模型的有點在于規則簡單,易懂,資源使用少,通用性好,大部分的執行計劃節點一般都可以用這種模式來實現。缺點也很顯而易見,由于每次迭代只返回一個tuple,迭代次數多,代碼局部性較差,同時對CPU cacheline也不是很友好。

708d08345f0a63bf16a4888a2236808b.png

向量化模型

第二種模型就是向量化模型,和迭代模型有一些相似之處,比如每一個執行節點實現一個next函數,但也有其不同之處。每一次迭代,執行節點返回一組tuple而非一個tuple,從而減少迭代次數,可以利用新的硬件特性如SIMD來加快一組tuple的處理。同時一組tuple在不同的節點之間傳輸,對列存也更加友好。執行節點實現一個循環,每次調用子執行節點的next函數來獲取它們的輸出,并能夠批量的處理數據。執行控制流方向自上而下,采用pull的方式。

f06329a18e474e9767dcdb07203a57d4.png

Push執行模型

第三種模型是目前比較熱門的模型——PUSH執行模型。每一個執行節點定義兩個函數
  • Produce函數

Produce函數:看起來像是一個執行節點tuple的生產函數,其實不然,對于非自主生產的執行節點,produce函數更像一個控制函數,它不做過多的生產的工作,想反它會立即調用子節點的produce函數。具有自主生產的執行節點(一般為葉子節點),其produce函數名副其實的生產tuple,并驅動父節點的consume函數提取數據。
  • Consume函數

Consume函數:被下層節點驅動調用,接收子節點數據,進行各種運算,并驅動其父節點的consume函數。

現在我們通過一個例子來看一下,下圖中有三個節點,一個掃描節點,一個投影節點,一個Join 節點。每個節點都生成了兩個函數,一個生產函數,一個消費函數。整個PUSH模型是怎么做的呢?圖中的紅框標注的為原發性的掃描節點,藍框標注的是非原發性的掃描節點。非原發性的掃描節點中的生產函數并不做真正的生產工作,而更多是承擔了控制工作,會調用它的子節點的生產函數。因此投影節點和Join節點會調用scan的生產函數。由于Scan是原發性的,因此會在生產并得到數據后,開始驅動數據的消耗。

c3a31e1fc749f76b1985da45c982b8c6.png

PUSH模型是由下層的節點驅動上層的節點來完成的。數據流向也是自下而上的。下層驅動模型可以相對容易的轉換成由數據驅動的代碼。好處就是,上層的操作就會變成本節點的算子,增加代碼的局部性。此外,這樣的代碼可以更方便進一步轉換為一個純計算代碼,例如使用LLVM優化等。個人認為這種模型通用性不強,只能做一些局部的優化。

Greenplum使用的是迭代模型,但我們正在積極探索向量化模型和PUSH模型。Greenplum正在開發相應的功能,并提交到PG社區,基本思路是利用custom scan 的可定制特性,實現向量化版本的AGG節點,SORT節點,并替換原有查詢執行樹中的相應節點。大家對這一塊感興趣也歡迎去相應的郵件列表查看。

4deebef39ee324304af43c31c8e3f8a4.png

而Greenplum執行器面臨了更大的挑戰,首先Greenplum是MPP架構,意味著大規模的并行計算,每個執行節點就需要更多的處理過程。同一個執行節點就會變成多個處理過程,而數據也會被拆分。執行節點之間進行輸入和輸出的過程中,需要不同的計算單元進行交換。

Greenplum執行的挑戰和解決方案——Motion

ca3130a701a423b680a7f2bd9ee89ee3.png

此外,Greenplum是一個Shared-Nothing的架構,這就意味著不同的計算單元之間的輸入輸出的過程會受阻。

a7033772c296d8769adfb28a2c469b25.png

面臨這樣的挑戰,Greenplum的解決方案是加了一個新的名為MOTION的執行節點,用來在不同的執行節點之間移動數據。

f5eed08e8cade469e6a4323350fc6418.png

加了Motion后,執行計劃仍然是樹狀結構。只是在不同的節點之間加了個Motion節點,并最終通過Motion節點,將數據進行匯總。

a0deb7bcac4899cc271f42dc9f5d43a8.png

接著我們來剖析一下并行化Plan。在下面的例子中,我們有一個Master和34個Segment節點。現在有兩張表:單身男和單身女,數據分布在不同的SEGMENT上。如果我們要進行一個查詢,將這兩張表格中,籍貫相同的單身男和單身女進行相親匹配,我們是如何生成一個可以被并行化執行的計劃樹呢?

為了更好的說明這個問題,我們可以在現實生活中進行映射,來方便大家理解。如果在現實生活中,我們會怎么辦?如果這些不同戶籍的單身男女在同一個省,此時處理方法就相對簡單,

  • 首先把單身女找出來

  • 再把單身男找出來

  • 再把同戶籍的男生女生分配到相同的會場

從而較為快速的把這些單身男女進行匹配和篩選。

如果這些單身男女并不在同一個省,而是分布在全國34個省中,此時要如何處理呢?

為了做一個最優的策略,我們會分情況來看,

1. 如果單身男女都居住在戶籍所在地

  • 可以由各省獨自舉辦相親會

  • 針對本省的單身男女組織相親

  • 將結果返回總部

對應到Greenplum上,是這樣的

c16f6fca50dc2c8b1b05ccb875c673e7.png

2. 對于單身女居住在戶籍所在地,而單身男生分散在全國各地。此時采取的策略可以是,

  • 各省的分部獨自舉辦相親會:
  • 將每個省的單身男青年找出來,并將他們通過火車派送回原戶籍所在地
  • 由每個省接待這些男青年,并在本省找出女單身青年,對他們進行相親配對。

如果女生數量很少,此時可以采用的策略是

  • 找到本省所有適齡單身女青年,并為其買好34個省份的車票,每個省份都去一趟。

  • 每個省接待這些單身女青年,并安排其與生活在本省的男青年相親,找出戶籍一致的配對。

對應到Greenplum上,是這樣的

fdf07ba4d18bc601fba9477b1c501b65.png

3. 如果單身男女隨機分布在全國各地,此時有兩種策略策略1:在總部舉辦相親會,各省把單身男女通過火車派送回總部,總部接待并安排相親配對。但由于總部資源有限,一般都不會采取這種策略;策略2:
  • 各分部舉辦相親會:

  • 各省找出居住在本省的適齡單身男,并按戶籍派送到相應的省。

  • 各省找出居住在本省的適齡單身女,并按戶籍派送到相應的省。

  • 各省接待全國歸來的男女,進行相親配對。

對應到Greenplum上,就是這樣的:

31e6be434897c71fb457cd808b7b2c73.png

在進行相親策劃后,我們得出了以下經驗總結:

  • 人多力量大的原則,盡量利有各省的分部

  • 要首先分析當前男女青年的地域分布

  • 必要時使用交通工具來打破地域的限制

其實在Greenplum里,也采用了類似的處理方式。每一張表都會有數據分布信息,Greenplum支持三種分布策略:鍵值分布(按列分布)、隨機分布、復制分布(數據在所有的segment上都保留了一份數據)。

Greenplum內部采用更通用的Locus信息來表示分布信息,所有的數據集合都會有數據分布狀態的。

612d9a0e8703635b13d67242a0235c17.png

Greenplum通過Motion來打破物理上的隔離。包括下圖中的四種Motion。Redistribute Motion是通過鍵值把Tuple在多個節點間進行重分布。Gather/Gather Merge Motion是把不同Segment上的數據聚集到一個節點上,Gather Merge保證了一個有序的收集過程。Broadcase Motion顧名思義就是廣播,每個節點都發送一份。Explict Redistribute Motion常用于Update/Delete操作,該類操作需要在數據原來所在的節點上進行更新或刪除,保證數據分布不會出現不一致。gp segment id隱藏列保存了數據所在原來節點信息。

并行化Plan

2df198e31e831bf9cdfbb42ad5e54ac2.png

Motion會引起數據的遷移,帶來執行代價,所以Greenplum會對需不需要做Motion進行代價評估,評估依據主要是當前數據集合的數據分布狀態和在當前數據集合上將要執行的操作。

08e53cce155b5c1171defb60932469a8.png

現在我們通過一個分布式Join的例子來鞏固一下。下面是一個簡單的inner join。A、B都是按照Hash分布的鍵值表。也就是數據被分散在各個Segment上,而每個Segment上只有部分數據。要做到A inner join B的完整數據集,就需要把B表全部復制到所有的segment上,和A的部分數據Join。得到的Plan就如下圖所示。前面我們提到,在Join完成后,也會有個數據分布。本例中,在Join完成后,還是會通過Hash分布。接著,由于QD會直接和Client進行交互,因此需要把所有的數據Gather到QD上,再由QD發送給Client。而其中的優化過程,會在本《深入淺出Greenplum內核》系列直播后續的課程中細講,請大家關注。

f6c09064cfbdd5f0099a1ca586915225.png

如果A是一個鍵值表,B是一個復制表。前面的Broadcast就不需要做了,可以直接進行Join。每個并行處理單元處理下圖中的計劃樹,再Gather到QD即可。

bd43a3ab8061ed1d5dccb6036aa88532.png

如果A是鍵值表,而B是general的數據分布。B會在每個segment上都能產生1-10的數據,就能滿足Join的需求。

9ef86c8ae27f05d758c5c38d4c092ce5.png

如果A不變,而B是一個子查詢,是SingleQE的數據分布,即在一個segment上提供這樣的數據。其中一種策略就是,把分布各個Segment上的A的數據都Gather到一個Segment上執行。此時Join后的數據模型就會變成SingleQE的數據分布。

4c020888bf5cdd8a5111d000e7741768.png

如果在Inner Join時加個條件,就可以將Broadcast Motion換成Redistribute Motion。讓c2這一列按照c1這個Hash重新分布到其他segment上,從而減少數據的移動。

960846a9ab0d094b0062ba6fdf354d1a.png

我們再來看一個要AGG操作的例子,在下面的例子中,對A進行AGG操作,計算c1的count值。此時,我們只需要在每個Segment上做AGG,再Gather到QD即可。

846567e70651ed1160613c2945d6d9bd.png

如果A表是按照C2做分布的(非兩階段),則前面的策略便不可用了。此時,我們可以將A可以按照C1做Redistrbute Motion,在前面提到的操作即可。

1d9211d3acacb70eea491258d60cd3bf.png

Dispatcher

講完分布式Plan的產生,我們再來看一下Greenplum中為了支持分布式plan而設計的模塊。第一個就是Dispatcher。上面提到的相親的策略,
  • 各省的分部獨自舉辦相親會。

  • 首先每個省的單身男青年找出來,并將他們通過火車派送回原戶籍所在地。

  • 然后每個省接待這些男青年,并在本省找出女單身青年,對他們進行相親配對。

具體實施起來是怎么樣的呢?

d65fa655e6ba07dca9724e0d72ca7666.png

對應到Greenplum上,有了分布式plan,一堆計算資源是如何分配調度和執行起來的呢?

218e3af1220af1c59fb2ff55a0f7c1bc.png

Dispatcher首先要做的就是分配QE資源。從plan的角度來看,會將計劃做成SliceTable,SliceTable中會告知Slice2從34個segment來分配資源,而Slice3只需要Segment2來提供資源即可。

07e4d921e90953c5b864766a5bf44492.png

Dispatcher從SliceTable中得到信息后,會去分配資源。它會向CdbComponentDatabases這個component來申請資源,并將得到的資源回寫到SliceTable中。原本,SliceTable中只包括了需要在哪幾個Segment上起QE資源的較模糊的指令,但在分配完后,每個SliceTable就會得到QE資源具體的節點信息,包括地址和端口等。

a7ba99b597925ce3e060f4ea787367d5.png

Dispatcher分配QE資源通過調用allocateGang()函數完成。GANG大小的分配非常靈活,最小可以只分配一個QE資源,而一般為segment的個數,甚至可以支持大于segment的個數的QE資源,即每個segment可以為一個gang分配多于一個的QE資源。此外QE資源閑置后,并不會被馬上回收,而是可以被后續的查詢重用,減少了重復分配QE帶來的開銷。

103ff24c344d2b763b1947b8ca30bc52.png

Dispatcher第二個功能是分發任務。CdbDispatchPlan可以分發并行性化plan的任務,SliceTable也會連同這個分布式plan一起發給QE。這樣的話所有的QE通過SliceTable可以找到自己預先被分配屬于哪個Gang,以及它的父節點的Gang是哪些以便于建立節點間通信。通過Parent Gang具體的QE描述符,我們就可以知道要把數據傳送到哪個端口。也可以分發純文本的、兩階段提交、查詢樹的任務。

2b16b2c57a02a64c81c51c153c546881.png

Dispatcher的第三個功能就是協調功能,通過cdbdisp_checkDispatchResult函數來控制QE的狀態。有下面四種等待模式。

dcbc5531a9eac27746991069bfe2f427.png

下圖就是一個典型的Dispatcher程序。Greenplum內的代碼基本都會遵循這樣的邏輯:分配上下文-分配資源-發送任務-等待發送的完成-等待QE的狀態-銷毀上下文。

356e637ca7428da31faf90fb71069598.png

Interconnect

第二個模塊就是Interconnect。Greenplum是通過網絡在QE之間移動數據,這個網絡模塊就是Interconnect。在Motion節點被初始化時,發送端和接收端就會建立Interconnect網絡連接。在Motion節點執行時,就會通過Interconnect來發送數據。

38958abf9f05de25ad6220b7bdca3b5f.png

下圖是Interconnect的分層介紹。從應用層來說,主要任務是發送數據。Interconnect會對Tuple進行包裝,將其包裝成一個個Chunk。有些Tuple很大,就會進行切割,將其切成多個Chunk。Chunk通過數據包發送給receiver端。應用層還有一些數據流控制的包,包括EOS包,STOP包等。所有的包都會通過系統傳輸層中的UDPIFC和TCP IC進行傳輸。

ee08ca886d0309d90d386f2069eebda1.png

UDPIFC是Greenplum自己實現的一種RUDP(Reliable User Datagram Protocol)協議。基于UDP協議開發的,為了支持傳輸可靠性,實現了重傳,亂序處理,重傳處理,不匹配處理,流量控制等功能。GPDB當初引入UDPIFC主要為了解決復雜OLAP查詢在大集群中使用連接數過多的問題。UDPIFC實際上是一種線程模型。

66093b8c0ddf48f43c198d2883094186.png

后續,我們也可能會增加一些新的Interconnect類型,包括QUIC協議,Proxy協議等,歡迎大家的關注。

8fae52269a090fddff696ef3aa313812.png

關于Hashjoin的內容,由于時間原因,本次分享就不做詳細的講解,如果大家對這一塊感興趣,可以反饋給我們社區,我們可以在后面添加專門的講解。大家可以參考一下之前Greenplum中文社區公眾號發布的關于Hashjoin的文章來了解相關內容。

40e7a73887c7826db1d8e3bf7e1051c3.gif

Greenplum小測試第二期

(ask.greenplum.cn/exam)

參加活動即可獲得“青梅稱號”,截圖分享至朋友圈還有機會獲得Greenplum變色杯!

c926c0376661ea72314332bfe035d2cf.png

44ca950a107296d3b1852b2f5a82b831.png

我知道你

在看

72166b3e047e0001c0769c5855da1454.png

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/534515.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/534515.shtml
英文地址,請注明出處:http://en.pswp.cn/news/534515.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

倩女幽魂服務器維護時間,9月5日在線維護公告

親愛的玩家:《倩女幽魂》手游將于本周四上午進行在線更新,更新期間無需停服,玩家可照常進行游戲。歡迎您屆時體驗本周放出的全新內容,祝全體玩家游戲愉快!本周四在線更新內容如下:系統1.跨服隊伍和團隊功能…

區塊鏈是什么通俗解釋_區塊鏈是什么?如何用一句話通俗解釋區塊鏈

站長之家(Chinaz.com)注:很多人可能跟小編一樣,為了搞懂區塊鏈的概念,看了無數篇文章,但是很多說的都是大概念。雖然每個字都看得懂,但是連在一起就看不懂了。那么,區塊鏈到底是什么東西呢?如何用一句話通…

和push的區別_還沒有理解let 和 const的用法和區別嗎,幾百字讓你立馬搞懂

本文主要講解ES6中變量的相關操作,變量的命名, 講解 var、 let 、 const 三者的區別正文ES6中的 let 和 const 都是用來聲明變量的, 他們與 var 有所區別let 命令我們都知道在for循環中,我們命名的變量 i 一般都只是為了在這個循環…

深度學習分類類別不平衡_「圖像分類」 關于圖像分類中類別不平衡那些事

作者&編輯 | 郭冰洋1 簡介小伙伴們在利用公共數據集動手搭建圖像分類模型時,有沒有注意到這樣一個問題呢——每個數據集不同類別的樣本數目幾乎都是一樣的。這是因為不同類別的樣例數目差異較小,對分類器的性能影響不大,可以在避免其他因…

vue設置多選框默認勾選_Vue實現全選和反選即Vue復選框增加全選功能

導語:Vue中單選下拉框開發起來非常簡單,直接select包裹一個帶v-for的option即可但是當我們想做個帶多選的下拉框該怎么辦呢?最簡方法是什么?比如下面這個圖:如果網上搜的話,搜的是一堆帶children的 ,那種是遍歷tree的思想,和多選下拉框不是一回事,而且寫起來復雜看不懂源碼再…

pyqt5 下拉 多頁 點擊_PyQt5實戰——自定義翻頁控件實現

分頁控件效果圖一、環境要求python解釋器:python3.7.4依賴:PyQt5、sys模塊二、思路分析1、布局:”上一頁“、”下一頁“等button、edit及label控件采用水平布局,使用該布局填充主控件QWidget2、類繼承關系:主界面繼承自…

三種平攤分析的方法分別為_干貨|電工必須學會的三極管電路分析方法

三極管有靜態和動態兩種工作狀態。未加信號時三極管的直流工作狀態稱為靜態,此時各極電流稱為靜態電流,給三極管加入交流信號之后的工作電流稱為動態工作電流,這時三極管是交流工作狀態,即動態。一個完整的三極管電路分析有四步&a…

休眠后gpio狀態_淺談Digi XBee模塊的休眠模式

淺談Digi XBee模塊的休眠模式2020-3-25Digi XBee S2C模塊,如果僅連接電源線可以測得,在待機情況下,大約是10.5mA左右的電流,在休眠時的功耗可以低到0.5uA。可以知道,休眠幾乎不耗電。在Spec上標的待機功耗會比實測只接…

xodo上的筆記不見了_一起來“終極筆記名場面批發市場”進貨嗎

俗話說得好,誰都逃不過“真香定律”,三天不見小筆記,想他。不對,不應該叫小筆記,應該叫粉絲起的名字——“【瓶邪黑花】《終極筆記》(原著風/連載)”下面請接受來自粉絲的瘋狂打call~自上周《終極筆記》播出&#xff0…

python3學習筆記 雨痕_Python 3 學習筆記:數字和布爾

數字基本類型整數在 Python 編程中,整數就是數學意義上的整數,包括正整數、負整數和零,且它的位數是任意的。根據表示方法的不同,可以分為:二進制整數八進制整數十進制整數十六進制整數浮點數浮點數,即數學…

不越獄換壁紙_那些不舍得換的手機插畫壁紙,你還差幾張?

酷愛收藏美作的微課菌可以大展拳腳了!分享3位自己收藏的畫師作品,絕對每一張都讓你舍不得換,手機可以扔,壁紙得先拷貝存起來!開始吧!!多圖預警!NO.1:Atey Ghailan&#x…

程序員績效總結_年終總結怎么寫?

每年這時候,都有必要寫一下一年的個人工作總結。年終總結或工作總結是個挺重要的事。寫好的話,有助于說明自己的工作績效,績效好可以多拿年終獎,還有機會爭取晉升。另外,一年下來的個人生活也需要總結,這一…

線程停止繼續_線程不是你想中斷就能中斷

這是我2021年的第2篇原創文章,原汁原味的技術之路盡在Jerrycodes為什么不強制停止如何用 interrupt 停止線程sleep 期間能否感受到中斷停止線程的方式有幾種總結啟動線程需要調用 Thread 類的 start() 方法,并在 run() 方法中定義需要執行的任務。啟動一…

倒序查10條數據_10 | 怎么給字符串字段加索引?

現在,幾乎所有的系統都支持郵箱登錄,如何在郵箱這樣的字段上建立合理的索引,是我們今天要討論的問題。假設,你現在維護一個支持郵箱登錄的系統,用戶表是這么定義的:mysql> create table SUser( ID bigin…

保留小數點后三位_【Meta分析】Stata制作森林圖時,如何保留三位小數?

系統評價/Meta分析指全面收集所有相關研究并逐個進行嚴格評價和分析,再用定性或定量合成的方法對資料進行處理得出綜合結論的研究方法。在指導學員的過程中發現初學者在學習過程中常常會碰到許多共性問題,本公眾號特此開設專欄解答,希望能夠和…

android自動計步_Android計步模塊實例代碼(類似微信運動)

最近在項目中研究計步模塊,每天0點開始記錄當天的步數,類似微信運動。碰到了不少坑今天有時間整理出來給大家看看。做之前在google、baidu、github上搜了個遍沒找到好的,大多數都是需要在后臺存活,需要后臺Service。對于現在的各大…

python井字棋ai_實現AI下井字棋的alpha-beta剪枝算法(python實現)

代碼參考自中國大學mooc上人工智能與信息社會陳斌老師的算法,我在原來的基礎上增加了玩家輸入的異常捕獲 AlphaBeta剪枝算法是對Minimax方法的優化,能夠極大提高搜索樹的效率,如果對這個算法感興趣的可以去參考相關資料。 當正確理解AlphaBet…

Redis小計(2)

目錄 1.exists命令 2.del命令 3.expire/pexpire命令 4.ttl命令 5.redis對于key過期的刪除策略 1.exists命令 exists X1 X2 X3 X4:返回四個key存在的個數。 2.del命令 del X1 X2:刪除key。 3.expire/pexpire命令 給key設置超時時間。 expire key…

unity 彩帶粒子_iOS動畫開發----粒子系統---彩帶效果

參考博文地址:http://my.oschina.net/u/2340880/blog/485095?fromerrbgjLq4Mw一、粒子發射器iOS中的粒子效果有兩部分組成,一部分為發射器,設置例子發射的宏觀屬性,另一部分是粒子單元,用于設置相應的粒子屬性。粒子發射器是基于…

一秒執行一次_《一秒鐘》:一貫的粗曠式抓大放小,張藝謀的自命題作業總是要觀眾自己再做一遍...

還有不變的永遠在奔跑的大棉褲花棉襖的圓臉妮子,這是導演張藝謀最新作品《一秒鐘》的最直接觀感。張藝謀是個善于從普世情懷處挖掘題材的導演。之前諸多現實題材類型作品,諸如講父子和解的《千里走單騎》、夫妻愛情的《歸來》以及《我的父親母親》&#…