沒有集群環境,怎麼學好大數據處理框架(Hadoop,Spark)的相關知識?

沒有集群環境,怎麼學好大數據處理框架(Hadoop,Spark)的相關知識?

在已經有Java,scala等豐富經驗的前提下。


推薦用docker搭建你的集群,真是太方便了。大數據是一個知識體系,不僅僅是spark,你其實應該學的還包括,文件系統hdfs,Nosql,我推薦的是Cassandra,分散式消息隊列,比如kafka,由於kafka綁定了zookeeper,所以zk也少不了。流式處理越來越重要,storm雖然很不錯,但spark streaming由於能夠和spark sql,mllib無縫整合,所以更加推薦。最後你會發現搭建可以拿來學習的開發環境是相當頭疼的。那麼,學習docker吧,github上有大量配置好的鏡像可以供你使用。可以任意組織你期望的集群。所有一切都可以跑在一台單機上。總之,docker是學習大數據的終極利器。


Spark雖然是大規模的計算框架,但也支持在單機上運行,對於入門學習者而言,單機環境已經足夠。實驗樓Spark訓練營Hadoop - Spark 大數據動手實驗第一節提供了免費的在線Spark學習環境。

Spark安裝非常簡單,如需本地安裝可以參考以下步驟。

1. 安裝

1.1 安裝前準備

安裝Spark之前需要先安裝Java,Scala及Python。

安裝Java

實驗樓環境中已經安裝了JDK,這裡打開桌面上的Xfce終端,執行查看Java版本:

可以看到實驗樓的Java版本是1.8.0_60,滿足Spark 1.5.1對Java版本的要求。

如果需要自己安裝可以在Oracle的官網下載Java SE JDK,下載鏈接:Java SE - Downloads。


安裝Scala

老版本的Spark安裝前需要先裝Scala,1.5.1版本可以無需這一步驟。但為了自己開發Scala程序調試的方便我們仍然安裝一個最新版本2.11.7的Scala。

Scala官網下載地址:http://www.scala-lang.org/download/

由於官網速度很慢,我們預先上傳到了實驗樓內網,下載並解壓到/opt/目錄:

wget http://labfile.oss.aliyuncs.com/courses/456/scala-2.11.7.tgz
tar zxvf scala-2.11.7.tgz
sudo mv scala-2.11.7 /opt/

測試scala命令,並查看版本:


安裝Python及IPython

安裝執行命令:

sudo apt-get update
sudo apt-get install python ipython

實驗樓中已經安裝了Python及IPython,分別查看版本:

1.2 Spark下載

課程中使用目前最新穩定版:Spark 1.5.1,官網上下載已經預編譯好的Spark binary,直接解壓即可。

Spark官方下載鏈接:Downloads | Apache Spark

下載頁面中我們如下圖選擇Pre-build for Hadoop 2.6 and later並點擊下載:

為了節約時間,我們選擇從阿里雲的鏡像下載:

wget http://mirrors.aliyuncs.com/apache/spark/spark-1.5.1/spark-1.5.1-bin-hadoop2.6.tgz

大約268M大小,下載完成後解壓並拷貝到/opt/目錄:

tar zxvf spark-1.5.1-bin-hadoop2.6.tgz
sudo mv spark-1.5.1-bin-hadoop2.6 /opt/

進入到spark目錄查看目錄結構,本節實驗中會用到bin/目錄下的操作命令以及conf/目錄下的配置文件。

1.3 配置路徑與日誌級別

為了避免每次都輸入/opt/spark-1.5.1-bin-hadoop2.6這一串前綴,我們將必要的路徑放到PATH環境變數中(實驗樓用的是zsh,所以配置文件為~/.zshrc):

# 添加配置到zshrc
echo "export PATH=$PATH:/opt/spark-1.5.1-bin-hadoop2.6/bin" &>&> ~/.zshrc

# 使zshrc起作用
source ~/.zshrc

# 測試下spark-shell的位置是否可以找到
which spark-shell

我們進入到spark的配置目錄/opt/spark-1.5.1-bin-hadoop2.6/conf進行配置:

# 進入配置目錄
cd /opt/spark-1.5.1-bin-hadoop2.6/conf

# 基於模板創建日誌配置文件
cp log4j.properties.template log4j.properties

# 使用vim或gedit編輯文件log4j.properties
# 修改log4j.rootCategory為WARN, console,可避免測試中輸出太多信息
log4j.rootCategory=WARN, console

# 基於模板創建配置文件
sudo cp spark-env.sh.template spark-env.sh

# 使用vim或gedit編輯文件spark-env.sh
# 添加以下內容設置spark的環境變數
export SPARK_HOME=/opt/spark-1.5.1-bin-hadoop2.6
export SCALA_HOME=/opt/scala-2.11.7

spark-env.sh配置如圖:

spark-env.sh腳本會在啟動spark時載入,內容包含很多配置選項及說明,在以後的實驗中會用到少部分,感興趣可以仔細閱讀這個文件的注釋內容。

至此,Spark就已經安裝好了,Spark安裝很簡單,依賴也很少。

後續幾節介紹簡單的Spark操作,為以後的實驗做基礎。

1.4 Spark-Shell

Spark-Shell是Spark自帶的一個Scala交互Shell,可以以腳本方式進行互動式執行,類似直接用Python及其他腳本語言的Shell。

進入Spark-Shell只需要執行spark-shell即可:

spark-shell

進入到Spark-Shell後可以使用Ctrl D組合鍵退出Shell。

在Spark-Shell中我們可以使用scala的語法進行簡單的測試,比如下圖所示我們運行下面幾個語句獲得文件/etc/protocols的行數以及第一行的內容:

上面的操作中創建了一個RDD file,執行了兩個簡單的操作:

  • count()獲取RDD的行數
  • first()獲取第一行的內容

我們繼續執行其他操作,比如查找有多少行含有tcp和udp字元串:

查看一共有多少個不同單詞的方法,這裡用到Mapreduce的思路:

上面兩步驟我們發現,/etc/protocols中各有一行含有tcp與udp字元串,並且一共有243個不同的單詞。

上面每個語句的具體含義這裡不展開,可以結合你閱讀的文章進行理解,後續實驗中會不斷介紹。Scala的語法我們在後續實驗中會單獨學習,這裡僅僅是提供一個簡單的例子讓大家對Spark運算有基本認識。

操作完成後,Ctrl D組合鍵退出Shell。


pyspark

pyspark類似spark-shell,是一個Python的交互Shell。

執行pyspark啟動進入pyspark:

退出方法仍然是Ctrl D組合鍵。

也可以直接使用IPython,執行命令:IPYTHON=1 pyspark

在pyspark中,我們可以用python語法執行spark-shell中的操作,比如下面幾個語句獲得文件/etc/protocols的行數以及第一行的內容:

操作完成後,Ctrl D組合鍵退出Shell。

在後續的實驗中我們將大量使用python和scala的互動式shell,可以及時的獲得實驗結果,實驗重在理解原理,內容將很少涉及Java的內容,如果你對Java很熟悉可以參考後續的實驗代碼練習。

2. 啟動spark服務

這一節我們將啟動spark的master主節點和slave從節點,這裡也會介紹spark單機模式和集群模式的部署區別。

2.1 啟動主節點

執行下面幾條命令啟動主節點:

# 進入到spark目錄
cd /opt/spark-1.5.1-bin-hadoop2.6

# 啟動主節點
./sbin/start-master.sh

沒有報錯的話表示master已經啟動成功,master默認可以通過web訪問http://localhost:8080,打開桌面上的firefox瀏覽器,訪問該鏈接:

圖中所示,master中暫時還沒有一個worker,我們啟動worker時需要master的參數,該參數已經在上圖中標誌出來:spark://7a1e9a46bf54:7077,請在執行後續命令時替換成你自己的參數。

2.2 啟動從節點

執行下面的命令啟動slave

./sbin/start-slave.sh spark://7a1e9a46bf54:7077

沒有報錯表示啟動成功,再次刷新firefox瀏覽器頁面可以看到下圖所示新的worker已經添加:

也可以用jps命令查看啟動的服務,應該會列出Master和Slave。

2.3 測試實例

使用pyspark連接master再次進行上述的文件行數測試,如下圖所示,注意把MASTER參數替換成你實驗環境中的實際參數:

刷新master的web頁面,可以看到新的Running Applications,如下圖所示:

當退出pyspark時,這個application會移動到Completed Applications一欄。

可以自己點擊頁面中的Application和Workers的鏈接查看並了解相關信息。

2.4 停止服務

停止服務的腳本為sbin/stop-all.sh,運行時需要輸入shiyanlou用戶的密碼,因為腳本中使用ssh遠程對slave節點進行管理:

cd /opt/spark-1.5.1-bin-hadoop2.6
./sbin/stop-all.sh

2.5 集群部署

上面的步驟介紹了我們在單機狀態Standalone Mode下部署的spark環境,如果要部署spark集群稍有區別:

  1. 主節點上配置spark,例如conf/spark-env.sh中的環境變數
  2. 主節點上配置conf/slaves,添加從節點的主機名,注意需要先把所有主機名輸入到/etc/hosts避免無法解析
  3. 把配置好的spark目錄拷貝到所有從節點,從節點上的目錄路徑與主節點一致,例如都設置為/opt/spark-1.5.1-bin-hadoop2.6
  4. 配置主節點到所有從節點的SSH無密碼登錄,使用ssh-keygen -t rsa和ssh-copy-id兩個命令
  5. 啟動spark集群,在主節點上執行sbin/start-all.sh
  6. 進入主節點的web界面查看所有worker是否成功啟動

轉載請註明課程鏈接:Hadoop - Spark 大數據動手實驗


Spark單機也能跑


關於學好大數據處理框架,我這裡這樣假設:

  1. 學會大數據框架應用開發;
  2. 學會大數據框架;

第一點相對簡單,前面很多位都有回答,沒有錯,spark可以單機部署的,基本上主流框架都能單機部署,所以老老實實低頭下載、編譯、部署、coding、測試吧,多看文檔和源碼,多寫代碼,少看《xx天精通xx》之類的東西(不過可以參考玩玩,別當真就行)。

關於機器配置,其實也不用太誇張,這無非又是給自己的惰性找個借口罷了,主流能跑得動LOL的機器都能滿足你基本的測試學習需要了,實在不行就壯士斷腕放棄遊戲直接把系統裝成linux……所以關鍵還是你學習慾望是否強烈的問題。

當然其實也沒有那麼簡單,大數據量和小數據量畢竟有本質上的區別,根本就是兩個世界的東西,處理100M和100T數據的區別不只是時間長短、節點多少的問題,有些問題只有在大規模數據處理時才會遇到,能夠解決好這類問題的人就很厲害了,也是這行門檻所在。

那麼重點我想談談的是第二個情況:假設你想學好框架。

Hadoop 和Spark發展到了今天,都已經不僅僅是一個計算框架了,而使已經演化成了生態完整的系統,很多這個行業最優秀的程序員為它們做了貢獻。讚美開源世界,這些代碼對你都是Open的,那麼就去閱讀好了,帶著目的的那種。比如你看到了spark standalone的任務提交流程代碼,那麼為什麼它這麼搞?能從中借鑒什麼?假設哪天自己要設計一個別的分散式系統時是否能夠參考?有什麼優缺點?這些東西我認為在沒有集群的情況下都是能夠做的。

假設有這樣的積累,當開始工作時,你放心:任何系統都會出現問題,當問題發生時對你來說應該一切都是脈絡清晰的;任何系統都不可能滿足所有需求,當新需求spark/hadoop或者其他什麼的滿足不了需求時需要重新開發或者改造時,你應該使思路活躍的,應該是能夠直擊問題關鍵點的。當然這些鍛煉在沒有集群和實際操作的情況下是很難做到的,但可以先做好準備。

我定義學好,在於系統的每個動作對你來說都是很清晰的,你知道它做這個動作的理由,它的實現方法,這個動作產生的影響,可能會出現問題的點……我比較笨,大概只能想到好好積累這一種手段……


已經有 java 和 scala 豐富經驗的前提下,這個問題不復存在了


這麼多人都說了,Hadoop和Spark都可以單機跑,


雲棲社區圍繞大數據技術——Spark整理了一份詳盡的學習、實踐課程,整個課程分為三大部分:

一、基礎打磨:《Scala入門到精通》《LinuxAkka基礎》的系列學習課程;

二、深入理解:本部分主要包含《源碼走讀》、《組件解析》兩部分內容;

三、應用實踐:從語言的場景實踐到技術應用提供了全方的實戰演練案例,讀者可以根據內容手把手地進行;

還有最後一部分是介紹基於阿里雲E-MapReduce(點擊了解產品詳情)的實踐應用講解。

小編特意為想深入學習的上雲「新」同學們準備了一份厚禮:10個阿里雲免費套餐邀請碼(想必很多同學知道他的價值,如果還有不了解的可以點擊此處查看)用於實踐學習使用。

獲得方式:各位同學請在知乎原文評論中,點擊括弧中的原文——&>(絕對乾貨:Spark核心技術系列課程和實踐學習 - 知乎專欄)回復:我要阿里雲免費套餐邀請碼(可以免費申請開通阿里雲伺服器、資料庫等30多款雲計算產品),為公平起見採用評論點贊排名前十位的小夥伴可得。

課程介紹如下,提供給各位同學提前預覽各取所需,課程學習頁面請點擊此處進入

一、基礎打磨篇:

1.1 《Scala入門到精通》

· 第一節:Scala語言初步

· 第二節:Scala基本類型及操作、程序控制結構

· 第三節:Array、List

· 第四節:Set、Map、Tuple、隊列操作實戰

· 第五節:函數與閉包

· 第六節:類和對象(一)

· 第七節:類和對象(二)

· 第八節:包和引入

· 第九節:繼承與組合

· 第十節:Scala類層次結構、Traits初步

· 第十一節:Trait進階

· 第十二節:I/O與正則表達式

· 第十三節:高階函數

· 第十四節:Case Class與模式匹配(一)

· 第十五節:Case Class與模式匹配(二)

· 第十六節:泛型與註解

· 第十七節:類型參數(一)

· 第十八節:隱式轉換與隱式參數(一)

· 第十九節:隱式轉換與隱式參數(二)

· 第二十節:類型參數(二)

· 第二十一節:類型參數(三)

· 第二十二節:高級類型 (一)

· 第二十三節:高級類型 (二)

· 第二十四節:高級類型 (三)

· 第二十五節:提取器(Extractor)

· 第二十六節:Scala並發編程基礎

· 第二十七節:Scala操縱XML

· 第二十八節:Scala與JAVA互操作

· 第二十九節:Scala資料庫編程

· 第三十節:Scala腳本編程與結束語

1.2 《Linux Akka基礎》

· 第一節:Linux介紹、安裝及使用初步

· 第二節:Linux文件系統、目錄(一)

· 第三節:用戶和組 · 第四節:Linux文件系統(二)

· 第五節:vi、vim編輯器(一)

· 第六節:vi、vim編輯器(二)

· 第七節:進程管理

· 第八節:網路管理

· 第九節:Shell編程入門(一)

· 第十節:Shell編程入門(二)

· 第十一節:Shell編程入門(三)

· 第十二節:Shell編程入門(四)

· 第十三節:Shell編程入門(五)

· 第十四節:Shell編程入門(六)

· 第十五節:基礎正則表達式(一)

· Akka並發編程(一):並發編程簡介

· Akka並發編程(二)::Actor模型(一)

· Akka並發編程(三):Actor模型(二)

· Akka並發編程(四):Actor模型(三)

· Akka並發編程(五):Actor模型(四)

· Akka並發編程(六):Actor模型(五)

· Akka並發編程(七):Actor模型(六)

· Akka並發編程(八):Actor模型(七)

二、深入理解篇

2.1 《源碼走讀》

· 第一節:Spark應用程序提交流程

· 第二節:SparkContext的創建

· 第三節:Spark Job的提交

· 第四節:Stage劃分

· 第五節:Stage提交

· 第六節:Task提交

· 第七節:ResourceOffers與LaunchTasks解析

· 第八節:Task執行

· 第九節:Task執行成功時的結果處理

· 第十節:Standalone運行模式解析

· 第十一節:Spark SQL 處理流程分析

· 第十二節:Spark SQL之SQLContext(一)

2.2 《組件解析》

· 第一節:Spark 1.5.0集群搭建

· 第二節:Hadoop、Spark生成圈簡介

· 第三節:Spark Intellij IDEA開發環境搭建

· 第四節:Spark編程模型(一)

· 第五節:Spark編程模型(二)

· 第六節:Spark編程模型(三)

· 第七節:Spark運行原理

· 第八節:Spark SQL與DataFrame(一)

· 第九節:Spark SQL運行流程解析

· 第十節:Spark SQL案例實戰(一)

· 第十一節:Spark Streaming(一)

· 第十二節:Spark Streaming(二)

· 第十三節:Spark Streaming(三)

· 第十四節:Spark Streaming(四)

· 第十五節:Kafka 0.8.2.1 集群搭建

· 第十六節:Spark Streaming與Kafka

三、應用實踐篇

3.1 《場景實踐》

· 和封神一起「深挖」Spark

· 商品搜索引擎---推薦系統設計

· 阿里Spark實踐與探索

· 如何做Spark 版本兼容

· 用線性回歸無編碼實現文章瀏覽數預測

· Spark Streaming+Spark SQL實現可配置ETL

· 2016中國spark技術峰會見聞摘要

· Spark Streaming 流式計算實戰

· 利用 Spark DataSource API 實現Rest數據源

· 協同過濾演算法多語言實現 · Spark 數據傾斜的一些表現

· Spark性能優化 · Spark性能優化——和shuffle搏鬥

· Kafka+Spark Streaming+Redis實時計算實踐

· 數據處理平台架構中的SMACK組合

3.2 《技術應用》

· SparkSQL實戰(一):DataFrames基礎

· SparkSQL實戰(二):DataFrames進階

· SparkSQL實戰(三):SparkSQL應用案例

· ML On Spark(一):基礎數據結構

· ML On Spark(二):基礎數據結構

· ML On Spark(三):統計基礎

· ML On Spark(四):統計基礎

更多:E-MapReduce實踐應用場景介紹

· 泰為基於EMR的考量與實踐

· 基於E-MapReduce梨視頻推薦系統

· Databricks、Intel、阿里、梨視頻的實踐

· E-MapReduce集群搭建HAWQ實踐

· E-MapReduce支持計算與存儲分離,成本降1倍

· 阿里雲MongoDB與EMR的HelloWorld

· E-MapReduce(Hadoop)10大類問題集群規劃

· 雲上Hadoop之優勢

點擊此處查看課程詳細內容

更多技術乾貨歡迎點擊關注知乎機構號:阿里云云棲社區 - 匯聚阿里技術精粹


如果你只是學習怎麼用hadoop和spark單機跑就是了。如果你非要用cluster,去組一個就是了。這麼多提供cloud服務的公司呢,也不貴。國內用阿里,國外Amazon的EC2和Google的GCP都行。


1. 前言

計算機的基本工作就是處理數據,包括磁碟文件中的數據,通過網路傳輸的數據流或數據包,資料庫中的結構化數據等。隨著互聯網、物聯網等技術得到越來越廣泛的應用,數據規模不斷增加,TB、PB量級成為常態,對數據的處理已無法由單台計算機完成,而只能由多台機器共同承擔計算任務。而在分散式環境中進行大數據處理,除了與存儲系統打交道外,還涉及計算任務的分工,計算負荷的分配,計算機之間的數據遷移等工作,並且要考慮計算機或網路發生故障時的數據安全,情況要複雜得多。

舉一個簡單的例子,假設我們要從銷售記錄中統計各種商品銷售額。在單機環境中,我們只需把銷售記錄掃描一遍,對各商品的銷售額進行累加即可。如果銷售記錄存放在關係資料庫中,則更省事,執行一個SQL語句就可以了。現在假定銷售記錄實在太多,需要設計出由多台計算機來統計銷售額的方案。為保證計算的正確、可靠、高效及方便,這個方案需要考慮下列問題:

  1. 如何為每台機器分配任務,是先按商品種類對銷售記錄分組,不同機器處理不同商品種類的銷售記錄,還是隨機向各台機器分發一部分銷售記錄進行統計,最後把各台機器的統計結果按商品種類合併?
  2. 上述兩種方式都涉及數據的排序問題,應選擇哪種排序演算法?應該在哪台機器上執行排序過程?
  3. 如何定義每台機器處理的數據從哪裡來,處理結果到哪裡去?數據是主動發送,還是接收方申請時才發送?如果是主動發送,接收方處理不過來怎麼辦?如果是申請時才發送,那發送方應該保存數據多久?
  4. 會不會任務分配不均,有的機器很快就處理完了,有的機器一直忙著?甚至,閑著的機器需要等忙著的機器處理完後才能開始執行?
  5. 如果增加一台機器,它能不能減輕其他機器的負荷,從而縮短任務執行時間?
  6. 如果一台機器掛了,它沒有完成的任務該交給誰?會不會遺漏統計或重複統計?
  7. 統計過程中,機器之間如何協調,是否需要專門的一台機器指揮調度其他機器?如果這台機器掛了呢?
  8. (可選)如果銷售記錄在源源不斷地增加,統計還沒執行完新記錄又來了,如何保證統計結果的準確性?能不能保證結果是實時更新的?再次統計時能不能避免大量重複計算?
  9. (可選)能不能讓用戶執行一句SQL就可以得到結果?

上述問題中,除了第1個外,其餘的都與具體任務無關,在其他分散式計算的場合也會遇到,而且解決起來都相當棘手。即使第1個問題中的分組、統計,在很多數據處理場合也會涉及,只是具體方式不同。如果能把這些問題的解決方案封裝到一個計算框架中,則可大大簡化這類應用程序的開發。

2004年前後,Google先後發表三篇論文分別介紹分散式文件系統GFS、並行計算模型MapReduce、非關係數據存儲系統BigTable,第一次提出了針對大數據分散式處理的可重用方案。在Google論文的啟發下,Yahoo的工程師Doug Cutting和Mike Cafarella開發了Hadoop。在借鑒和改進Hadoop的基礎上,又先後誕生了數十種應用於分散式環境的大數據計算框架。本文在參考業界慣例的基礎上,對這些框架按下列標準分類:

  1. 如果不涉及上面提出的第8、9兩個問題,則屬於批處理框架。批處理框架重點關心數據處理的吞吐量,又可分為非迭代式和迭代式兩類,迭代式包括DAG(有向無環圖)、圖計算等模型。
  2. 若針對第8個問題提出來應對方案,則分兩種情況:如果重點關心處理的實時性,則屬於流計算框架;如果側重於避免重複計算,則屬於增量計算框架。
  3. 如果重點關注的是第9個問題,則屬於互動式分析框架。

本文下面分別討論批處理、流計算、互動式分析三種類別的框架,然後簡要介紹大數據計算框架的一些發展趨勢。文章最後介紹這一領域的學習資料。

圖1. 大數據計算框架全景圖

2. 批處理框架

2.1. Hadoop

Hadoop最初主要包含分散式文件系統HDFS和計算框架MapReduce兩部分,是從Nutch中獨立出來的項目。在2.0版本中,又把資源管理和任務調度功能從MapReduce中剝離形成YARN,使其他框架也可以像MapReduce那樣運行在Hadoop之上。與之前的分散式計算框架相比,Hadoop隱藏了很多繁瑣的細節,如容錯、負載均衡等,更便於使用。

Hadoop也具有很強的橫向擴展能力,可以很容易地把新計算機接入到集群中參與計算。在開源社區的支持下,Hadoop不斷發展完善,並集成了眾多優秀的產品如非關係資料庫HBase、數據倉庫Hive、數據處理工具Sqoop、機器學習演算法庫Mahout、一致性服務軟體ZooKeeper、管理工具Ambari等,形成了相對完整的生態圈和分散式計算事實上的標準。

圖2. Hadoop生態圈(刪減版)

MapReduce可以理解為把一堆雜亂無章的數據按照某種特徵歸併起來,然後處理並得到最後的結果。基本處理步驟如下:

  1. 把輸入文件按照一定的標準分片,每個分片對應一個map任務。一般情況下,MapReduce和HDFS運行在同一組計算機上,也就是說,每台計算機同時承擔存儲和計算任務,因此分片通常不涉及計算機之間的數據複製。
  2. 按照一定的規則把分片中的內容解析成鍵值對。通常選擇一種預定義的規則即可。
  3. 執行map任務,處理每個鍵值對,輸出零個或多個鍵值對。
  4. MapReduce獲取應用程序定義的分組方式,並按分組對map任務輸出的鍵值對排序。默認每個鍵名一組。
  5. 待所有節點都執行完上述步驟後,MapReduce啟動Reduce任務。每個分組對應一個Reduce任務。
  6. 執行reduce任務的進程通過網路獲取指定組的所有鍵值對。
  7. 把鍵名相同的值合併為列表。
  8. 執行reduce任務,處理每個鍵對應的列表,輸出結果。

圖3. MapReduce處理過程

在上面的步驟中,應用程序主要負責設計map和reduce任務,其他工作均由框架負責。在定義map任務輸出數據的方式時,鍵的選擇至關重要,除了影響結果的正確性外,也決定數據如何分組、排序、傳輸,以及執行reduce任務的計算機如何分工。前面提到的商品銷售統計的例子,可選擇商品種類為鍵。MapReduce執行商品銷售統計的過程大致如下:

  1. 把銷售記錄分片,分配給多台機器。
  2. 每條銷售記錄被解析成鍵值對,其中值為銷售記錄的內容,鍵可忽略。
  3. 執行map任務,每條銷售記錄被轉換為新的鍵值對,其中鍵為商品種類,值為該條記錄中商品的銷售額。
  4. MapReduce把map任務生成的數據按商品種類排序。
  5. 待所有節點都完成排序後,MapReduce啟動reduce任務。每個商品種類對應一個reduce任務。
  6. 執行reduce任務的進程通過網路獲取指定商品種類的各次銷售額。
  7. MapReduce把同一種商品下的各次銷售額合併到列表中。
  8. 執行reduce任務,累加各次銷售額,得到該種商品的總銷售額。

上面的過程還有優化的空間。在傳輸各種商品每次的銷售額數據前,可先在map端對各種商品的銷售額進行小計,由此可大大減少網路傳輸的負荷。MapReduce通過一個可選的combine任務支持該類型的優化。

2.2. DAG模型

現在假設我們的目標更進一步,希望知道銷售得最好的前10種商品。我們可以分兩個環節來計算:

  1. 統計各種商品的銷售額。通過MapReduce實現,這在前面已經討論過。
  2. 對商品種類按銷售額排名。可以通過一個排序過程完成。假定商品種類非常多,需要通過多台計算機來加快計算速度的話,我們可以用另一個MapReduce過程來實現,其基本思路是把map和reduce分別當作小組賽和決賽,先計算各分片的前10名,匯總後再計算總排行榜的前10名。

從上面的例子可以看出,通過多個MapReduce的組合,可以表達複雜的計算問題。不過,組合過程需要人工設計,比較麻煩。另外,每個階段都需要所有的計算機同步,影響了執行效率。

為克服上述問題,業界提出了DAG(有向無環圖)計算模型,其核心思想是把任務在內部分解為若干存在先後順序的子任務,由此可更靈活地表達各種複雜的依賴關係。Microsoft Dryad、Google FlumeJava、Apache Tez是最早出現的DAG模型。Dryad定義了串接、全連接、融合等若干簡單的DAG模型,通過組合這些簡單結構來描述複雜的任務,FlumeJava、Tez則通過組合若干MapReduce形成DAG任務。

圖4. MapReduce(左)與Tez(右)

執行複雜任務時對比

MapReduce的另一個不足之處是使用磁碟存儲中間結果,嚴重影響了系統的性能,這在機器學習等需要迭代計算的場合更為明顯。加州大學伯克利分校AMP實驗室開發的Spark克服了上述問題。Spark對早期的DAG模型作了改進,提出了基於內存的分散式存儲抽象模型RDD(Resilient Distributed Datasets,可恢復分散式數據集),把中間數據有選擇地載入並駐留到內存中,減少磁碟IO開銷。與Hadoop相比,Spark基於內存的運算要快100倍以上,基於磁碟的運算也要快10倍以上。

圖5. MapReduce與Spark中間結果

保存方式對比

Spark為RDD提供了豐富的操作方法,其中map、 filter、 flatMap、 sample、groupByKey、 reduceByKey、union、join、cogroup、mapValues、sort、partionBy用於執行數據轉換,生成新的RDD,而count、collect、 reduce、lookup、save用於收集或輸出計算結果。如前面統計商品銷售額的例子,在Spark中只需要調用map和reduceByKey兩個轉換操作就可以實現,整個程序包括載入銷售記錄和保存統計結果在內也只需要寥寥幾行代碼,並且支持Java、Scala、Python、R等多種開發語言,比MapReduce編程要方便得多。下圖說明reduceByKey的內部實現。

圖6. RDD reduceByKey內部實現

RDD由於把數據存放在內存中而不是磁碟上,因此需要比Hadoop更多地考慮容錯問題。分散式數據集的容錯有兩種方式:數據檢查點和記錄數據的更新。處理海量數據時,數據檢查點操作成本很高, 因此Spark默認選擇記錄更新的方式。不過如果更新粒度太細太多,記錄更新成本也不低。因此,RDD只支持粗粒度轉換,即只記錄單個塊上執行的單個操作,然後將創建RDD的一系列變換序列記錄下來,類似於資料庫中的日誌。

當RDD的部分分區數據丟失時,Spark根據之前記錄的演變過程重新運算,恢復丟失的數據分區。Spark生態圈的另一項目Alluxio(原名Tachyon)也採用類似的思路,使數據寫入速度比HDFS有數量級的提升。

下面總結Spark對MapReduce的改進:

  • MapReduce抽象層次低,需要手工編寫代碼完成;Spark基於RDD抽象,使數據處理邏輯的代碼非常簡短。
  • MapReduce只提供了map和reduce兩個操作,表達力欠缺;Spark提供了很多轉換和動作,很多關係資料庫中常見的操作如JOIN、GROUP BY已經在RDD中實現。
  • MapReduce中,只有map和reduce兩個階段,複雜的計算需要大量的組合,並且由開發者自己定義組合方式;Spark中,RDD可以連續執行多個轉換操作,如果這些操作對應的RDD分區不變的話,還可以放在同一個任務中執行。
  • MapReduce處理邏輯隱藏在代碼中,不直觀;Spark代碼不包含操作細節,邏輯更清晰。
  • MapReduce中間結果放在HDFS中;Spark中間結果放在內存中,內存放不下時才寫入本地磁碟而不是HDFS,這顯著提高了性能,特別是在迭代式數據處理的場合。
  • MapReduce中,reduce任務需要等待所有map任務完成後才可以開始;在Spark中,分區相同的轉換構成流水線放到同一個任務中運行。

3. 流計算框架

3.1. 流計算概述

在大數據時代,數據通常都是持續不斷動態產生的。在很多場合,數據需要在非常短的時間內得到處理,並且還要考慮容錯、擁塞控制等問題,避免數據遺漏或重複計算。流計算框架則是針對這一類問題的解決方案。流計算框架一般採用DAG(有向無環圖)模型。圖中的節點分為兩類:一類是數據的輸入節點,負責與外界交互而向系統提供數據;另一類是數據的計算節點,負責完成某種處理功能如過濾、累加、合併等。從外部系統不斷傳入的實時數據則流經這些節點,把它們串接起來。如果把數據流比作水的話,輸入節點好比是噴頭,源源不斷地出水,計算節點則相當於水管的轉介面。如下圖所示。

圖7. 流計算DAG模型示意圖

為提高並發性,每一個計算節點對應的數據處理功能被分配到多個任務(相同或不同計算機上的線程)。在設計DAG時,需要考慮如何把待處理的數據分發到下游計算節點對應的各個任務,這在實時計算中稱為分組(Grouping)。最簡單的方案是為每個任務複製一份,不過這樣效率很低,更好的方式是每個任務處理數據的不同部分。隨機分組能達到負載均衡的效果,應優先考慮。不過在執行累加、數據關聯等操作時,需要保證同一屬性的數據被固定分發到對應的任務,這時應採用定向分組。在某些情況下,還需要自定義分組方案。

圖8. 流計算分組

由於應用場合的廣泛性,目前市面上已經有不少流計算平台,包括Google MillWheel、Twitter Heron和Apache項目Storm、Samza、S4、Flink、Apex、Gearpump。

3.2. Storm及Trident

在流計算框架中,目前人氣最高,應用最廣泛的要數Storm。這是由於Storm具有簡單的編程模型,且支持Java、Ruby、Python等多種開發語言。Storm也具有良好的性能,在多節點集群上每秒可以處理上百萬條消息。Storm在容錯方面也設計得很優雅。下面介紹Storm確保消息可靠性的思路。

在DAG模型中,確保消息可靠的難點在於,原始數據被當前的計算節點成功處理後,還不能被丟棄,因為它生成的數據仍然可能在後續的計算節點上處理失敗,需要由該消息重新生成。而如果要對消息在各個計算節點的處理情況都作跟蹤記錄的話,則會消耗大量資源。

Storm的解決思路,是為每條消息分派一個ID作為唯一性標識,並在消息中包含原始輸入消息的ID。同時用一個響應中心(Acker)維護每條原始輸入消息的狀態,狀態的初值為該原始輸入消息的ID。每個計算節點成功執行後,則把輸入和輸出消息的ID進行異或,再異或對應的原始輸入消息的狀態。由於每條消息在生成和處理時分別被異或一次,則成功執行後所有消息均被異或兩次,對應的原始輸入消息的狀態為0。因此當狀態為0後可安全清除原始輸入消息的內容,而如果超過指定時間間隔後狀態仍不為0,則認為處理該消息的某個環節出了問題,需要重新執行。

圖9. Storm保證消息可靠性過程示意圖

Storm還實現了更高層次的抽象框架Trident。Trident以微批處理的方式處理數據流,比如每次處理100條記錄。Trident提供了過濾、分組、連接、窗口操作、聚合、狀態管理等操作,支持跨批次進行聚合處理,並對執行過程進行優化,包括多個操作的合併、數據傳輸前的本地聚合等。以微批處理方式處理數據流的框架還有Spark Streaming。

(1) 實時流處理

(2) 微批處理

圖10. 實時流處理與微批處理比較

下面是Storm、Trident與另外幾種流計算框架的對比:

4. 互動式分析框架

4.1. 概述

在解決了大數據的可靠存儲和高效計算後,如何為數據分析人員提供便利日益受到關注,而最便利的分析方式莫過於互動式查詢。這幾年互動式分析技術發展迅速,目前這一領域知名的平台有十餘個,包括Google開發的Dremel和PowerDrill,Facebook開發的Presto, Hadoop服務商Cloudera和HortonWorks分別開發的Impala和Stinger,以及Apache項目Hive、Drill、Tajo、Kylin、MRQL等。

一些批處理和流計算平台如Spark和Flink也分別內置了互動式分析框架。由於SQL已被業界廣泛接受,目前的互動式分析框架都支持用類似SQL的語言進行查詢。早期的互動式分析平台建立在Hadoop的基礎上,被稱作SQL-on-Hadoop。後來的分析平台改用Spark、Storm等引擎,不過SQL-on-Hadoop的稱呼還是沿用了下來。SQL-on-Hadoop也指為分散式數據存儲提供SQL查詢功能。

4.2. Hive

Apache Hive是最早出現的架構在Hadoop基礎之上的大規模數據倉庫,由Facebook設計並開源。Hive的基本思想是,通過定義模式信息,把HDFS中的文件組織成類似傳統資料庫的存儲系統。Hive 保持著 Hadoop 所提供的可擴展性和靈活性。Hive支持熟悉的關係資料庫概念,比如表、列和分區,包含對非結構化數據一定程度的 SQL 支持。它支持所有主要的原語類型(如整數、浮點數、字元串)和複雜類型(如字典、列表、結構)。它還支持使用類似 SQL 的聲明性語言 Hive Query Language (HiveQL) 表達的查詢,任何熟悉 SQL 的人都很容易理解它。HiveQL被編譯為MapReduce過程執行。下圖說明如何通過MapReduce實現JOIN和GROUP BY。

(1) 實現JOIN

(2) 實現GROUP BY

圖11. 部分HiveQL操作的實現方式

Hive與傳統關係資料庫對比如下:

Hive的主要弱點是由於建立在MapReduce的基礎上,性能受到限制。很多互動式分析平台基於對Hive的改進和擴展,包括Stinger、Presto、Kylin等。其中Kylin是中國團隊提交到Apache上的項目,其與眾不同的地方是提供多維分析(OLAP)能力。Kylin對多維分析可能用到的度量進行預計算,供查詢時直接訪問,由此提供快速查詢和高並發能力。Kylin在eBay、百度、京東、網易、美團均有應用。

4.3. SQL引擎Calcite

對於互動式分析,SQL查詢引擎的優劣對性能的影響舉足輕重。Spark開發了自己的查詢引擎Catalyst,而包括Hive、Drill、Kylin、Flink在內的很多互動式分析平台及數據倉庫使用Calcite(原名optiq)作為SQL引擎。Calcite是一個Apache孵化項目,其創建者Julian Hyde曾是Oracle資料庫SQL引擎的主要開發者。Calcite具有下列幾個技術特點:

  • 支持標準SQL語言。
  • 支持OLAP。
  • 支持對流數據的查詢。
  • 獨立於編程語言和數據源,可以支持不同的前端和後端。
  • 支持關係代數、可定製的邏輯規劃規則和基於成本模型優化的查詢引擎。
  • 支持物化視圖(materialized view)的管理。

由於分散式場景遠比傳統的數據存儲環境更複雜,Calcite和Catalyst都還處於向Oracle、MySQL等經典關係資料庫引擎學習的階段,在性能優化的道路上還有很長的路要走。

5. 其他類型的框架

除了上面介紹的幾種類型的框架外,還有一些目前還不太熱門但具有重要潛力的框架類型。圖計算是DAG之外的另一種迭代式計算模型,它以圖論為基礎對現實世界建模和計算,擅長表達數據之間的關聯性,適用於PageRank計算、社交網路分析、推薦系統及機器學習。這一類框架有Google Pregel、Apache Giraph、Apache Hama、PowerGraph、,其中PowerGraph是這一領域目前最傑出的代表。很多圖資料庫也內置圖計算框架。

另一類是增量計算框架,探討如何只對部分新增數據進行計算來極大提升計算過程的效率,可應用到數據增量或周期性更新的場合。這一類框架包括Google Percolator、Microsoft Kineograph、阿里Galaxy等。

另外還有像Apache Ignite、Apache Geode(GemFire的開源版本)這樣的高性能事務處理框架。

6. 總結與展望

從Hadoop橫空出世到現在10餘年的時間中,大數據分散式計算技術得到了迅猛發展。不過由於歷史尚短,這方面的技術遠未成熟。各種框架都還在不斷改進,並相互競爭。

性能優化毫無疑問是大數據計算框架改進的重點方向之一。而性能的提高很大程度上取決於內存的有效利用。這包括前面提到的內存計算,現已在各種類型的框架中廣泛採用。內存資源的分配管理對性能也有重要影響,JVM垃圾回收在給開發人員帶來便利的同時,也制約了內存的有效利用。另外,Java的對象創建及序列化也比較浪費資源。在內存優化方面做足功夫的代表是Flink。出於性能方面的考慮,Flink很多組件自行管理內存,無需依賴JVM垃圾回收機制。Flink還用到開闢內存池、用二進位數據代替對象、量身定製序列化、定製緩存友好的演算法等優化手段。Flink還在任務的執行方面進行優化,包括多階段並行執行和增量迭代。

擁抱機器學習和人工智慧也是大數據計算的潮流之一。Spark和Flink分別推出機器學習庫Spark ML和Flink ML。更多的平台在第三方大數據計算框架上提供機器學習,如Mahout、Oryx及一干Apache孵化項目SystemML、HiveMall、PredictionIO、SAMOA、MADLib。這些機器學習平台一般都同時支持多個計算框架,如Mahout同時以Spark、Flink、H2O為引擎,SAMOA則使用S4、Storm、Samza。在深度學習掀起熱潮後,又有社區探索把深度學習框架與現有分散式計算框架結合起來,這樣的項目有SparkNet、Caffe on Spark、TensorFrames等。

在同一平台上支持多種框架也是發展趨勢之一,尤其對於那些開發實力較為雄厚的社區。Spark以批處理模型為核心,實現了互動式分析框架Spark SQL、流計算框架Spark Streaming(及正在實現的Structured Streaming)、圖計算框架GraphX、機器學習庫Spark ML。而Flink在提供低延遲的流計算的同時,批處理、關係計算、圖計算、機器學習,一個也沒落下,目標直奔大數據通用計算平台。Google的BEAM(意為Batch+strEAM)則試圖把Spark、Flink、Apex這樣的計算框架納入自己制定的標準之下,頗有號令江湖之意。

圖12. BEAM的統一模型

7. 學習資料

最後介紹一下大數據計算方面的學習資料。入門前的了解、知識面的拓展及知識的零散積累靠長期訪問相關的網站、論壇、微信訂閱號,問題解答則靠對搜索引擎的熟練駕馭。需要指出的是,網上的內容良萎不齊,很多資料是過時的,以訛傳訛也是常有的事,要注意鑒別。

論壇首推知乎、Quora、Stack Overflow,運氣好的話開發者親自給你解答。其他值得關注的網站或論壇包括煉數成金、人大經濟論壇、CSDN、博客園、雲棲社區、360大數據、推酷、伯樂在線、小象學院等。微信訂閱號中,InfoQ是最權威的,其他還有THU數據派、大數據雜談、CSDN大數據、數據猿、Hadoop技術博文等,各人根據偏好取捨。

若要進行系統的學習,則首先應參考官方網站文檔。不少大數據平台的官方文檔內容都比較詳實,勝過多數教材。另外,官方文檔與產品通常同步更新,這個優勢是其他資料無法做到的。不過要說可讀性,書籍或視頻教程要強得多。視頻資料可以從上文提到的部分網站論壇下載。

書籍方面,國外O"Reilly、Manning兩家出版社在大數據領域出版了不少優秀書籍,特別是Manning的In Action系列和O"Reilly的Definitive Guide系列。前者側重提高動手能力,後者則知識比較全面。In Action和Definitive Guide系列的書籍很多已翻譯為中文,一般分別譯為xxx實戰、xxx權威指南。另外一家出版社Packt也值得關注。Packt的書比較薄,適合入門。至於中文原創書籍,推薦張俊林的《大數據日知錄》,該書是對大數據存儲和處理技術的全面梳理,系統性強。其他書籍不逐一點評,若想購買或閱讀可參考豆瓣對該書的評分。

圖13. 部分推薦書籍

對希望對大數據框架內部機制有深入的理解的讀者,建議首先檢索相關論文來閱讀。

Google的那幾篇論文這裡就不一一列出了,網上很容易搜到。其他推薦的論文如下:

作者:王小鑒


沒有集群環境,可以單機跑應用,但是還是沒有解決學習大數據平台處理框架的問題。

我覺得有幾個方面,可以給你參考:

1、你需要一個比較強勁的機器,內存/CPU要稍大一些,這些大數據的傢伙都是吃資源的;

2、你可以選擇採用虛擬化技術,比如VMware,VirtualBox,多跑幾個linux,應該問題不大;

3、你還可以選擇最近比較流行的Docker技術,很大程度上比虛擬化要便捷的多;

4、你還可以很土豪,買很多實體機,選擇用Ambari搭建一個真實的hadoop環境,我覺得那樣你會學得更快。


自己電腦建三四個虛擬機


首先要明確你要學習什麼知識,學習這些知識是用來解決什麼問題?帶著問題去學習!例如你要做web日誌分析:數據從哪裡來,存儲到哪裡,如何分析,達到什麼目的,如何展現,……


那些說單機的,誰都知道可以單機跑,但是,在單機和集群搭建一個可以使用的環境是完全不一樣的,而且,有很多bug在單機環境下是無法觸發的。所以很多時候你在單機上能跑的代碼在集群上是會不斷報錯的。學習不是只是知道的就行。


我當時是用虛擬機,8G內存分四台虛擬機,性能已經無力吐槽,不過練練mapreduce還是可以的


沒有集群環境,不可能學好大數據處理框架(Hadoop,Spark)的相關知識。


可以單機跑,推薦intellij。

其實自學更應該關心的問題是做什麼練手項目


ucloud和百度雲都能買彈性計算集群,按分鐘計費


最好還是單個環境動手玩玩,偽分散式的就可以了。這裡有個大數據技術博客:https://www.iteblog.com/


你是想學習搭建環境吧,我自己有個學習網址,我就是按照這個學習的,成功搭建了,你可以試試。Hadoop集群安裝配置教程_Hadoop2.6.0_Ubuntu/CentOS_廈大資料庫實驗室博客


vmware can help you!


推薦閱讀:

大數據工程師需要學習哪些?
國家推行某一項改革後,怎麼分析這項改革的經濟效果?畢竟經濟社會又不能設對照組,經濟學有沒有相關方法?
用計算機和大數據進行社會科學研究會成為未來主流嗎?
海量日誌數據存儲用 elasticsearch 和 hbase 哪個好?

TAG:Hadoop | 數據處理 | 大數據 | 大數據處理 | Spark |