机器学习实战-Logistic回归,其中的向量x是分类器


1.基于 Logistic 回归和 Sigmoid 函数的分类

逻辑回归适合于01情况的分类就是描述一个问题是或者不是,所以就引入sigmoid函数,因为这个函数可以将所有值变成0-1之间的一个值,这样就方便算概率 首先我们可以先看看Sigmoid函数(又叫Logistic函数)将任意的输入映射到了[0,1]区间我们在线性回归中可以得到一个预测值,再将该值映射到sigmoid函数中这样就完成了由值到概率的转换,也就是分类任务,公式如下:

整合成一个公式,就变成了如下公式:

z是一个矩阵,θ是参数列向量(要求解的),x是样本列向量(给定的数据集),θ^T表示θ的转置

Sigmoid函数的输入记为z,由下面公式得出:
z=w0x0+w1x1+w2x2+...+wnxn
如果采用向量的写法,上述公式可以写成z = wTx,它表示将这两个数值向量对应元素相乘然后
全部加起来即得到z值。其中的向量x是分类器的输入数据,向量w也就是我们要找到的最佳参数 (系数),从而使得分类器尽可能地精确。

逻辑回归的简单来说,就是根据数据得到的回归直线方程z=a*x+b方程之后,将z作为sigmoid的输入使得z的值转化为在0-1之间的值,然后计算概率,最后根据sigmod函数的特点也就是当输入为零的时候,函数值为0.5,以0.5为分界线来划分数据的类型。

1.1梯度上升法

梯度上升算法用来求函数的最大值,而梯度下降算法用来求函数的最小值。
求一个函数的最大值,在数学中,是不是通过对函数求导,然后算出导数等于0,或者导数不存在的位置作为极值,然后如果极值只有一个开区间内是不是极值就是最大值。但是在实际应用中却不是这么简单的去求最大值,而是通过迭代的方式一步一步向最值点靠近,最后得到最值。这也就是梯度上升法的思想


其中,m为样本的总数,y(i)表示第i个样本的类别,x(i)表示第i个样本,需要注意的是θ是多维向量,x(i)也是多维向量。

梯度上升迭代公式为:

代码实现:

import matplotlib.pyplot as plt
import numpy as np
import warnings
warnings.filterwarnings('ignore')

def loadDataSet():
    dataMat = []   #创建数据列表
    labelMat = []   #创建标签列表
    fr = open('testSet.txt')    #打开文件
    for line in fr.readlines():     #逐行读取
        lineArr = line.strip().split()  #去回车,放入列表
        dataMat.append([1.0, float(lineArr[0]), float(lineArr[1])]) #添加数据
        labelMat.append(int(lineArr[2]))   #添加标签
    fr.close()   #关闭文件
    return dataMat, labelMat    #返回


def sigmoid(inX):
    return  1.0/(1+np.exp(-inX))

#dataMatIn,它是一个2维NumPy数组,每列分别代表每个不同的特征,每行则代表每个训练样本
def gradAscent(dataMatIn,classLabels):
    dataMatrix=np.mat(dataMatIn)#转换成numpy的mat
    labelMat=np.mat(classLabels).transpose()#为了便于矩阵运算,需要将该行向量转换为列向量,做法是将原向量转置,再将它赋值给labelMat
    m,n=np.shape(dataMatrix) #返回dataMatrix的大小。m为行数,n为列数。
    alpha=0.001 #向目标移动的步长
    maxCycles=500 #maxCycles是迭代次数
    weights=np.ones((n,1))#权重初始化都为1
    for k in range(maxCycles):
        h=sigmoid(dataMatrix*weights)#梯度上升矢量化公式
        error=(labelMat-h)#相当于公式中的y(i)-h(x(i))
        weights=weights+alpha*dataMatrix.transpose()*error#公式里面的累加在这里使用矩阵相乘来实现(矩阵相乘的过程中先乘再加)
    return weights.getA()  #返回权重数组


def plotBestFit(weights):
    dataMat, labelMat = loadDataSet() #加载数据集
    dataArr = np.array(dataMat) #转换成numpy的array数组
    n = np.shape(dataMat)[0]  #数据个数
    xcord1 = []; ycord1 = []   #正样本
    xcord2 = []; ycord2 = []   #负样本
    for i in range(n):    #根据数据集标签进行分类
        if int(labelMat[i]) == 1:
            xcord1.append(dataArr[i,1]); ycord1.append(dataArr[i,2]) #1为正样本
        else:
            xcord2.append(dataArr[i,1]); ycord2.append(dataArr[i,2])#0为负样本
    fig = plt.figure()
    ax = fig.add_subplot(111) #添加subplot
    ax.scatter(xcord1, ycord1, s = 20, c = 'red', marker = 's',alpha=.5)#绘制正样本
    ax.scatter(xcord2, ycord2, s = 20, c = 'green',alpha=.5)  #绘制负样本
    x = np.arange(-3.0, 3.0, 0.1)
    y = (-weights[0] - weights[1] * x) / weights[2]
    ax.plot(x, y)
    plt.title('BestFit')  #绘制title
    plt.xlabel('X1'); plt.ylabel('X2') #绘制label
    plt.show()

if __name__ == '__main__':
    dataMat, labelMat = loadDataSet()
    weights = gradAscent(dataMat, labelMat)
    plotBestFit(weights)
    #plotDataSet()

正常运行之前报了如下错误AttributeError: partially initialized module ‘matplotlib.cbook’ has no attribute ‘deprecated’ (most likely due to a circular import)

然后就是你安装的matplotlib版本太高了,可以将以前的版本给卸载了,然后安装一个版本较低的matplotlib,然后就可以解决

测试结果

总结:代码运行的大致思路如下:

  • 拿到数据之后通过loadDataSet函数将数据的坐标和类别分别存放在两个数组中

  • 然后通过拿到的坐标数组和分类数组,首先先将数组转换为矩阵,方便后面矩阵的运算,然后根据sigmoid函数将所有数据转化为0-1之间的数据,也即公式中的h(xi)的值,然后用每个样本的类标签减去梯度上升的矢量值,最后带入公式算出当前的权重值

  • 绘图

    • 然后绘图的时候定义两种数据类型的x坐标的数组和对应的y坐标的数组,然后通过数据集,将每种类别对应的x坐标和y坐标分开存入对应的数组中

    • 然后绘制一张空的面板,添加坐标系,然后将每个同一类别的点画在面板上,并且同一类别的点的颜色相同

    • 最后画拟合的直线,横坐标的范围已知,然后取sigmoid 函数为0。0是两个分类(类别1和类别0)的分界处。因此,我们设定 0 = w0x0 + w1x1 + w2x2,然后解出X2和X1的关系式(即分隔线的方程,注意X0=1)。

    • 最后根据函数方程和x的值将直线画在面板上即可

1.2改进的梯度上升算法

改进的第一点也就是改变每次向目标点靠近的步长的值,最初的时候也能稍微大点,然后随着迭代次数的增加,也就是离目标点越来越近,此时每次向前的步长也越来越小;第二点就是样本的选取也是随机的;第三点就是原来计算出来的h是一个100乘以1的矩阵,而现在算出来的是一个0-1之间的数值;第四点就是原来计算回归系数的时候使用一个3*100的矩阵乘以一个100*1的一个矩阵,现在是三个数值的乘积
#改进的梯度上升算法
#迭代次数150是根据上面的代码测试出来的
def stocGradAscent1(dataMatrix, classLabels, numIter=150):
    m,n = np.shape(dataMatrix)  #返回dataMatrix的大小。m为行数,n为列数。
    weights = np.ones(n)   #参数初始化
    for j in range(numIter):
        dataIndex = list(range(m))
        for i in range(m):
            alpha = 4/(1.0+j+i)+0.01  #降低alpha的大小,每次减小1/(j+i)。
            randIndex = int(random.uniform(0,len(dataIndex)))  #随机选取样本
            h = sigmoid(sum(dataMatrix[randIndex]*weights)) #选择随机选取的一个样本,计算h
            error = classLabels[randIndex] - h   #计算误差
            weights = weights + alpha * error * dataMatrix[randIndex]  #更新回归系数
            del(dataIndex[randIndex])  #删除已经使用的样本
    return weights

1.3回归系数与迭代次数的关系

#2.为改进之前查看回归系数与迭代次数的关系
def gradAscent1(dataMatIn, classLabels):
    dataMatrix = np.mat(dataMatIn)   #转换成numpy的mat
    labelMat = np.mat(classLabels).transpose()  #转换成numpy的mat,并进行转置
    m, n = np.shape(dataMatrix)    #返回dataMatrix的大小。m为行数,n为列数。
    alpha = 0.01  #移动步长,也就是学习速率,控制更新的幅度。
    maxCycles = 500    #最大迭代次数
    weights = np.ones((n,1))
    weights_array = np.array([])
    for k in range(maxCycles):
        h = sigmoid(dataMatrix * weights)  #梯度上升矢量化公式
        error = labelMat - h
        weights = weights + alpha * dataMatrix.transpose() * error
        weights_array = np.append(weights_array,weights)
    weights_array = weights_array.reshape(maxCycles,n)
    return weights.getA(),weights_array #将矩阵转换为数组,并返回

#改进之后的
def stocGradAscent(dataMatrix, classLabels, numIter=150):
    m,n = np.shape(dataMatrix)                                                #返回dataMatrix的大小。m为行数,n为列数。
    weights = np.ones(n)                                                       #参数初始化
    weights_array = np.array([])                                            #存储每次更新的回归系数
    for j in range(numIter):
        dataIndex = list(range(m))
        for i in range(m):
            alpha = 4/(1.0+j+i)+0.01                                            #降低alpha的大小,每次减小1/(j+i)。
            randIndex = int(random.uniform(0,len(dataIndex)))                #随机选取样本
            h = sigmoid(sum(dataMatrix[randIndex]*weights))                    #选择随机选取的一个样本,计算h
            error = classLabels[randIndex] - h                                 #计算误差
            weights = weights + alpha * error * dataMatrix[randIndex]       #更新回归系数
            weights_array = np.append(weights_array,weights,axis=0)         #添加回归系数到数组中
            del(dataIndex[randIndex])                                         #删除已经使用的样本
    weights_array = weights_array.reshape(numIter*m,n)                         #改变维度
    return weights,weights_array

测试结果

让我们分析一下。我们一共有100个样本点,改进的随机梯度上升算法迭代次数为150。而上图显示15000次迭代次数的原因是,使用一次样本就更新一下回归系数。因此,迭代150次,相当于更新回归系数150*100=15000次。简而言之,迭代150次,更新1.5万次回归参数。从上图左侧的改进随机梯度上升算法回归效果中可以看出,其实在更新2000次回归系数的时候,已经收敛了。相当于遍历整个数据集20次的时候,回归系数已收敛。训练已完成。

上图右侧的梯度上升算法回归效果,梯度上升算法每次更新回归系数都要遍历整个数据集。从图中可以看出,当迭代次数为300多次的时候,回归系数才收敛。凑个整,就当它在遍历整个数据集300次的时候已经收敛好了。

2.从疝气病症状预测病马的死亡率

代码实现
def classifyVector(inX, weights):
    prob = sigmoid(sum(inX*weights))
    if prob > 0.5: return 1.0
    else: return 0.0

def colicSklearn():
    frTrain = open('horseColicTraining.txt')  #打开训练集
    frTest = open('horseColicTest.txt') #打开测试集
    trainingSet = []; trainingLabels = []
    testSet = []; testLabels = []
    for line in frTrain.readlines():#取出训练集中的每一行的所有数据
        currLine = line.strip().split('\t')#每一行里数据的划分以空格划分
        lineArr = []
        for i in range(len(currLine)-1):#遍历每一行的每个元素
            lineArr.append(float(currLine[i]))#lineArr里存放的就是每一行中所有的数据
        trainingSet.append(lineArr)#将每一行的数据存放在训练集中
        trainingLabels.append(float(currLine[-1]))#拿到每一行的最后一列也即数据类别
    for line in frTest.readlines():#取出测试集中每一行的所有数据
        currLine = line.strip().split('\t')
        lineArr =[]
        for i in range(len(currLine)-1):
            lineArr.append(float(currLine[i]))
        testSet.append(lineArr)
        testLabels.append(float(currLine[-1]))
    classifier = LogisticRegression(solver='liblinear',max_iter=10).fit(trainingSet, trainingLabels)
    test_accurcy = classifier.score(testSet, testLabels) * 100
    print('正确率:%f%%' % test_accurcy)

测试结果

评论关闭