锐单电子商城 , 一站式电子元器件采购平台!
  • 电话:400-990-0325

练习2:逻辑回归Python

时间:2023-07-23 13:07:00 300kg传感器accu

练习2:逻辑回归


介绍

在这个练习中,您将在两个不同的数据集中实现逻辑回归和应用。通过将正则化添加到训练算法中,提高算法的鲁棒性,并在更复杂的情况下测试模型算法。

在开始练习之前,需要下载以下文件上传数据

  • ex2data1.txt -前半部分的训练数据集
  • ex2data2.txt -训练数据集的后半部分

在整个练习中,涉及以下内容必做作业

  • 绘制2D分类数据函数-(3分)
  • 实现Sigmoid函数--------(5分)
  • 实现Logistic回归成本函数和梯度函数-(60分)
  • 回归预测函数-(5分)
  • 实现正则Logisitic回归成本函数-(27分)

1 Logistic回归

在这部分练习中,将建立逻辑回归模型,以预测学生是否能被大学录取。

假设你是大学某个部门的负责人,你要根据两次考试的结果来决定每个申请人的入学机会。目前已经有了以往申请者的历史数据,并且可以用作逻辑回归的训练集。对于每行数据,都包含对应申请者的两次考试分数和最终的录取结果。

在这个练习中,你根据这两次考试的分数,需要建立一个分类模型来预测申请人的录取结果

1.1 数据可视化

在实施任何算法模型之前,最好先可视化数据,这样会更直观地获取数据特征。

现在,你需要它编写代码完成数据绘图,如下图所示。

要点

  • 需要使用的导入python并将从文件出发ex2data1.txt中读数据,显示前五行
  • x-y两次考试两次考试的分数
  • 不同的标记需要显示正负示例(不同的颜色)
####在这里填写代码## ####主要实现要点1############################################################################################################################################################################################################################################################################################################################################################################################################################ import numpy as np import pandas as pd import matplotlib.pyplot as plt path = 'ex2data1.txt' data = pd.read_csv(path, header=None, names=['Exam 1', 'Exam 2', 'Admitted']) data.head()

Exam 1 Exam 2 Admitted
0 34.623660 78.024693 0
1 30.286711 43.894998 0
2 35.847409 72.902198 0
3 60.182599 86.308552 1
4 79.032736 75.344376 1

####在这里填写代码## ###绘制数据散点图### positive = data[data['Admitted'].isin([1])] negative = data[data['Admitted'].isin([0])] fig, ax = plt.subplots(figsize=(12,8)) # 画50个样本,正向类,c=‘b’颜色,maker=‘o’绘制的形状 ax.scatter(positive['Exam 1'], positive['Exam 2'], s=50, c='b', marker='o', label='Admitted') ax.scatter(negative['Exam 1'], negative['Exam 2'], s=50, c='r', marker='x', label='Not Admitted') ax.legend()# Legend 图例,获取label如图右上角所示 ax.set_xlabel('Exam 1 Score') ax.set_ylabel('Exam 2 Score') plt.show()

1.2 实现

从前一部分实践中绘制的数据分布图可以看出,在不同标志的数据点之间有一个相对清晰的决策边界。现在有必要实现逻辑回归,并使用逻辑回归来训练模型来预测分类结果。

1.2.1 Sigmoid函数

在正式开始之前,让我们先了解一个函数:Sigmoid函数。 我们还记得逻辑回归假设的定义是:

???(??)=??(???)hθ(x)=g(θTX)

其中 g 代表S形函数的常用逻辑函数(Sigmoid function),公式为:

??(??)=11 ???g(z)=11 e?z

我们得到了逻辑回归模型的假设函数:

???(??)=11 ???hθ(x)=11 e?θTX

接下来,你需要它实现编写代码Sigmoid函数,试着在编写后测试一些值,如果x如果正值较大,函数值应接近1;x如果负值较大,函数值应接近0。x等于0,函数值为0.5。

确保您正在调用您的实现Sigmoid函数后,以下代码将输出以下图片:

####在这里填写代码## def sigmoid(z):        return 1.0 / (1.0   np.exp(-z)) 

####请运行并测试您的代码### nums = np.arange(-10, 10, step=1)  fig, ax = plt.subplots(figsize=(12,8)) ax.plot(nums, sigmoid(nums), 'r') plt.show()

1.2.2 成本函数和梯度

1.2.2.1 代价函数

逻辑回归的成本函数是:??(??)=1??∑??=1.[()log(()(1)(()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()())()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()()())()()()()()()()()()()())()()())()())()()()()()())())())()))()()))()))))()))()()()()))())))))))((())))(()))())))((()()())())))()))))))))))((((()))))))))))((((())))))))))))))))((((())()()))))))))))))))))(((((((((())))))))))))))))))))((((((((()))))))))(((((()))))))))))(((()))))))))))))))))))))))))))))))))(((((())))))((((((((()))))))))))))))))))))))))))))((((((()))))))))))))))))(((((((((((((((log(1???(??(??)))]J(θ)=1m∑i=1m[?y(i)log(hθ(x(i)))?(1?y(i))log(1?hθ(x(i)))]

现在,你需要它编写代码实现代价函数计算逻辑回归的成本,经过给定的数据测试后,初始成本约为0.693。

要点

  • 实现cost函数,参数为theta,X,y.
  • 返回计算的成本值。
  • 其中theta为参数,X集中训练的特征列,y三者都是矩阵,是训练集的标签列。
####在这里填写代码## def cost(theta,X,y):     theta = np.matrix(theta)     X = np.matrix(X)     y = np.matrix(y)     first = np.multiply(-y, np.log(sigmoid(X* theta.T)))     second = np.multiply((1 - y), np.log(1 - sigmoid(X* theta.T)))     return np.sum(first - second) / (len(X)) 

&bsp;

###请运行并测试你的代码###

#增加一列值为1,这和我们在练习1中的操作很相似
data.insert(0, 'Ones', 1)

# 定义X为训练数据,y为目的变量
cols = data.shape[1]
X = data.iloc[:,0:cols-1]
y = data.iloc[:,cols-1:cols]

# 将X,y转换为numpy数组,并初始化theta值为0
X = np.array(X.values)
y = np.array(y.values)
theta = np.zeros(3)

cost(theta, X, y)
0.6931471805599453

1.2.2.2 梯度下降

接下来,我们需要编写代码实现梯度下降用来计算我们的训练数据、标签和一些参数𝜃θ的梯度

要点

  • 代码实现gradient函数,参数为theta,X,y.
  • 返回计算的梯度值。
  • 其中theta为参数,X为训练集中的特征列,y为训练集的标签列,三者均为矩阵。

批量梯度下降转化为向量化计算: 1𝑚𝑋𝑇(𝑆𝑖𝑔𝑚𝑜𝑖𝑑(𝑋𝜃)−𝑦)1mXT(Sigmoid(Xθ)−y)

∂𝐽(𝜃)∂𝜃𝑗=1𝑚∑𝑖=1𝑚(ℎ𝜃(𝑥(𝑖))−𝑦(𝑖))𝑥(𝑖)𝑗∂J(θ)∂θj=1m∑i=1m(hθ(x(i))−y(i))xj(i)

这里需要注意的是,我们实际上没有在这个函数中执行梯度下降,我们仅仅在计算一个梯度步长。由于我们使用Python,我们可以用SciPy的optimize命名空间来做同样的事情。

 

###在这里填入代码###
def gradient(theta, X, y):
    theta = np.matrix(theta)
    X = np.matrix(X)
    y = np.matrix(y)
    
    parameters = int(theta.ravel().shape[1])
    grad = np.zeros(parameters)
    
    error = sigmoid(X * theta.T) - y
    
    for i in range(parameters):
        term = np.multiply(error, X[:,i])
        grad[i] = np.sum(term) / len(X)
    
    return grad
###请运行并测试你的代码###
gradient(theta, X, y)
array([ -0.1       , -12.00921659, -11.26284221])

 

1.2.3 寻找最优参数

现在可以用SciPy's truncated newton(TNC)实现寻找最优参数。

###请运行并测试你的代码###
import scipy.optimize as opt
result = opt.fmin_tnc(func=cost, x0=theta, fprime=gradient, args=(X, y))
result

 

(array([-25.16131863,   0.20623159,   0.20147149]), 36, 0)

让我们看看在这个结论下代价函数的值:

###请运行并测试你的代码###
cost(result[0], X, y)

 

0.20349770158947458

1.2.4 评估逻辑回归

接下来,我们需要编写代码实现预测函数,用所学的最优参数𝜃θ来为数据集X输出预测结果。然后,可以使用这个函数来给我们定义的分类器的训练精度进行打分。

逻辑回归的假设函数:

ℎ𝜃(𝑥)=11+𝑒−𝜃𝑇𝑋hθ(x)=11+e−θTX

当ℎ𝜃hθ大于等于0.5时,预测 y=1

当ℎ𝜃hθ小于0.5时,预测 y=0。

要点

  • 代码实现predict函数,参数为theta,X.
  • 返回X中的每行数据对应的预测结果。
  • 其中theta为参数,X为训练集中的特征列。
###在这里填入代码###
def predict(theta, X):
    probability = sigmoid(X * theta.T)
    return [1 if x >= 0.5 else 0 for x in probability]
###请运行并测试你的代码###
theta_min = np.matrix(result[0])
predictions = predict(theta_min, X)
correct = [1 if ((a == 1 and b == 1) or (a == 0 and b == 0)) else 0 for (a, b) in zip(predictions, y)]
accuracy = (sum(map(int, correct)) % len(correct))
print ('accuracy = {0}%'.format(accuracy))
accuracy = 89%
###请运行并测试你的代码###
theta_min = np.matrix(result[0])
predictions = predict(theta_min, X)
correct = [1 if ((a == 1 and b == 1) or (a == 0 and b == 0)) else 0 for (a, b) in zip(predictions, y)]
accuracy = (sum(map(int, correct)) % len(correct))
print ('accuracy = {0}%'.format(accuracy))

 

accuracy = 89%

2 正则化逻辑回归

在本部分练习中,我们将要通过加入正则项提升逻辑回归算法。

正则化是成本函数中的一个术语,它使算法更倾向于“更简单”的模型。这个理论助于减少过拟合,提高模型的泛化能力。

设想你是工厂的生产主管,你有一些芯片在两次测试中的测试结果。对于这两次测试,你想决定芯片是要被接受或抛弃。为了帮助你做出艰难的决定,你拥有过去芯片的测试数据集,从其中你可以构建一个逻辑回归模型

2.1 数据可视化

与第一部分的练习类似,首先对数据进行可视化:

 

path =  'ex2data2.txt'
data2 = pd.read_csv(path, header=None, names=['Test 1', 'Test 2', 'Accepted'])
data2.head()

 

Test 1 Test 2 Accepted
0 0.051267 0.69956 1
1 -0.092742 0.68494 1
2 -0.213710 0.69225 1
3 -0.375000 0.50219 1
4 -0.513250 0.46564 1
positive = data2[data2['Accepted'].isin([1])]
negative = data2[data2['Accepted'].isin([0])]

fig, ax = plt.subplots(figsize=(12,8))
ax.scatter(positive['Test 1'], positive['Test 2'], s=50, c='b', marker='o', label='Accepted')
ax.scatter(negative['Test 1'], negative['Test 2'], s=50, c='r', marker='x', label='Rejected')
ax.legend()
ax.set_xlabel('Test 1 Score')
ax.set_ylabel('Test 2 Score')
plt.show()

 

 

对于这部分数据,我们可以看出不同类别的数据点之间没有明显的线性决策界限用于划分两类数据。

因此,逻辑回归无法在此数据集上得到较好的效果,因为逻辑回归只能知道线性决策边界。

2.2 特征映射

一种能够更好地拟合数据的方法是构造从原始特征的多项式中得到的特征,即特征映射。如下图所示,作为这种映射的结果,我们的两个特征向量𝑥1,𝑥2x1,x2(两次质量保证测试的分数)已经被转换成了28维的向量。

在这个高维特征向量上训练的逻辑回归分类器将具有更复杂的决策边界,并在二维图中绘制时呈现非线性的划分曲线。

虽然特征映射允许我们构建一个更具有表现力的分类器,但它也更容易过拟合。接下来,你需要实现正则化逻辑回归用于拟合数据,并使用正则化来帮助解决过拟合问题

我们通过创建一组多项式特征来开始! 

 

# 设定映射深度
degree = 5
# 分别取两次测试的分数
x1 = data2['Test 1']
x2 = data2['Test 2']

data2.insert(3, 'Ones', 1)

# 设定计算方式进行映射
for i in range(1, degree):
    for j in range(0, i):
        data2['F' + str(i) + str(j)] = np.power(x1, i-j) * np.power(x2, j)

# 整理数据列
data2.drop('Test 1', axis=1, inplace=True)
data2.drop('Test 2', axis=1, inplace=True)

print("特征映射后具有特征维数:%d" %data2.shape[1])
data2.head()

 

特征映射后具有特征维数:12

Out[33]:

Accepted Ones F10 F20 F21 F30 F31 F32 F40 F41 F42 F43
0 1 1 0.051267 0.002628 0.035864 0.000135 0.001839 0.025089 0.000007 0.000094 0.001286 0.017551
1 1 1 -0.092742 0.008601 -0.063523 -0.000798 0.005891 -0.043509 0.000074 -0.000546 0.004035 -0.029801
2 1 1 -0.213710 0.045672 -0.147941 -0.009761 0.031616 -0.102412 0.002086 -0.006757 0.021886 -0.070895
3 1 1 -0.375000 0.140625 -0.188321 -0.052734 0.070620 -0.094573 0.019775 -0.026483 0.035465 -0.047494
4 1 1 -0.513250 0.263426 -0.238990 -0.135203 0.122661 -0.111283 0.069393 -0.062956 0.057116 -0.051818

2.3 代价函数和梯度 

接下来,你需要编写代码来实现计算正则化逻辑回归的代价函数和梯度,并返回计算的代价值和梯度

正则化逻辑回归的代价函数如下:

𝐽(𝜃)=1𝑚∑𝑖=1𝑚[−𝑦(𝑖)log(ℎ𝜃(𝑥(𝑖)))−(1−𝑦(𝑖))log(1−ℎ𝜃(𝑥(𝑖)))]+𝜆2𝑚∑𝑗=1𝑛𝜃2𝑗J(θ)=1m∑i=1m[−y(i)log⁡(hθ(x(i)))−(1−y(i))log⁡(1−hθ(x(i)))]+λ2m∑j=1nθj2

其中𝜆λ是“学习率”参数,其值会影响函数中的正则项值。且不应该正则化参数𝜃0θ0。

 

###在这里填入代码###
def costReg(theta, X, y, learningRate):
    theta = np.matrix(theta)
    X = np.matrix(X)
    y = np.matrix(y)
    first = np.multiply(-y, np.log(sigmoid(X * theta.T)))
    second = np.multiply((1 - y), np.log(1 - sigmoid(X * theta.T)))
    reg = (learningRate / (2 * len(X))) * np.sum(np.power(theta[:,1:theta.shape[1]], 2))
    return np.sum(first - second) / len(X) + reg  

 

###在这里填入代码###
def gradientReg(theta, X, y, learningRate):
    theta = np.matrix(theta)
    X = np.matrix(X)
    y = np.matrix(y)
    
    parameters = int(theta.ravel().shape[1])
    grad = np.zeros(parameters)
    
    error = sigmoid(X * theta.T) - y
    
    for i in range(parameters):
        term = np.multiply(error, X[:,i])
        
        if (i == 0):
            grad[i] = np.sum(term) / len(X)
        else:
            grad[i] = (np.sum(term) / len(X)) + ((learningRate / len(X)) * theta[:,i])
    
    return grad

 接下来,类似于第一部分的练习中,进行变量的初始化。

# 从数据集中取得对应的特征列和标签列
cols = data2.shape[1]
X2 = data2.iloc[:,1:cols]
y2 = data2.iloc[:,0:1]

# 转换为Numpy数组并初始化theta为零矩阵
X2 = np.array(X2.values)
y2 = np.array(y2.values)
theta2 = np.zeros(11)

# 设置初始学习率为1,后续可以修改
learningRate = 1

接下来,使用初始化的变量值来测试你实现的代价函数和梯度函数。

###请运行并测试你的代码###
costReg(theta2, X2, y2, learningRate)
### 0.6931471805599454
###请运行并测试你的代码###
gradientReg(theta2, X2, y2, learningRate)
### array([0.00847458, 0.01878809, 0.05034464, 0.01150133, 0.01835599,
       0.00732393, 0.00819244, 0.03934862, 0.00223924, 0.01286005,
       0.00309594])

2.4 寻找最优参数

现在我们可以使用和第一部分相同的优化函数来计算优化后的结果。

result2 = opt.fmin_tnc(func=costReg, x0=theta2, fprime=gradientReg, args=(X2, y2, learningRate))
result2
### (array([ 0.53010249,  0.29075567, -1.60725763, -0.5821382 ,  0.01781027,
        -0.21329509, -0.40024142, -1.37144139,  0.02264303, -0.9503358 ,
         0.0344085 ]), 22, 1)

 

2.5 评估正则化逻辑回归

最后,我们可以使用第1部分中的预测函数来查看我们的方案在训练数据上的准确度。

theta_min = np.matrix(result2[0])
predictions = predict(theta_min, X2)
correct = [1 if ((a == 1 and b == 1) or (a == 0 and b == 0)) else 0 for (a, b) in zip(predictions, y2)]
accuracy = (sum(map(int, correct)) % len(correct))
print ('accuracy = {0}%'.format(accuracy))

## accuracy = 78%

 

锐单商城拥有海量元器件数据手册IC替代型号,打造电子元器件IC百科大全!

相关文章