原文:http://www.52cs.org/?p=429node
做者:陳天奇,畢業於上海交通大學ACM班,現就讀於華盛頓大學,從事大規模機器學習研究。git
註解:truth4sex github
編者按:本文是對開源xgboost庫理論層面的介紹,在陳天奇原文《梯度提高法和Boosted Tree》的基礎上,作了以下註解:1)章節劃分;2)註解和參考連接(以藍色和紅色字體標註)。備註:圖片可點擊查看清晰版。算法
1. 前言
應 @龍星鏢局 兄邀請寫這篇文章。做爲一個很是有效的機器學習方法,Boosted Tree是數據挖掘和機器學習中最經常使用的算法之一。由於它效果好,對於輸入要求不敏感,每每是從統計學家到數據科學家必備的工具之一,它同時也是kaggle比賽冠軍選手最經常使用的工具。最後,由於它的效果好,計算複雜度不高,也在工業界中有大量的應用。機器學習
2. Boosted Tree的若干同義詞
說到這裏可能有人會問,爲何我沒有聽過這個名字。這是由於Boosted Tree有各類馬甲,好比GBDT, GBRT (gradient boosted regression tree),MART,LambdaMART也是一種boosted tree的變種。網上有不少介紹Boosted tree的資料,不過大部分都是基於Friedman的最先一篇文章Greedy Function Approximation: A Gradient Boosting Machine的翻譯。我的以爲這不是最好最通常地介紹boosted tree的方式。而網上除了這個角度以外的介紹並很少。這篇文章是我我的對於boosted tree和gradient boosting 類算法的總結,其中不少材料來自於我TA UW機器學習時的一份講義。分佈式
3. 有監督學習算法的邏輯組成
要講boosted tree,要先從有監督學習講起。在有監督學習裏面有幾個邏輯上的重要組成部件,初略地分能夠分爲:模型,參數 和 目標函數。ide
i. 模型和參數
模型指給定輸入如何去預測 輸出 。咱們比較常見的模型如線性模型(包括線性迴歸和logistic regression)採用了線性疊加的方式進行預測 。其實這裏的預測能夠有不一樣的解釋,好比咱們能夠用它來做爲迴歸目標的輸出,或者進行sigmoid 變換獲得機率,或者做爲排序的指標等。而一個線性模型根據的解釋不一樣(以及設計對應的目標函數)用到迴歸,分類或排序等場景。參數指咱們須要學習的東西,在線性模型中,參數指咱們的線性係數。函數
ii. 目標函數:損失 + 正則
模型和參數自己指定了給定輸入咱們如何作預測,可是沒有告訴咱們如何去尋找一個比較好的參數,這個時候就須要目標函數登場了。通常的目標函數包含下面兩項工具
常見的偏差函數有 好比平方偏差 ,logistic偏差函數( )等。而對於線性模型常見的正則化項有正則和正則。這樣目標函數的設計來自於統計學習裏面的一個重要概念叫作Bias-variance tradeoff。比較感性的理解,Bias能夠理解爲假設咱們有無限多數據的時候,能夠訓練出最好的模型所拿到的偏差。而Variance是由於咱們只有有限數據,其中隨機性帶來的偏差。目標中偏差函數鼓勵咱們的模型儘可能去擬合訓練數據,這樣相對來講最後的模型會有比較少的 bias。而正則化項則鼓勵更加簡單的模型。由於當模型簡單以後,有限數據擬合出來結果的隨機性比較小,不容易過擬合,使得最後模型的預測更加穩定。學習
iii. 優化算法
講了這麼多有監督學習的基本概念,爲何要講這些呢? 是由於這幾部分包含了機器學習的主要成分,也是機器學習工具設計中劃分模塊比較有效的辦法。其實這幾部分以外,還有一個優化算法,就是給定目標函數以後怎麼學的問題。之因此我沒有講優化算法,是由於這是你們每每比較熟悉的「機器學習的部分」。而有時候咱們每每只知道「優化算法」,而沒有仔細考慮目標函數的設計的問題,比較常見的例子如決策樹的學習,你們知道的算法是每一步去優化gini entropy,而後剪枝,可是沒有考慮到後面的目標是什麼。
4. Boosted Tree
i. 基學習器:分類和迴歸樹(CART)
話題回到boosted tree,咱們也是從這幾個方面開始講,首先講模型。Boosted tree 最基本的組成部分叫作迴歸樹(regression tree),也叫作CART。
上面就是一個CART的例子。CART會把輸入根據輸入的屬性分配到各個葉子節點,而每一個葉子節點上面都會對應一個實數分數。上面的例子是一個預測一我的是否會喜歡電腦遊戲的 CART,你能夠把葉子的分數理解爲有多可能這我的喜歡電腦遊戲。有人可能會問它和decision tree的關係,其實咱們能夠簡單地把它理解爲decision tree的一個擴展。從簡單的類標到分數以後,咱們能夠作不少事情,如機率預測,排序。
ii. Tree Ensemble
一個CART每每過於簡單沒法有效地預測,所以一個更增強力的模型叫作tree ensemble。
在上面的例子中,咱們用兩棵樹來進行預測。咱們對於每一個樣本的預測結果就是每棵樹預測分數的和。到這裏,咱們的模型就介紹完畢了。如今問題來了,咱們常見的隨機森林和boosted tree和tree ensemble有什麼關係呢?若是你仔細的思考,你會發現RF和boosted tree的模型都是tree ensemble,只是構造(學習)模型參數的方法不一樣。第二個問題:在這個模型中的「參數」是什麼。在tree ensemble中,參數對應了樹的結構,以及每一個葉子節點上面的預測分數。
最後一個問題固然是如何學習這些參數。在這一部分,答案可能千奇百怪,可是最標準的答案始終是一個:定義合理的目標函數,而後去嘗試優化這個目標函數。在這裏我要多說一句,由於決策樹學習每每充滿了heuristic。 如先優化吉尼係數,而後再剪枝啦,限制最大深度,等等。其實這些heuristic的背後每每隱含了一個目標函數,而理解目標函數自己也有利於咱們設計學習算法,這個會在後面具體展開。
對於tree ensemble,咱們能夠比較嚴格的把咱們的模型寫成是:
其中每一個是一個在函數空間()裏面的函數,而對應了全部regression tree的集合。咱們設計的目標函數也須要遵循前面的主要原則,包含兩部分
iii. 模型學習:additive training
其中第一部分是訓練偏差,也就是你們相對比較熟悉的如平方偏差, logistic loss等。而第二部分是每棵樹的複雜度的和。這個在後面會繼續講到。由於如今咱們的參數能夠認爲是在一個函數空間裏面,咱們不能採用傳統的如SGD之類的算法來學習咱們的模型,所以咱們會採用一種叫作additive training的方式(另外,在我我的的理解裏面,boosting就是指additive training的意思)。每一次保留原來的模型不變,加入一個新的函數$f$到咱們的模型中。
如今還剩下一個問題,咱們如何選擇每一輪加入什麼呢?答案是很是直接的,選取一個來使得咱們的目標函數儘可能最大地下降。
這個公式可能有些過於抽象,咱們能夠考慮當是平方偏差的狀況。這個時候咱們的目標能夠被寫成下面這樣的二次函數:
更加通常的,對於不是平方偏差的狀況,咱們會採用以下的泰勒展開近似來定義一個近似的目標函數,方便咱們進行這一步的計算。
當咱們把常數項移除以後,咱們會發現以下一個比較統一的目標函數。這一個目標函數有一個很是明顯的特色,它只依賴於每一個數據點的在偏差函數上的一階導數和二階導數。有人可能會問,這個材料彷佛比咱們以前學過的決策樹學習難懂。爲何要花這麼多力氣來作推導呢?
由於這樣作使得咱們能夠很清楚地理解整個目標是什麼,而且一步一步推導出如何進行樹的學習。這一個抽象的形式對於實現機器學習工具也是很是有幫助的。傳統的GBDT可能你們能夠理解如優化平法殘差,可是這樣一個形式包含可全部能夠求導的目標函數。也就是說有了這個形式,咱們寫出來的代碼能夠用來求解包括迴歸,分類和排序的各類問題,正式的推導可使得機器學習的工具更加通常。
iv. 樹的複雜度
到目前爲止咱們討論了目標函數中訓練偏差的部分。接下來咱們討論如何定義樹的複雜度。咱們先對於f的定義作一下細化,把樹拆分紅結構部分和葉子權重部分。下圖是一個具體的例子。結構函數把輸入映射到葉子的索引號上面去,而給定了每一個索引號對應的葉子分數是什麼。
當咱們給定了如上定義以後,咱們能夠定義一棵樹的複雜度以下。這個複雜度包含了一棵樹裏面節點的個數,以及每一個樹葉子節點上面輸出分數的$L2$模平方。固然這不是惟一的一種定義方式,不過這必定義方式學習出的樹效果通常都比較不錯。下圖還給出了複雜度計算的一個例子。
v. 關鍵步驟
接下來是最關鍵的一步,在這種新的定義下,咱們能夠把目標函數進行以下改寫,其中I被定義爲每一個葉子上面樣本集合 = { }
這一個目標包含了個相互獨立的單變量二次函數。咱們能夠定義
那麼這個目標函數能夠進一步改寫成以下的形式,假設咱們已經知道樹的結構,咱們能夠經過這個目標函數來求解出最好的,以及最好的對應的目標函數最大的增益
這兩個的結果對應以下,左邊是最好的,右邊是這個對應的目標函數的值。到這裏你們可能會以爲這個推導略複雜。其實這裏只涉及到了如何求一個一維二次函數的最小值的問題。若是以爲沒有理解不妨再仔細琢磨一下
vi. 打分函數計算舉例
Obj表明了當咱們指定一個樹的結構的時候,咱們在目標上面最多減小多少。咱們能夠把它叫作結構分數(structure score)。你能夠認爲這個就是相似吉尼係數同樣更加通常的對於樹結構進行打分的函數。下面是一個具體的打分函數計算的例子
vii. 枚舉全部不一樣樹結構的貪心法
因此咱們的算法也很簡單,咱們不斷地枚舉不一樣樹的結構,利用這個打分函數來尋找出一個最優結構的樹,加入到咱們的模型中,再重複這樣的操做。不過枚舉全部樹結構這個操做不太可行,因此經常使用的方法是貪心法,每一次嘗試去對已有的葉子加入一個分割。對於一個具體的分割方案,咱們能夠得到的增益能夠由以下公式計算
對於每次擴展,咱們仍是要枚舉全部可能的分割方案,如何高效地枚舉全部的分割呢?我假設咱們要枚舉全部 這樣的條件,對於某個特定的分割咱們要計算左邊和右邊的導數和。
咱們能夠發現對於全部的,咱們只要作一遍從左到右的掃描就能夠枚舉出全部分割的梯度和和。而後用上面的公式計算每一個分割方案的分數就能夠了。
觀察這個目標函數,你們會發現第二個值得注意的事情就是引入分割不必定會使得狀況變好,由於咱們有一個引入新葉子的懲罰項。優化這個目標對應了樹的剪枝, 當引入的分割帶來的增益小於一個閥值的時候,咱們能夠剪掉這個分割。你們能夠發現,當咱們正式地推導目標的時候,像計算分數和剪枝這樣的策略都會天然地出現,而再也不是一種由於heuristic而進行的操做了。
講到這裏文章進入了尾聲,雖然有些長,但願對你們有所幫助,這篇文章介紹瞭如何經過目標函數優化的方法比較嚴格地推導出boosted tree的學習。由於有這樣通常的推導,獲得的算法能夠直接應用到迴歸,分類排序等各個應用場景中去。
5. 尾聲:xgboost
這篇文章講的全部推導和技術都指導了xgboost https://github.com/dmlc/xgboost 的設計。xgboost是大規模並行boosted tree的工具,它是目前最快最好的開源boosted tree工具包,比常見的工具包快10倍以上。在數據科學方面,有大量kaggle選手選用它進行數據挖掘比賽,其中包括兩個以上kaggle比賽的奪冠方案。在工業界規模方面,xgboost的分佈式版本有普遍的可移植性,支持在YARN, MPI, Sungrid Engine等各個平臺上面運行,而且保留了單機並行版本的各類優化,使得它能夠很好地解決於工業界規模的問題。有興趣的同窗能夠嘗試使用一下,也歡迎貢獻代碼。
-----------------------------
註解和連接: