上节用了Sequential类来构造模型。这里我们另外一种基于Block类的模型构造方法,它让构造模型更加灵活,也将让你能更好的理解Sequential的运行机制。

回顾:

  • 序列模型生成
  • 层填充
  • 初始化模型参数
  1. net = gluon.nn.Sequential()
  2. with net.name_scope():
  3. net.add(gluon.nn.Dense(1))
  4. net.collect_params().initialize(mx.init.Normal(sigma=1)) # 模型参数初始化选择normal分布

两点讲解:

super(MLP, self).__init__(**kwargs):调用nn.Block的__init__,提供了prefix(指定名称)和params(指定参数)两个参数。

self.name_scope():调用nn.Block的name_scope,给域内层、参数名加上前缀prefix,和TensorFlow类似

继承Block类来构造模型

  • 存储参数
  • 定义向前传播如何进行
  • 自动求导

Block类是gluon.nn里提供的一个模型构造类,我们可以继承它来定义我们想要的模型。例如,我们在这里构造一个同前提到的相同的多层感知机。这里定义的MLP类重载了Block类的两个函数:__init__forward.

  1. from mxnet import nd
  2. from mxnet.gluon import nn
  3.  
  4. class MLP(nn.Block):
  5. # 声明带有模型参数的层,这里我们声明了两个全链接层。
  6. def __init__(self, **kwargs):
  7. # 调用 MLP 父类 Block 的构造函数来进行必要的初始化。这样在构造实例时还可以指定
  8. # 其他函数参数,例如下下一节将介绍的模型参数 params.
  9. super(MLP, self).__init__(**kwargs)
  10. # 隐藏层。
  11. self.hidden = nn.Dense(256, activation='relu')
  12. # 输出层。
  13. self.output = nn.Dense(10)
  14. # 定义模型的前向计算,即如何根据输出计算输出。
  15. def forward(self, x):
  16. return self.output(self.hidden(x))

建立之后进行forward测试,

  1. x = nd.random.uniform(shape=(2,20))
  2. net = MLP()
  3. net.initialize()
  4. net(x)
  1. [[ 0.09543004 0.04614332 -0.00286654 -0.07790349 -0.05130243 0.02942037
  2. 0.08696642 -0.0190793 -0.04122177 0.05088576]
  3. [ 0.0769287 0.03099705 0.00856576 -0.04467199 -0.06926839 0.09132434
  4. 0.06786595 -0.06187842 -0.03436673 0.04234694]]
  5. <NDArray 2x10 @cpu(0)>

其中,net(x)会调用了MLP继承至Block的__call__函数,这个函数将调用MLP定义的forward函数来完成前向计算。

我们无需在这里定义反向传播函数,系统将通过自动求导,来自动生成backward函数。

注意到我们不是将Block叫做层或者模型之类的名字,这是因为它是一个可以自由组建的部件。它的子类既可以一个层,例如Gluon提供的Dense类,也可以是一个模型,我们定义的MLP类,或者是模型的一个部分,例如ResNet的残差块。我们下面通过两个例子说明它。

Sequential:Block的容器

Sequential类继承自Block类,实质来说就是将初始化各个层的过程从__init__移到了add方法中。

当模型的前向计算就是简单串行计算模型里面各个层的时候,我们可以将模型定义变得更加简单,这个就是Sequential类的目的,它通过add函数来添加Block子类实例,前向计算时就是将添加的实例逐一运行。下面我们实现一个跟Sequential类有相同功能的类,这样你可以看的更加清楚它的运行机制。

  1. class MySequential(nn.Block):
  2. def __init__(self, **kwargs):
  3. super(MySequential, self).__init__(**kwargs)
  4.  
  5. def add(self, block):
  6. # block 是一个 Block 子类实例,假设它有一个独一无二的名字。我们将它保存在
  7. # Block 类的成员变量 _children 里,其类型是 OrderedDict. 当调用
  8. # initialize 函数时,系统会自动对 _children 里面所有成员初始化。
  9. self._children[block.name] = block
  10.  
  11. def forward(self, x):
  12. # OrderedDict 保证会按照插入时的顺序遍历元素。
  13. for block in self._children.values():
  14. x = block(x)
  15. return

我们用MySequential类来实现MLP类:

  1. net = MySequential()
  2. net.add(nn.Dense(256, activation='relu'))
  3. net.add(nn.Dense(10))
  4. net.initialize()
  5. net(x)
  1. [[ 0.00362228 0.00633332 0.03201144 -0.01369375 0.10336449 -0.03508018
  2. -0.00032164 -0.01676023 0.06978628 0.01303309]
  3. [ 0.03871715 0.02608213 0.03544959 -0.02521311 0.11005433 -0.0143066
  4. -0.03052466 -0.03852827 0.06321152 0.0038594 ]]
  5. <NDArray 2x10 @cpu(0)>

构造复杂的模型

虽然Sequential类可以使得模型构造更加简单,不需要定义forward函数,但直接继承Block类可以极大的拓展灵活性。下面我们构造一个稍微复杂点的网络:

  1. 前向计算中使用了NDArray函数和Python的控制流:forward函数内部是自由发挥的舞台
  2. 多次调用同一层
  1. class FancyMLP(nn.Block):
  2. def __init__(self, **kwargs):
  3. super(FancyMLP, self).__init__(**kwargs)
  4. # 不会被更新的随机权重。
  5. self.rand_weight = nd.random.uniform(shape=(20, 20))
  6. self.dense = nn.Dense(20, activation='relu')
  7.  
  8. def forward(self, x):
  9. x = self.dense(x)
  10. # 使用了 nd 包下 relu 和 dot 函数。
  11. x = nd.relu(nd.dot(x, self.rand_weight) + 1)
  12. # 重用了 dense,等价于两层网络但共享了参数。
  13. x = self.dense(x)
  14. # 控制流,这里我们需要调用 asscalar 来返回标量进行比较。
  15. while x.norm().asscalar() > 1:
  16. x /= 2
  17. if x.norm().asscalar() < 0.8:
  18. x *= 10
  19. return x.sum()

在这个FancyMLP模型中,我们使用了常数权重rand_weight(注意它不是模型参数)、做了矩阵乘法操作(nd.dot)并重复使用了相同的Dense层。测试一下:

  1. net = FancyMLP()
  2. net.initialize()
  3. net(x)

[ 18.57195282]

<NDArray 1 @cpu(0)>

由于FancyMLP和Sequential都是Block的子类,我们可以嵌套调用他们:

  1. class NestMLP(nn.Block):
  2. def __init__(self, **kwargs):
  3. super(NestMLP, self).__init__(**kwargs)
  4. self.net = nn.Sequential()
  5. self.net.add(nn.Dense(64, activation='relu'),
  6. nn.Dense(32, activation='relu'))
  7. self.dense = nn.Dense(16, activation='relu')
  8.  
  9. def forward(self, x):
  10. return self.dense(self.net(x))
  11.  
  12. net = nn.Sequential()
  13. net.add(NestMLP(), nn.Dense(20), FancyMLP())
  14.  
  15. net.initialize()
  16. net(x)

[ 24.86621094]

<NDArray 1 @cpu(0)>

『MXNet』第二弹_Gluon构建模型的更多相关文章

  1. 『PyTorch』第二弹重置_Tensor对象

    『PyTorch』第二弹_张量 Tensor基础操作 简单的初始化 import torch as t Tensor基础操作 # 构建张量空间,不初始化 x = t.Tensor(5,3) x -2. ...

  2. 关于『HTML5』:第二弹

    关于『HTML5』:第二弹 建议缩放90%食用 咕咕咕咕咕咕咕!!1 (蒟蒻大鸽子终于更新啦) 自开学以来,经过了「一脸蒙圈的 半期考试」.「二脸蒙圈的 体测」的双重洗礼,我终于有空肝 HTML5 辣 ...

  3. 关于『Markdown』:第二弹

    关于『Markdown』:第二弹 建议缩放90%食用 道家有云:一生二,二生三,三生万物 为什么我的帖子不是这样 各位打工人们! 自从我学了Markdown以来 发现 Markdown 语法真的要比 ...

  4. 关于『HTML』:第二弹

    关于『HTML』:第二弹 建议缩放90%食用 第二弹! 它来了! 它来了! 我竟然没有拖更,对了,你们昨天用草稿纸了么 开始正文之前提一个问题:大家知道"%%%"是什么意思吗?就这 ...

  5. 『MXNet』第一弹_基础架构及API

    MXNet是基础,Gluon是封装,两者犹如TensorFlow和Keras,不过得益于动态图机制,两者交互比TensorFlow和Keras要方便得多,其基础操作和pytorch极为相似,但是方便不 ...

  6. 『MXNet』第九弹_分类器以及迁移学习DEMO

    解压文件命令: with zipfile.ZipFile('../data/kaggle_cifar10/' + fin, 'r') as zin: zin.extractall('../data/k ...

  7. 『TensorFlow』第二弹_线性拟合&神经网络拟合_恰是故人归

    Step1: 目标: 使用线性模拟器模拟指定的直线:y = 0.1*x + 0.3 代码: import tensorflow as tf import numpy as np import matp ...

  8. 『PyTorch』第二弹_张量

    参考:http://www.jianshu.com/p/5ae644748f21# 几个数学概念: 标量(Scalar)是只有大小,没有方向的量,如1,2,3等 向量(Vector)是有大小和方向的量 ...

  9. 『MXNet』专题汇总

    MXNet文档 MXNet官方教程 持久化模型 框架介绍 『MXNet』第一弹_基础架构及API 『MXNet』第二弹_Gluon构建模型 『MXNet』第三弹_Gluon模型参数 『MXNet』第四 ...

随机推荐

  1. C语言变量的作用域和存储类型

    1.动态局部变量:也称局部变量.自动变量,是指在函数内部定义的自动变量,不带static修饰,作用域是定义该变量的子程序.在退出函数后,变量自带内存会自动释放. 2.静态局部变量:是指在函数内部定义的 ...

  2. 【C#】C#学习笔记_1

    C#的程序入口为某一个类里面的static void Main(string[] args){}方法,如果一个工程有多个Main方法,那么需要在工程配置中选择一个作为程序入口. C#的输入.输出操作在 ...

  3. Java基础 【Math、Random、System、BigInteger、BigDecimal、Date、Calendar等常用类的使用】

    学习的这几个类  是日常工作中经常要使用到的类 Math 类包含用于执行基本数序运算的方法,如初等指数.对数.平方根和 三角函数. 成员方法 1.public static int abs(int a ...

  4. 浅谈IIS 和 asp.net的应用之间的关系

    IIS可以理解为一个web服务器. 用于提供web相关的各种服务. IIS6.0中添加了一个新的功能, application pool. application pool的作用是将运行在同一个ser ...

  5. CentOS7使用firewalld和selinux

    转载自莫小安的博客:https://www.cnblogs.com/moxiaoan/p/5683743.html 如何查看和使用selinux https://blog.csdn.net/edide ...

  6. JMeter中关于动态切换不同CSV文件解决方案

    最近写case,需要当前播放节目的数据作为输入数据,所以每个时刻所用的数据只能是当前时刻附件的数据,尝试用CSV Data Set Config动态加载不同的文件,没有成功,好像CSV Data Se ...

  7. Scala的配置

    Scala基于Java的JVM,所以先检查是否安装JDK. 在官网上下载并安装好了之后,就是配置环境变量了. SCALA_HOME 变量:C:\Program Files (x86)\scala. P ...

  8. CSS3一些常用动画

    /* animation */ .a-bounce,.a-flip,.a-flash,.a-shake,.a-swing,.a-wobble,.a-ring{-webkit-animation:1s ...

  9. Qt5鼠标事件及实例

    mainwindow.h #ifndef MAINWINDOW_H #define MAINWINDOW_H #include <QMainWindow> #include <QLa ...

  10. MYSQL的常用函数(字符串函数)

    ASCII(char)返回字符的ASCII码值 BIT_LENGTH(str)返回字符串的比特长度 CONCAT(s1,s2...,sn)将s1,s2...,sn连接成字符串 CONCAT_WS(se ...