(pytorch-深度学习系列)pytorch避免过拟合-权重衰减的实现-学习笔记
pytorch避免過擬合-權重衰減的實現
首先學習基本的概念背景
L0范數是指向量中非0的元素的個數;(L0范數難優(yōu)化求解)
L1范數是指向量中各個元素絕對值之和;
L2范數是指向量各元素的平方和然后求平方根。
權重衰減等價于 L2范數正則化(regularization)。正則化通過為模型損失函數添加懲罰項使學出的模型參數值較小,是應對過擬合的常用手段。
對于線性回歸損失函數
?(w1,w2,b)=1n∑i=1n12(x1(i)w1+x2(i)w2+b?y(i))2\ell(w_1, w_2, b) = \frac{1}{n} \sum_{i=1}^n \frac{1}{2}\left(x_1^{(i)} w_1 + x_2^{(i)} w_2 + b - y^{(i)}\right)^2 ?(w1?,w2?,b)=n1?i=1∑n?21?(x1(i)?w1?+x2(i)?w2?+b?y(i))2
其中w1,w2w_1, w_2w1?,w2?是權重參數,bbb是偏差參數,樣本iii的輸入為x1(i),x2(i)x_1^{(i)}, x_2^{(i)}x1(i)?,x2(i)?,標簽為y(i)y^{(i)}y(i),樣本數為nnn。將權重參數用向量w=[w1,w2]\boldsymbol{w} = [w_1, w_2]w=[w1?,w2?]表示,帶有L2L_2L2?范數懲罰項的新損失函數為
?(w1,w2,b)+λ2n∣w∣2,\begin{aligned}\ell(w_1, w_2, b) + \frac{\lambda}{2n} |\boldsymbol{w}|^2,\end{aligned}?(w1?,w2?,b)+2nλ?∣w∣2,?
其中超參數λ>0\lambda > 0λ>0。當權重參數均為0時,懲罰項最小。當λ\lambdaλ較大時,懲罰項在損失函數中的比重較大,這通常會使學到的權重參數的元素較接近0。當λ\lambdaλ設為0時,懲罰項完全不起作用。上式中L2L_2L2?范數平方∣w∣2|\boldsymbol{w}|^2∣w∣2展開后得到w12+w22w_1^2 + w_2^2w12?+w22?。有了L2L_2L2?范數懲罰項后,在小批量隨機梯度下降中,我們將線性回歸中權重w1w_1w1?和w2w_2w2?的迭代方式更改為
w1←(1?ηλ∣B∣)w1?η∣B∣∑i∈Bx1(i)(x1(i)w1+x2(i)w2+b?y(i)),\begin{aligned} w_1 &\leftarrow \left(1- \frac{\eta\lambda}{|\mathcal{B}|} \right)w_1 - \frac{\eta}{|\mathcal{B}|} \sum_{i \in \mathcal{B}}x_1^{(i)} \left(x_1^{(i)} w_1 + x_2^{(i)} w_2 + b - y^{(i)}\right),\end{aligned}w1??←(1?∣B∣ηλ?)w1??∣B∣η?i∈B∑?x1(i)?(x1(i)?w1?+x2(i)?w2?+b?y(i)),?
w2←(1?ηλ∣B∣)w2?η∣B∣∑i∈Bx2(i)(x1(i)w1+x2(i)w2+b?y(i)).\begin{aligned}\ w_2 &\leftarrow \left(1- \frac{\eta\lambda}{|\mathcal{B}|} \right)w_2 - \frac{\eta}{|\mathcal{B}|} \sum_{i \in \mathcal{B}}x_2^{(i)} \left(x_1^{(i)} w_1 + x_2^{(i)} w_2 + b - y^{(i)}\right). \end{aligned} ?w2??←(1?∣B∣ηλ?)w2??∣B∣η?i∈B∑?x2(i)?(x1(i)?w1?+x2(i)?w2?+b?y(i)).?
注:
原線性回歸的w1w_1w1?和w2w_2w2?的迭代方式為
w1←w1?η∣B∣∑i∈B??(i)(w1,w2,b)?w1=w1?η∣B∣∑i∈Bx1(i)(x1(i)w1+x2(i)w2+b?y(i)),\begin{aligned} w_1 &\leftarrow w_1 - \frac{\eta}{|\mathcal{B}|} \sum_{i \in \mathcal{B}} \frac{ \partial \ell^{(i)}(w_1, w_2, b) }{\partial w_1} = w_1 - \frac{\eta}{|\mathcal{B}|} \sum_{i \in \mathcal{B}}x_1^{(i)} \left(x_1^{(i)} w_1 + x_2^{(i)} w_2 + b - y^{(i)}\right), \end{aligned} w1??←w1??∣B∣η?i∈B∑??w1???(i)(w1?,w2?,b)?=w1??∣B∣η?i∈B∑?x1(i)?(x1(i)?w1?+x2(i)?w2?+b?y(i)),?
w2←w2?η∣B∣∑i∈B??(i)(w1,w2,b)?w2=w2?η∣B∣∑i∈Bx2(i)(x1(i)w1+x2(i)w2+b?y(i)),\begin{aligned}\ w_2 &\leftarrow w_2 - \frac{\eta}{|\mathcal{B}|} \sum_{i \in \mathcal{B}} \frac{ \partial \ell^{(i)}(w_1, w_2, b) }{\partial w_2} = w_2 - \frac{\eta}{|\mathcal{B}|} \sum_{i \in \mathcal{B}}x_2^{(i)} \left(x_1^{(i)} w_1 + x_2^{(i)} w_2 + b - y^{(i)}\right), \end{aligned} ?w2??←w2??∣B∣η?i∈B∑??w2???(i)(w1?,w2?,b)?=w2??∣B∣η?i∈B∑?x2(i)?(x1(i)?w1?+x2(i)?w2?+b?y(i)),?
b←b?η∣B∣∑i∈B??(i)(w1,w2,b)?b=b?η∣B∣∑i∈B(x1(i)w1+x2(i)w2+b?y(i)).\begin{aligned}\ b &\leftarrow b - \frac{\eta}{|\mathcal{B}|} \sum_{i \in \mathcal{B}} \frac{ \partial \ell^{(i)}(w_1, w_2, b) }{\partial b} = b - \frac{\eta}{|\mathcal{B}|} \sum_{i \in \mathcal{B}}\left(x_1^{(i)} w_1 + x_2^{(i)} w_2 + b - y^{(i)}\right). \end{aligned} ?b?←b?∣B∣η?i∈B∑??b??(i)(w1?,w2?,b)?=b?∣B∣η?i∈B∑?(x1(i)?w1?+x2(i)?w2?+b?y(i)).?
在上式中,∣B∣|\mathcal{B}|∣B∣ 代表每個小批量中的樣本個數(批量大小,batch size),η\etaη 稱作學習率(learning rate)并取正數。這里的批量大小和學習率的值是人為設定的,并不是通過模型訓練學出的,因此叫作超參數(hyperparameter)。我們通常所說的“調參”指的正是調節(jié)超參數,例如通過反復試錯來找到超參數合適的值。在少數情況下,超參數也可以通過模型訓練學出。
可見,L2L_2L2?范數正則化令權重w1w_1w1?和w2w_2w2?先自乘小于1的數,再減去不含懲罰項的梯度。因此,L2L_2L2?范數正則化又叫權重衰減。權重衰減通過懲罰絕對值較大的模型參數為需要學習的模型增加了限制,這可能對過擬合有效。實際場景中,我們有時也在懲罰項中添加偏差元素的平方和。
設置一個過擬合問題
以高維線性回歸為例來引入一個過擬合問題,并使用權重衰減來應對過擬合。設數據樣本特征的維度為ppp。對于訓練數據集和測試數據集中特征為x1,x2,…,xpx_1, x_2, \ldots, x_px1?,x2?,…,xp?的任一樣本,我們使用如下的線性函數來生成該樣本的標簽:
y=0.05+∑i=1p0.01xi+?y = 0.05 + \sum_{i = 1}^p 0.01x_i + \epsilon y=0.05+i=1∑p?0.01xi?+?
其中噪聲項?\epsilon?服從均值為0、標準差為0.01的正態(tài)分布。
為了較容易地觀察過擬合,我們考慮高維線性回歸問題,如設維度p=200p=200p=200;同時,我們特意把訓練數據集的樣本數設低,如20。
這里就定義好了線性回歸問題,現在開始設置模型進行線性回歸求解:
隨機初始化模型參數的函數:
def init_params():w = torch.randn((num_inputs, 1), requires_grad=True)b = torch.zeros(1, requires_grad=True)return [w, b]定義L2L_2L2?范數懲罰項:
def l2_penalty(w):return (w**2).sum() / 2定義訓練模型需要的函數
def linreg(X, w, b):return torch.mm(X, w) + bdef squared_loss(y_hat, y): # 注意這里返回的是向量, 另外, pytorch里的MSELoss并沒有除以 2return ((y_hat - y.view(y_hat.size())) ** 2) / 2def sgd(params, lr, batch_size):# 為了和原書保持一致,這里除以了batch_size,但是應該是不用除的,因為一般用PyTorch計算loss時就默認已經# 沿batch維求了平均了。for param in params:param.data -= lr * param.grad / batch_size # 注意這里更改param時用的是param.datadef semilogy(x_vals, y_vals, x_label, y_label, x2_vals=None, y2_vals=None,legend=None, figsize=(3.5, 2.5)):set_figsize(figsize)plt.xlabel(x_label)plt.ylabel(y_label)plt.semilogy(x_vals, y_vals)if x2_vals and y2_vals:plt.semilogy(x2_vals, y2_vals, linestyle=':')plt.legend(legend)# plt.show()訓練模型:
batch_size, num_epochs, lr = 1, 100, 0.003 net, loss = linreg, squared_lossdataset = torch.utils.data.TensorDataset(train_features, train_labels) train_iter = torch.utils.data.DataLoader(dataset, batch_size, shuffle=True)def fit_and_plot(lambd):w, b = init_params()train_ls, test_ls = [], []for _ in range(num_epochs):for X, y in train_iter:# 添加了L2范數懲罰項l = loss(net(X, w, b), y) + lambd * l2_penalty(w)l = l.sum()if w.grad is not None:w.grad.data.zero_()b.grad.data.zero_()l.backward()sgd([w, b], lr, batch_size)train_ls.append(loss(net(train_features, w, b), train_labels).mean().item())test_ls.append(loss(net(test_features, w, b), test_labels).mean().item())semilogy(range(1, num_epochs + 1), train_ls, 'epochs', 'loss',range(1, num_epochs + 1), test_ls, ['train', 'test'])print('L2 norm of w:', w.norm().item())訓練并測試高維線性回歸模型。當lambd設為0時,我們沒有使用權重衰減。結果訓練誤差遠小于測試集上的誤差。這是典型的過擬合現象。
fit_and_plot(lambd=0)使用權重衰減
fit_and_plot(lambd=3)你會發(fā)現訓練誤差雖然有所提高,但測試集上的誤差有所下降。
可以直接在構造優(yōu)化器實例時通過weight_decay參數來指定權重衰減超參數默認下,PyTorch會對權重和偏差同時衰減。我們可以分別對權重和偏差構造優(yōu)化器實例,從而只對權重衰減。
修改上面的訓練代碼:
通過設置不同的衰減權重:
fit_and_plot_pytorch(0) #labmda=0,不衰減 fit_and_plot_pytorch(3) #labmda=3,衰減總結
以上是生活随笔為你收集整理的(pytorch-深度学习系列)pytorch避免过拟合-权重衰减的实现-学习笔记的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 如何理解马尔可夫决策过程?
- 下一篇: pb 打印html页面,用PB开发WEB