机器学习实战学习笔记(二)-KNN算法(2)-KNN算法改进约会网站的配对效果

机器学习实战学习笔记(二)-KNN算法(2)-KNN算法改进约会网站的配对效果

情景概要

某个妹子交往过三种类型的人:

  • 不喜欢的人
  • 魅力一般的人.
  • 极具魅力的人

这个妹子想要知道自己到底喜欢哪一类男人,于是提供了她收集的约会数据(1000行,吐槽一波,手动狗头),并希望能创建一种分类机制来帮她完成这件事情。

数据表格如下:

实际数据集是这样的:

datingTestSet.txt

datingTestSet2.txt

导入数据

# 判断分类
def isWhichClass(className):
    if className=='didntLike':
        return 1
    elif className=='smallDoses':
        return 2
    elif className=='largeDoses':
        return 3

#加载数据

def file2matrix(filename):
    fr=open(filename)
    arrayOLines=fr.readlines()
    numbersOfLines=len(arrayOLines)#获取一共有多少行
    returnMat=np.zeros((numbersOfLines,3))#创建n行3列的全零矩阵
    classLabelVector=[]#标签向量
    index=0
    for line in arrayOLines:
        line=line.strip()
        listFormLine=line.split('\t')
        returnMat[index,:]=listFormLine[0:3]
        classLabelVector.append(isWhichClass(listFormLine[-1]))
        index+=1
    return returnMat,classLabelVector

数据可视化

#绘图
def makeGraph():
    returnMat,classLabelVector=file2matrix('KNN\datingTestSet.txt')
    fig=plt.figure()
    ax=fig.add_subplot(111)
    x1=[x[1] for i,x in enumerate(returnMat) if classLabelVector[i]==1]
    y1=[x[2] for i,x in enumerate(returnMat) if classLabelVector[i]==1]
    x2=[x[1] for i,x in enumerate(returnMat) if classLabelVector[i]==2]
    y2=[x[2] for i,x in enumerate(returnMat) if classLabelVector[i]==2]
    x3=[x[1] for i,x in enumerate(returnMat) if classLabelVector[i]==3]
    y3=[x[2] for i,x in enumerate(returnMat) if classLabelVector[i]==3]
    type1=plt.scatter(x1,y1,c="red",cmap='brg',alpha=0.2, marker='8', linewidth=0)
    type2=plt.scatter(x2,y2,c="green",cmap='brg',alpha=0.2, marker='8', linewidth=0)
    type3=plt.scatter(x3,y3,c="blue",cmap='brg', alpha=0.2, marker='8', linewidth=0)
    ax.legend([type1, type2, type3], ["Did Not Like", "Liked in Small Doses", "Liked in Large Doses"], loc=2)
    mpl.rcParams['font.sans-serif']=['SimHei'] #指定默认字体 SimHei为黑体
    mpl.rcParams['axes.unicode_minus']=False #用来正常显示负号
    plt.xlabel('玩游戏时间所占百分比', fontsize=14)
    plt.ylabel('每周消耗的冰激凌的公斤数', fontsize=14)
    plt.show()

归一化数据

原理:

为什么要归一化?

数字差值最大的属性对计算结果的影响最大,也就是说,但这三种特征是同等重要的,因此要归一化

归一化方法

在处理这种不同取值范围的特征值时,我们通常采用的方法是将数值归一化,如将取值范围
处理为0到1或者-1到1之间。下面的公式可以将任意取值范围的特征值转化为0到1区间内的值:

newvalue=(o1dValue -min) / (max- min)

其中min和max分别是数据集中的最小特征值和最大特征值。

# 自动归一化
def autoNorm(dataSet):
    minVal=dataSet.min(0)
    maxVal=dataSet.max(0)
    ranges=maxVal-minVal
    normDataSet=np.zeros(np.shape(dataSet))
    m=dataSet.shape[0]
    normDataSet=dataSet-np.tile(minVal,(m,1))
    normDataSet=normDataSet/np.tile(ranges,(m,1))
    return normDataSet,ranges,minVal

测试分类效果

#测试数据
def datingAllClassTest():
    hoRatio = 0.10      #测试集数量比例
    datingDataMat,datingLabels = file2matrix("KNN\datingTestSet.txt")     #load data setfrom file
    normMat, ranges, minVals = autoNorm(datingDataMat)
    m = normMat.shape[0]#总样本数
    numTestVecs = int(m*hoRatio)
    errorCount = 0.0
    for i in range(numTestVecs):
        classifierResult=knnClassify(normMat[i,:],normMat[numTestVecs:m,:],datingLabels,3)
        print("预测的结果是%d 真正的答案是%d"%(classifierResult,datingLabels[i]))
        if classifierResult!=datingLabels[i]:
            errorCount+=1
    print("总的正确率是 %f"%(errorCount/float(numTestVecs)))
    print("错误数量是 %d"%errorCount)

实际应用作为分类器

#使用数据判断妹子是否喜欢你
def datingClassTest():
    resultList = ['妹子不喜欢你', '妹子有点喜欢你', '妹子特别喜欢你']
    ffMiles=float(input("你一年坐的飞机飞多少里:"))
    iceCream=float(input("你一年吃多少冰激凌?"))
    gameTime=float(input("你一年打多少小时的游戏?"))
    inArr=np.array([ffMiles,gameTime,iceCream])
    resDataMatrix,labels=file2matrix("KNN\datingTestSet.txt")
    normData,ranges,minVal=autoNorm(resDataMatrix)
    classResult=knnClassify((inArr-minVal)/ranges,normData,labels,3)
    print("估计是%s"%(resultList[classResult-1]))

源代码

import numpy as np 
import matplotlib.pyplot as plt 
import matplotlib as mpl
import operator as op
# 判断分类
def isWhichClass(className):
    if className=='didntLike':
        return 1
    elif className=='smallDoses':
        return 2
    elif className=='largeDoses':
        return 3

#加载数据

def file2matrix(filename):
    fr=open(filename)
    arrayOLines=fr.readlines()
    numbersOfLines=len(arrayOLines)#获取一共有多少行
    returnMat=np.zeros((numbersOfLines,3))#创建n行3列的全零矩阵
    classLabelVector=[]#标签向量
    index=0
    for line in arrayOLines:
        line=line.strip()
        listFormLine=line.split('\t')
        returnMat[index,:]=listFormLine[0:3]
        classLabelVector.append(isWhichClass(listFormLine[-1]))
        index+=1
    return returnMat,classLabelVector


def file3matrix(filename):
    fr=open(filename)
    arrayOLines=fr.readlines()
    numbersOfLines=len(arrayOLines)#获取一共有多少行
    returnMat=np.zeros((numbersOfLines,3))#创建n行3列的全零矩阵
    classLabelVector=[]#标签向量
    index=0
    for line in arrayOLines:
        line=line.strip()
        listFormLine=line.split('\t')
        returnMat[index,:]=listFormLine[0:3]
        classLabelVector.append(int(listFormLine[-1]))
        index+=1
    return returnMat,classLabelVector

#绘图
def makeGraph():
    returnMat,classLabelVector=file2matrix('KNN\datingTestSet.txt')
    fig=plt.figure()
    ax=fig.add_subplot(111)
    x1=[x[1] for i,x in enumerate(returnMat) if classLabelVector[i]==1]
    y1=[x[2] for i,x in enumerate(returnMat) if classLabelVector[i]==1]
    x2=[x[1] for i,x in enumerate(returnMat) if classLabelVector[i]==2]
    y2=[x[2] for i,x in enumerate(returnMat) if classLabelVector[i]==2]
    x3=[x[1] for i,x in enumerate(returnMat) if classLabelVector[i]==3]
    y3=[x[2] for i,x in enumerate(returnMat) if classLabelVector[i]==3]
    type1=plt.scatter(x1,y1,c="red",cmap='brg',alpha=0.2, marker='8', linewidth=0)
    type2=plt.scatter(x2,y2,c="green",cmap='brg',alpha=0.2, marker='8', linewidth=0)
    type3=plt.scatter(x3,y3,c="blue",cmap='brg', alpha=0.2, marker='8', linewidth=0)
    ax.legend([type1, type2, type3], ["Did Not Like", "Liked in Small Doses", "Liked in Large Doses"], loc=2)
    mpl.rcParams['font.sans-serif']=['SimHei'] #指定默认字体 SimHei为黑体
    mpl.rcParams['axes.unicode_minus']=False #用来正常显示负号
    plt.xlabel('玩游戏时间所占百分比', fontsize=14)
    plt.ylabel('每周消耗的冰激凌的公斤数', fontsize=14)
    plt.show()

#K近邻算法
#inX是输入的数据
#dataSet是训练的数据
#labels是标签,类别
#k是周围邻居的数量
#返回预测的类别
def knnClassify(inX,dataSet,labels,k):
    #计算欧式距离
    dataSetSize=dataSet.shape[0]#shape是(4,2),要获取点的数量显然是shape[0]
    diffMat=np.tile(inX,(dataSetSize,1))-dataSet
    '''
    np.tile(inX,(dataSetSize,1))=[[0,0],[0,0],[0,0],[0,0]]
    dataSet=[[1.0,1.1],[1.0,1.0],[0,0],[0,0.1]]
    np.tile-dataSet=[[-1.0,-1.1],[-1.0,-1.0],[0,0],[0,-0.1]]
    即([x1-x0,y1-y0],[x2-x0,y2-x0],[x3-x0,y3-y0],[x4-x0,y4-y0])
    '''
    sqdiffMat=diffMat**2
    #([(x1-x0)**2,(y1-y0)**2],[(x2-x0)**2,(y2-x0)**2],[(x3-x0)**2,(y3-y0)**2],[(x4-x0)**2,(y4-y0)**2])
    sqDistances=sqdiffMat.sum(axis=1)
    #([(x1-x0)**2+y1-y0)**2],[(x2-x0)**2+(y2-x0)**2],[(x3-x0)**2+(y3-y0)**2],[(x4-x0)**2+(y4-y0)**2])
    # print(sqDistances)
    distances=sqDistances**0.5
    #([(x1-x0)**2+(y1-y0)**2]**(0.5),[(x2-x0)**2+(y2-x0)**2]**(0.5),[(x3-x0)**2+(y3-y0)**2]**(0.5),[(x4-x0)**2+(y4-y0)**2])**(0.5)
    sortedDistIndicies=distances.argsort()#按照数值大小对下标排序,[2 3 1 0]
    classCount={}
    #选择距离最小的k个点
    for i in range(k):
        votIlabel=labels[sortedDistIndicies[i]]#获取最近的K个邻居的距离,对应的目标值
        # print(votIlabel)
        classCount[votIlabel]=classCount.get(votIlabel,0)+1#get(key,default)当key不存在时候默认值是default
    sortedClassCount = sorted(classCount.items(), key=op.itemgetter(1), reverse=True)
    # [('B', 2), ('A', 1)]
    return sortedClassCount[0][0]#返回与其最近的k个邻居中,出现最多的次数

# 自动归一化
def autoNorm(dataSet):
    minVal=dataSet.min(0)
    maxVal=dataSet.max(0)
    ranges=maxVal-minVal
    normDataSet=np.zeros(np.shape(dataSet))
    m=dataSet.shape[0]
    normDataSet=dataSet-np.tile(minVal,(m,1))
    normDataSet=normDataSet/np.tile(ranges,(m,1))
    return normDataSet,ranges,minVal


#测试数据
def datingAllClassTest():
    hoRatio = 0.10      #hold out 10%
    datingDataMat,datingLabels = file2matrix("KNN\datingTestSet.txt")     #load data setfrom file
    normMat, ranges, minVals = autoNorm(datingDataMat)
    m = normMat.shape[0]
    numTestVecs = int(m*hoRatio)
    errorCount = 0.0
    for i in range(numTestVecs):
        classifierResult=knnClassify(normMat[i,:],normMat[numTestVecs:m,:],datingLabels,3)
        print("预测的结果是%d 真正的答案是%d"%(classifierResult,datingLabels[i]))
        if classifierResult!=datingLabels[i]:
            errorCount+=1
    print("总的正确率是 %f"%(errorCount/float(numTestVecs)))
    print("错误数量是 %d"%errorCount)

#使用数据判断妹子是否喜欢你
def datingClassTest():
    resultList = ['妹子不喜欢你', '妹子有点喜欢你', '妹子特别喜欢你']
    ffMiles=float(input("你一年坐的飞机飞多少里:"))
    iceCream=float(input("你一年吃多少冰激凌?"))
    gameTime=float(input("你一年打多少小时的游戏?"))
    inArr=np.array([ffMiles,gameTime,iceCream])
    resDataMatrix,labels=file2matrix("KNN\datingTestSet.txt")
    normData,ranges,minVal=autoNorm(resDataMatrix)
    classResult=knnClassify((inArr-minVal)/ranges,normData,labels,3)
    print("估计是%s"%(resultList[classResult-1]))
   

if __name__ == "__main__":
    datingClassTest()
    

目录结构

数据集

https://github.com/pbharrin/machinelearninginaction

参考书籍

《机器学习实战》-彼得·哈灵顿

posted @ 2020-01-22 22:47  梦小冷  阅读(343)  评论(0编辑  收藏  举报