PyTorch上搭建簡單神經網絡實現(xiàn)回歸和分類的示例
本文介紹了PyTorch上搭建簡單神經網絡實現(xiàn)回歸和分類的示例,分享給大家,具體如下:

一、PyTorch入門
1. 安裝方法
登錄PyTorch官網,http://pytorch.org,可以看到以下界面:

按上圖的選項選擇后即可得到Linux下conda指令:
conda install pytorch torchvision -c soumith
目前PyTorch僅支持MacOS和Linux,暫不支持Windows。安裝 PyTorch 會安裝兩個模塊,一個是torch,一個 torchvision, torch 是主模塊,用來搭建神經網絡的,torchvision 是輔模塊,有數(shù)據庫,還有一些已經訓練好的神經網絡等著你直接用,比如 (VGG, AlexNet, ResNet)。
2. Numpy與Torch
torch_data = torch.from_numpy(np_data)可以將numpy(array)格式轉換為torch(tensor)格式;torch_data.numpy()又可以將torch的tensor格式轉換為numpy的array格式。注意Torch的Tensor和numpy的array會共享他們的存儲空間,修改一個會導致另外的一個也被修改。
對于1維(1-D)的數(shù)據,numpy是以行向量的形式打印輸出,而torch是以列向量的形式打印輸出的。
其他例如sin, cos, abs,mean等numpy中的函數(shù)在torch中用法相同。需要注意的是,numpy中np.matmul(data, data)和data.dot(data)矩陣相乘會得到相同結果;torch中torch.mm(tensor, tensor)是矩陣相乘的方法,得到一個矩陣,tensor.dot(tensor)會把tensor轉換為1維的tensor,然后逐元素相乘后求和,得到與一個實數(shù)。
相關代碼:
import torch import numpy as np np_data = np.arange(6).reshape((2, 3)) torch_data = torch.from_numpy(np_data) # 將numpy(array)格式轉換為torch(tensor)格式 tensor2array = torch_data.numpy() print( '\nnumpy array:\n', np_data, '\ntorch tensor:', torch_data, '\ntensor to array:\n', tensor2array, ) # torch數(shù)據格式在print的時候前后自動添加換行符 # abs data = [-1, -2, 2, 2] tensor = torch.FloatTensor(data) print( '\nabs', '\nnumpy: \n', np.abs(data), '\ntorch: ', torch.abs(tensor) ) # 1維的數(shù)據,numpy是行向量形式顯示,torch是列向量形式顯示 # sin print( '\nsin', '\nnumpy: \n', np.sin(data), '\ntorch: ', torch.sin(tensor) ) # mean print( '\nmean', '\nnumpy: ', np.mean(data), '\ntorch: ', torch.mean(tensor) ) # 矩陣相乘 data = [[1,2], [3,4]] tensor = torch.FloatTensor(data) print( '\nmatrix multiplication (matmul)', '\nnumpy: \n', np.matmul(data, data), '\ntorch: ', torch.mm(tensor, tensor) ) data = np.array(data) print( '\nmatrix multiplication (dot)', '\nnumpy: \n', data.dot(data), '\ntorch: ', tensor.dot(tensor) )
3. Variable
PyTorch中的神經網絡來自于autograd包,autograd包提供了Tensor所有操作的自動求導方法。
autograd.Variable這是這個包中最核心的類??梢詫ariable理解為一個裝有tensor的容器,它包裝了一個Tensor,并且?guī)缀踔С炙械亩x在其上的操作。一旦完成運算,便可以調用 .backward()來自動計算出所有的梯度。也就是說只有把tensor置于Variable中,才能在神經網絡中實現(xiàn)反向傳遞、自動求導等運算。
可以通過屬性 .data 來訪問原始的tensor,而關于這一Variable的梯度則可通過 .grad屬性查看。
相關代碼:
import torch
from torch.autograd import Variable
tensor = torch.FloatTensor([[1,2],[3,4]])
variable = Variable(tensor, requires_grad=True)
# 打印展示Variable類型
print(tensor)
print(variable)
t_out = torch.mean(tensor*tensor) # 每個元素的^ 2
v_out = torch.mean(variable*variable)
print(t_out)
print(v_out)
v_out.backward() # Variable的誤差反向傳遞
# 比較Variable的原型和grad屬性、data屬性及相應的numpy形式
print('variable:\n', variable)
# v_out = 1/4 * sum(variable*variable) 這是計算圖中的 v_out 計算步驟
# 針對于 v_out 的梯度就是, d(v_out)/d(variable) = 1/4*2*variable = variable/2
print('variable.grad:\n', variable.grad) # Variable的梯度
print('variable.data:\n', variable.data) # Variable的數(shù)據
print(variable.data.numpy()) #Variable的數(shù)據的numpy形式
部分輸出結果:
variable:
Variable containing:
1 2
3 4
[torch.FloatTensor of size 2x2]
variable.grad:
Variable containing:
0.5000 1.0000
1.5000 2.0000
[torch.FloatTensor of size 2x2]
variable.data:
1 2
3 4
[torch.FloatTensor of size 2x2]
[[ 1. 2.]
[ 3. 4.]]
4. 激勵函數(shù)activationfunction
Torch的激勵函數(shù)都在torch.nn.functional中,relu,sigmoid, tanh, softplus都是常用的激勵函數(shù)。

相關代碼:
import torch import torch.nn.functional as F from torch.autograd import Variable import matplotlib.pyplot as plt x = torch.linspace(-5, 5, 200) x_variable = Variable(x) #將x放入Variable x_np = x_variable.data.numpy() # 經過4種不同的激勵函數(shù)得到的numpy形式的數(shù)據結果 y_relu = F.relu(x_variable).data.numpy() y_sigmoid = F.sigmoid(x_variable).data.numpy() y_tanh = F.tanh(x_variable).data.numpy() y_softplus = F.softplus(x_variable).data.numpy() plt.figure(1, figsize=(8, 6)) plt.subplot(221) plt.plot(x_np, y_relu, c='red', label='relu') plt.ylim((-1, 5)) plt.legend(loc='best') plt.subplot(222) plt.plot(x_np, y_sigmoid, c='red', label='sigmoid') plt.ylim((-0.2, 1.2)) plt.legend(loc='best') plt.subplot(223) plt.plot(x_np, y_tanh, c='red', label='tanh') plt.ylim((-1.2, 1.2)) plt.legend(loc='best') plt.subplot(224) plt.plot(x_np, y_softplus, c='red', label='softplus') plt.ylim((-0.2, 6)) plt.legend(loc='best') plt.show()
二、PyTorch實現(xiàn)回歸
先看完整代碼:
import torch
from torch.autograd import Variable
import torch.nn.functional as F
import matplotlib.pyplot as plt
x = torch.unsqueeze(torch.linspace(-1, 1, 100), dim=1) # 將1維的數(shù)據轉換為2維數(shù)據
y = x.pow(2) + 0.2 * torch.rand(x.size())
# 將tensor置入Variable中
x, y = Variable(x), Variable(y)
#plt.scatter(x.data.numpy(), y.data.numpy())
#plt.show()
# 定義一個構建神經網絡的類
class Net(torch.nn.Module): # 繼承torch.nn.Module類
def __init__(self, n_feature, n_hidden, n_output):
super(Net, self).__init__() # 獲得Net類的超類(父類)的構造方法
# 定義神經網絡的每層結構形式
# 各個層的信息都是Net類對象的屬性
self.hidden = torch.nn.Linear(n_feature, n_hidden) # 隱藏層線性輸出
self.predict = torch.nn.Linear(n_hidden, n_output) # 輸出層線性輸出
# 將各層的神經元搭建成完整的神經網絡的前向通路
def forward(self, x):
x = F.relu(self.hidden(x)) # 對隱藏層的輸出進行relu激活
x = self.predict(x)
return x
# 定義神經網絡
net = Net(1, 10, 1)
print(net) # 打印輸出net的結構
# 定義優(yōu)化器和損失函數(shù)
optimizer = torch.optim.SGD(net.parameters(), lr=0.5) # 傳入網絡參數(shù)和學習率
loss_function = torch.nn.MSELoss() # 最小均方誤差
# 神經網絡訓練過程
plt.ion() # 動態(tài)學習過程展示
plt.show()
for t in range(300):
prediction = net(x) # 把數(shù)據x喂給net,輸出預測值
loss = loss_function(prediction, y) # 計算兩者的誤差,要注意兩個參數(shù)的順序
optimizer.zero_grad() # 清空上一步的更新參數(shù)值
loss.backward() # 誤差反相傳播,計算新的更新參數(shù)值
optimizer.step() # 將計算得到的更新值賦給net.parameters()
# 可視化訓練過程
if (t+1) % 10 == 0:
plt.cla()
plt.scatter(x.data.numpy(), y.data.numpy())
plt.plot(x.data.numpy(), prediction.data.numpy(), 'r-', lw=5)
plt.text(0.5, 0, 'L=%.4f' % loss.data[0], fontdict={'size': 20, 'color': 'red'})
plt.pause(0.1)
首先創(chuàng)建一組帶噪聲的二次函數(shù)擬合數(shù)據,置于Variable中。定義一個構建神經網絡的類Net,繼承torch.nn.Module類。Net類的構造方法中定義輸入神經元、隱藏層神經元、輸出神經元數(shù)量的參數(shù),通過super()方法獲得Net父類的構造方法,以屬性的方式定義Net的各個層的結構形式;定義Net的forward()方法將各層的神經元搭建成完整的神經網絡前向通路。
定義好Net類后,定義神經網絡實例,Net類實例可以直接print打印輸出神經網絡的結構信息。接著定義神經網絡的優(yōu)化器和損失函數(shù)。定義好這些后就可以進行訓練了。optimizer.zero_grad()、loss.backward()、optimizer.step()分別是清空上一步的更新參數(shù)值、進行誤差的反向傳播并計算新的更新參數(shù)值、將計算得到的更新值賦給net.parameters()。循環(huán)迭代訓練過程。
運行結果:
Net (
(hidden): Linear (1 -> 10)
(predict): Linear (10 -> 1)
)

三、PyTorch實現(xiàn)簡單分類
完整代碼:
import torch
from torch.autograd import Variable
import torch.nn.functional as F
import matplotlib.pyplot as plt
# 生成數(shù)據
# 分別生成2組各100個數(shù)據點,增加正態(tài)噪聲,后標記以y0=0 y1=1兩類標簽,最后cat連接到一起
n_data = torch.ones(100,2)
# torch.normal(means, std=1.0, out=None)
x0 = torch.normal(2*n_data, 1) # 以tensor的形式給出輸出tensor各元素的均值,共享標準差
y0 = torch.zeros(100)
x1 = torch.normal(-2*n_data, 1)
y1 = torch.ones(100)
x = torch.cat((x0, x1), 0).type(torch.FloatTensor) # 組裝(連接)
y = torch.cat((y0, y1), 0).type(torch.LongTensor)
# 置入Variable中
x, y = Variable(x), Variable(y)
class Net(torch.nn.Module):
def __init__(self, n_feature, n_hidden, n_output):
super(Net, self).__init__()
self.hidden = torch.nn.Linear(n_feature, n_hidden)
self.out = torch.nn.Linear(n_hidden, n_output)
def forward(self, x):
x = F.relu(self.hidden(x))
x = self.out(x)
return x
net = Net(n_feature=2, n_hidden=10, n_output=2)
print(net)
optimizer = torch.optim.SGD(net.parameters(), lr=0.012)
loss_func = torch.nn.CrossEntropyLoss()
plt.ion()
plt.show()
for t in range(100):
out = net(x)
loss = loss_func(out, y) # loss是定義為神經網絡的輸出與樣本標簽y的差別,故取softmax前的值
optimizer.zero_grad()
loss.backward()
optimizer.step()
if t % 2 == 0:
plt.cla()
# 過了一道 softmax 的激勵函數(shù)后的最大概率才是預測值
# torch.max既返回某個維度上的最大值,同時返回該最大值的索引值
prediction = torch.max(F.softmax(out), 1)[1] # 在第1維度取最大值并返回索引值
pred_y = prediction.data.numpy().squeeze()
target_y = y.data.numpy()
plt.scatter(x.data.numpy()[:, 0], x.data.numpy()[:, 1], c=pred_y, s=100, lw=0, cmap='RdYlGn')
accuracy = sum(pred_y == target_y)/200 # 預測中有多少和真實值一樣
plt.text(1.5, -4, 'Accu=%.2f' % accuracy, fontdict={'size': 20, 'color': 'red'})
plt.pause(0.1)
plt.ioff()
plt.show()
神經網絡結構部分的Net類與前文的回歸部分的結構相同。
需要注意的是,在循環(huán)迭代訓練部分,out定義為神經網絡的輸出結果,計算誤差loss時不是使用one-hot形式的,loss是定義在out與y上的torch.nn.CrossEntropyLoss(),而預測值prediction定義為out經過Softmax后(將結果轉化為概率值)的結果。
運行結果:
Net (
(hidden): Linear (2 -> 10)
(out):Linear (10 -> 2)
)

四、補充知識
1. super()函數(shù)
在定義Net類的構造方法的時候,使用了super(Net,self).__init__()語句,當前的類和對象作為super函數(shù)的參數(shù)使用,這條語句的功能是使Net類的構造方法獲得其超類(父類)的構造方法,不影響對Net類單獨定義構造方法,且不必關注Net類的父類到底是什么,若需要修改Net類的父類時只需修改class語句中的內容即可。
2. torch.normal()
torch.normal()可分為三種情況:(1)torch.normal(means,std, out=None)中means和std都是Tensor,兩者的形狀可以不必相同,但Tensor內的元素數(shù)量必須相同,一一對應的元素作為輸出的各元素的均值和標準差;(2)torch.normal(mean=0.0, std, out=None)中mean是一個可定義的float,各個元素共享該均值;(3)torch.normal(means,std=1.0, out=None)中std是一個可定義的float,各個元素共享該標準差。
3. torch.cat(seq, dim=0)
torch.cat可以將若干個Tensor組裝連接起來,dim指定在哪個維度上進行組裝。
4. torch.max()
(1)torch.max(input)→ float
input是tensor,返回input中的最大值float。
(2)torch.max(input,dim, keepdim=True, max=None, max_indices=None) -> (Tensor, LongTensor)
同時返回指定維度=dim上的最大值和該最大值在該維度上的索引值。
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關文章
使用wxPython和OpenCV實現(xiàn)手勢識別相機功能
在這篇博客中,我將分享一個有趣的?Python?項目:通過?wxPython?創(chuàng)建圖形界面,利用?OpenCV?的計算機視覺技術實現(xiàn)實時手勢識別,以下是項目的完整實現(xiàn)過程,包括代碼分析、使用說明和可能的優(yōu)化建議,需要的朋友可以參考下2025-04-04
python實現(xiàn)npy格式文件轉換為txt文件操作
這篇文章主要介紹了python實現(xiàn)npy格式文件轉換為txt文件操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-07-07
Python中在for循環(huán)中嵌套使用if和else語句的技巧
Python的語法糖非常強大,比如Python中在for循環(huán)中嵌套使用if和else語句的技巧便十分給力,下面我們就舉幾個例子來看詳細的用法:2016-06-06
python類參數(shù)定義及數(shù)據擴展方式unsqueeze/expand
本文主要介紹了python類參數(shù)定義及數(shù)據擴展方式unsqueeze/expand,文章通過圍繞主題展開詳細的內容介紹,具有一定的參考價值,需要的小伙伴可以參考一下2022-08-08
PyTorch中view()與?reshape()的區(qū)別詳析
這篇文章主要給大家介紹了關于PyTorch中view()?與?reshape()?區(qū)別的相關資料,文中通過實例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2022-01-01

