機器學習實戰筆記1)K-近鄰算法:實戰篇

問題
為了提高我的B格,我打算買瓶紅酒嘗嘗。但是我對紅酒一竅不通,不知道該如何鑒別紅酒質量的好壞。于是突發奇想,能否使用K-近鄰算法來幫助我選擇呢?

數據準備
我在網上下載了紅酒的數據集。該數據集收集了紅酒的11種特征及專家對每種酒的評價。這是數據的下載地址,下面列舉了部分樣本數據:

|序號|非揮發性酸度|揮發性酸度|檸檬酸|殘留糖|氯化物|游離二氧化硫|總二氧化硫|濃度|PH值|硫酸鹽|酒精度|質量|
|---|
|1|7|0.27|0.36|20.7|0.045|45|170|1.001|3|0.45|8.8|6|
|2|6.3|0.3|0.34|1.6|0.049|14|132|0.994|3.3|0.49|9.5|6|
|3|7.9|0.18|0.37|1.2|0.04|16|75|0.992|3.18|0.63|10.8|5|

數據的處理一共包括兩個步驟:讀取數據及數據預處理。讀取數據部分將數據分成紅酒數據部分及標簽部分。數據預處理主要是對數據部分進行歸一化處理。代碼如下:

import numpy as np
import csv

def generateData(filename):
    data = []
    with open(filename, 'r') as csvfile:
    reader = csv.reader(csvfile, delimiter=';')
    data = [row for row in reader]
    data = data[1:]

data = np.array(data)

label = data[:,-1]
data = data[:,:-1]

data = data.astype(np.float)
label = label.astype(np.float)

#normlize to [0,1]
minVals = data.min(0)
maxVals = data.max(0)
data = (data-minVals)/(maxVals-minVals)

return data, label

K-近鄰算法
下面來寫K-近鄰算法的核心算法,算法的流程大概如下:首先計算測試樣本與所有訓練樣本之間的距離,選出距離最近的K個樣本,將這K個樣本中標簽最多的標簽作為這個測試樣本的標簽,代碼如下:

 import numpy as np
 import csv
 import operator

def knnclassify(testData, trainData, trainLabel,k):
    diff = trainData - testData
    diffSq = diff**2
    distances = np.sqrt(np.sum(diffSq, axis = 1))
    sortedDistdicies = distances.argsort()
    classCount = {}
  for i in range(k):
      votedLabel = trainLabel[sortedDistdicies[i]]
      classCount[votedLabel] = classCount.get(votedLabel, 0) + 1
  sortedClassCount = sorted(classCount.iteritems(),key=operator.itemgetter(1), reverse=True)
  return sortedClassCount[0][0]

現在我買了一瓶紅酒,并記錄下它的各種特征,判斷它等級的代碼如下:

testData = np.array([0.2,0.3,0.11,0.13,0.16,0.2,0.3,0.11,0.13,0.16,0.27])
data, label = generateData('winequality-red.csv')

k=9
level = knnclassify(testData, data, label, k)
print level

從結果可以看出,我們這瓶酒的等級為5,屬于中等紅酒。對于我這種level來說,應該足夠了。

評價分類器 在我品嘗這瓶紅酒的時候,我心中不禁有點不安:如果這個K-近鄰分類方法“騙了”我怎么辦? 作為一個嚴謹的人,我一定得找到一個方法來檢驗這個分類器的效度。

驗證分類器的方法非常簡單,首先我們將之前的數據一分為二,一份作為訓練集,一份作為測試集。用訓練集來預測測試集,因為我們已經知道測試集的真正等級,所以,我們可以通過比較我們的預測結果與真實結果之間的差異來評估分類器的效度。測試的代碼為:

data, label = generateData('winequality-red.csv')
ratio = 0.1
k = 9
m = int(ratio*data.shape[0])
testData = data[:m];
trainData = data[m:]
testLabel = label[:m]
trainLabel = label[m:]

errorCount = 0
for i in range(testData.shape[0]):
    predict = knnclassify(testData[i], trainData, trainLabel, k)
    print "the predict is %f, the ground truth is %f" %(predict, testLabel[i])
    if(predict != testLabel[i]):
        errorCount = errorCount + 1;

print "the total error rate is %f" %(errorCount/float(testData.shape[0]))

最后輸出的錯誤率為0.39。還是比較可信,至少是個及格的分數。在后面的文章中,我會嘗試使用其他的分類方法,以期達到更好的結果。

關于K的選擇
在前面的代碼中,K的值我都取做9,但是在實踐中,我們需要通過實驗得出K的最佳取值。操作也非常簡單,就是K取不同的值,然后得出不同的錯誤率,取錯誤率最低的那個K值作為我們最終的選擇。

本篇文章的完整代碼可以在這里下載

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,818評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,185評論 3 414
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,656評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,647評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,446評論 6 405
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 54,951評論 1 321
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,041評論 3 440
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,189評論 0 287
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,718評論 1 333
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,602評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,800評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,316評論 5 358
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,045評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,419評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,671評論 1 281
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,420評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,755評論 2 371

推薦閱讀更多精彩內容