赞
踩
PyTorch代码中存在两种常用的模型构造方式
1.普通方法——在构造函数中定义模型不同的层,在forward函数中输入的X依据指定的顺序进行前向传递
- import torch
- import torch.nn as nn
-
- class Net(nn.Module):
- def __init__(self, n_feature, n_hidden, n_output):
- super(Net, self).__init__()
- self.hidden = nn.Linear(n_feature, n_hidden)
- self.predict = nn.Linear(n_hidden, n_output)
-
- def forward(self, x):
- x = F.relu(self.hidden(x)) # hidden后接relu层
- x = self.predict(x)
- return x
-
- model_1 = Net(1, 10, 1)
- print(model_1)
-
-
- '''运行结果为:
- Net(
- (hidden): Linear(in_features=1, out_features=10, bias=True)
- (predict): Linear(in_features=10, out_features=1, bias=True)
- )
- '''
2.使用nn.Sequential按顺序构造所有层,在forward函数中直接调用
nn.Sequential的特点:将容器视为单个模块,即一个模块可以包含许多层
nn.Sequential有三种常见定义模型的方式:
① 基本实现方式:顺序定义每一层,特点:每一层没有名字,仅能通过下标访问各层
- import torch
- import torch.nn as nn
-
- class Net(nn.Module):
- def __init__(self, n_feature, n_hidden, n_output):
- super(Net,self).__init__()
- self.net_1 = nn.Sequential(
- nn.Linear(n_feature, n_hidden),
- nn.ReLU(),
- nn.Linear(n_hidden, n_output)
- )
-
- def forward(self,x):
- x = self.net_1(x)
- return x
-
- model_2 = Net(1,10,1)
- print(model_2)
-
-
- '''运行结果为:
- Net(
- (net_1): Sequential(
- (0): Linear(in_features=1, out_features=10, bias=True)
- (1): ReLU()
- (2): Linear(in_features=10, out_features=1, bias=True)
- )
- )
- '''
② 给每一层自定义名称
- import torch.nn as nn
- from collections import OrderedDict
-
-
- model = nn.Sequential(OrderedDict([
- ('conv1', nn.Conv2d(1, 20, 5)),
- ('relu1', nn.ReLU()),
- ('conv2', nn.Conv2d(20, 64, 5)),
- ('relu2', nn.ReLU())
- ]))
-
- print(model)
- print(model[2]) # 通过索引获取第几个层
- print(model.conv1)
-
- '''运行结果为:
- Sequential(
- (conv1): Conv2d(1, 20, kernel_size=(5, 5), stride=(1, 1))
- (relu1): ReLU()
- (conv2): Conv2d(20, 64, kernel_size=(5, 5), stride=(1, 1))
- (relu2): ReLU()
- )
- Conv2d(20, 64, kernel_size=(5, 5), stride=(1, 1))
- Conv2d(1, 20, kernel_size=(5, 5), stride=(1, 1))
- '''
③ 使用add_module方法逐层加入Sequential中,该方法是从nn.Module类继承而来,nn.Sequental本身没有该方法,可通过自定义名称访问。
- 1 import torch.nn as nn
- 2 from collections import OrderedDict
- 3
-
- 4 model = nn.Sequential()
- 5 model.add_module("conv1", nn.Conv2d(1, 20, 5))
- 6 model.add_module('relu1', nn.ReLU())
- 7 model.add_module('conv2', nn.Conv2d(20, 64, 5))
- 8 model.add_module('relu2', nn.ReLU())
- 9
- 10 print(model)
- 11 print(model[2]) # 通过索引获取第几个层
- 12 print(model.conv1)
-
-
- 13 '''运行结果为:
- 14 Sequential(
- 15 (conv1): Conv2d(1, 20, kernel_size=(5, 5), stride=(1, 1))
- 16 (relu1): ReLU()
- 17 (conv2): Conv2d(20, 64, kernel_size=(5, 5), stride=(1, 1))
- 18 (relu2): ReLU()
- 19 )
- 20 Conv2d(20, 64, kernel_size=(5, 5), stride=(1, 1))
- 21 Conv2d(1, 20, kernel_size=(5, 5), stride=(1, 1))
- 22 '''
通过观察两种不同的模型构造方式,可以发现以下区别:
① nn.Sequential在模型定义和前向传递过程中,定义的层只需出现一次,可以简化代码
② 普通模型构造方法需要多个函数嵌套,在模型较大,层数较多时会非常混乱,容易出错
③ nn.Sequential是将内部各模块按顺序排列的,因此不适合实现特殊结构,比如ResNet就无法使用nn.Sequential完整表示出来(这是普通方法构造模型的必然性)
④ 较小的模型可以使用nn.Sequential或者普通方法构造模型,较大的模型可以使用nn.Sequential方法,比较复杂的模型可以结合两者优点,来达到既简化模型,又灵活实现的目的。
参考资料 :
(45条消息) nn.Sequential方法介绍_Raywit的博客-CSDN博客_nn.sequential
1、nn.Sequential类-使用Sequential类来自定义顺序连接模型 - 小吴的日常 - 博客园 (cnblogs.com)
最后:
受作者水平限制,文章可能有内容描述不够准确恰当,欢迎在评论中指出,文章将持续更新
谢谢!
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。