Skip to the content.

《Machine Learning In Action》 中SVM部分代码精简优化


Contact me:

Blog -> https://cugtyt.github.io/blog/index
Email -> cugtyt@qq.com
GitHub -> Cugtyt@GitHub


本系列博客主页及相关见此处
此处粘贴的代码来源为https://github.com/Jack-Cherish/Machine-Learning


1. clipAlpha函数,用来计算alpha

原代码:

def clipAlpha(aj,H,L):
    if aj > H:
        aj = H
    if L > aj:
        aj = L
    return aj

问题:

使用min和max就可以很简便的实现

参考修改:

def clipAlpha(aj,H,L):
    aj = min(aj, H)
    aj = max(L, aj)
    return aj

2. 二维矩阵转置用T而不是函数

原代码:

np.transpose(data_minus_np)[0]

问题:

维度小于等于2的使用.T更方便

参考修改:

data_minus_np.T[0]

2. 使用随机整数而不是随机数取整

原代码:

j = int(random.uniform(0, m))

问题:

随机整数有方便的获取函数

参考修改:

j = random.randint(0, m - 1)

3. smoSimple函数简化

原代码:

def smoSimple(dataMatIn, classLabels, C, toler, maxIter):
    #转换为numpy的mat存储
    dataMatrix = np.mat(dataMatIn); labelMat = np.mat(classLabels).transpose()
    #初始化b参数,统计dataMatrix的维度
    b = 0; m,n = np.shape(dataMatrix)
    #初始化alpha参数,设为0
    alphas = np.mat(np.zeros((m,1)))
    #初始化迭代次数
    iter_num = 0
    #最多迭代matIter次
    while (iter_num < maxIter):
        alphaPairsChanged = 0
        for i in range(m):
            #步骤1:计算误差Ei
            fXi = float(np.multiply(alphas,labelMat).T*(dataMatrix*dataMatrix[i,:].T)) + b
            Ei = fXi - float(labelMat[i])
            #优化alpha,更设定一定的容错率。
            if ((labelMat[i]*Ei < -toler) and (alphas[i] < C)) or ((labelMat[i]*Ei > toler) and (alphas[i] > 0)):
                #随机选择另一个与alpha_i成对优化的alpha_j
                j = selectJrand(i,m)
                #步骤1:计算误差Ej
                fXj = float(np.multiply(alphas,labelMat).T*(dataMatrix*dataMatrix[j,:].T)) + b
                Ej = fXj - float(labelMat[j])
                #保存更新前的aplpha值,使用深拷贝
                alphaIold = alphas[i].copy(); alphaJold = alphas[j].copy();
                #步骤2:计算上下界L和H
                if (labelMat[i] != labelMat[j]):
                    L = max(0, alphas[j] - alphas[i])
                    H = min(C, C + alphas[j] - alphas[i])
                else:
                    L = max(0, alphas[j] + alphas[i] - C)
                    H = min(C, alphas[j] + alphas[i])
                if L==H: print("L==H"); continue
                #步骤3:计算eta
                eta = 2.0 * dataMatrix[i,:]*dataMatrix[j,:].T - dataMatrix[i,:]*dataMatrix[i,:].T - dataMatrix[j,:]*dataMatrix[j,:].T
                if eta >= 0: print("eta>=0"); continue
                #步骤4:更新alpha_j
                alphas[j] -= labelMat[j]*(Ei - Ej)/eta
                #步骤5:修剪alpha_j
                alphas[j] = clipAlpha(alphas[j],H,L)
                if (abs(alphas[j] - alphaJold) < 0.00001): print("alpha_j变化太小"); continue
                #步骤6:更新alpha_i
                alphas[i] += labelMat[j]*labelMat[i]*(alphaJold - alphas[j])
                #步骤7:更新b_1和b_2
                b1 = b - Ei- labelMat[i]*(alphas[i]-alphaIold)*dataMatrix[i,:]*dataMatrix[i,:].T - labelMat[j]*(alphas[j]-alphaJold)*dataMatrix[i,:]*dataMatrix[j,:].T
                b2 = b - Ej- labelMat[i]*(alphas[i]-alphaIold)*dataMatrix[i,:]*dataMatrix[j,:].T - labelMat[j]*(alphas[j]-alphaJold)*dataMatrix[j,:]*dataMatrix[j,:].T
                #步骤8:根据b_1和b_2更新b
                if (0 < alphas[i]) and (C > alphas[i]): b = b1
                elif (0 < alphas[j]) and (C > alphas[j]): b = b2
                else: b = (b1 + b2)/2.0
                #统计优化次数
                alphaPairsChanged += 1
                #打印统计信息
                print("第%d次迭代 样本:%d, alpha优化次数:%d" % (iter_num,i,alphaPairsChanged))
        #更新迭代次数
        if (alphaPairsChanged == 0): iter_num += 1
        else: iter_num = 0
        print("迭代次数: %d" % iter_num)
    return b,alphas

问题:

  1. 对于二维及以下的矩阵使用.T更方便
  2. 在python中不要使用不必要的括号
  3. 在numpy中[index, :]与[index]是等价的
  4. 不必使用不必要的float转换
  5. 使用换行而不是分号
  6. 计算可以通过数学简化
  7. 简单的条件赋值可以简化

参考修改:

def smoSimple(dataMatIn, classLabels, C, toler, maxIter):
    #转换为numpy的mat存储
    dataMatrix = np.mat(dataMatIn)
    labelMat = np.mat(classLabels).T
    #初始化b参数,统计dataMatrix的维度
    b = 0
    m, n = np.shape(dataMatrix)
    #初始化alpha参数,设为0
    alphas = np.mat(np.zeros((m, 1)))
    #初始化迭代次数
    iter_num = 0
    #最多迭代matIter次
    while iter_num < maxIter:
        alphaPairsChanged = 0
        for i in range(m):
            #步骤1:计算误差Ei
            Ei = np.multiply(alphas, labelMat).T * (dataMatrix * dataMatrix[i].T) + b - labelMat[i]
            #优化alpha,更设定一定的容错率。
            if labelMat[i]*Ei < -toler and alphas[i] < C or labelMat[i]*Ei > toler and alphas[i] > 0:
                #随机选择另一个与alpha_i成对优化的alpha_j
                j = selectJrand(i,m)
                #步骤1:计算误差Ej
                Ej = np.multiply(alphas, labelMat).T * (dataMatrix * dataMatrix[j].T) + b - labelMat[j]
                #保存更新前的aplpha值,使用深拷贝
                alphaIold = alphas[i].copy()
                alphaJold = alphas[j].copy()
                #步骤2:计算上下界L和H
                if labelMat[i] != labelMat[j]:
                    L = max(0, alphas[j] - alphas[i])
                    H = min(C, C + alphas[j] - alphas[i])
                else:
                    L = max(0, alphas[j] + alphas[i] - C)
                    H = min(C, alphas[j] + alphas[i])
                if L==H: 
                    print("L == H")
                    continue
                #步骤3:计算eta
                eta = dataMatrix[i] - dataMatrix[j]
                eta = - eta * eta.T
                if eta >= 0: 
                    print("eta >= 0")
                    continue
                #步骤4:更新alpha_j
                alphas[j] -= labelMat[j] * (Ei - Ej) / eta
                #步骤5:修剪alpha_j
                alphas[j] = clipAlpha(alphas[j], H, L)
                if abs(alphas[j] - alphaJold) < 0.00001: 
                    print("alpha_j变化太小")
                    continue
                #步骤6:更新alpha_i
                alphas[i] += labelMat[j] * labelMat[i] * (alphaJold - alphas[j])
                #步骤7:更新b_1和b_2
                b1 = b - Ei- labelMat[i] * (alphas[i] - alphaIold) * dataMatrix[i] * dataMatrix[i].T \
                    - labelMat[j] * (alphas[j] - alphaJold) * dataMatrix[i] * dataMatrix[j].T
                b2 = b - Ej- labelMat[i] * (alphas[i] - alphaIold) * dataMatrix[i] * dataMatrix[j].T \
                    - labelMat[j] * (alphas[j] - alphaJold) * dataMatrix[j] * dataMatrix[j].T
                #步骤8:根据b_1和b_2更新b
                if 0 < alphas[i] and C > alphas[i]: 
                    b = b1
                elif 0 < alphas[j] and C > alphas[j]: 
                    b = b2
                else: 
                    b = (b1 + b2)/2.0
                #统计优化次数
                alphaPairsChanged += 1
                #打印统计信息
                print("第%d次迭代 样本:%d, alpha优化次数:%d" % (iter_num,i,alphaPairsChanged))
        #更新迭代次数
        iter_num = iter_num + 1 if not alphaPairsChanged else 0
    return b,alphas

3. get_w函数优化

原代码:

def get_w(dataMat, labelMat, alphas):
    alphas, dataMat, labelMat = np.array(alphas), np.array(dataMat), np.array(labelMat)
    w = np.dot((np.tile(labelMat.reshape(1, -1).T, (1, 2)) * dataMat).T, alphas)
    return w.tolist()

问题:

尽量不要使用tile,大多数情况broadcasting足可以替代

参考修改:

def get_w(dataMat, labelMat, alphas):
    alphas, dataMat, labelMat = np.array(alphas), np.array(dataMat), np.array(labelMat)
    w = np.dot((labelMat.reshape(-1, 1) * dataMat).T, alphas)
    return w.tolist()