• 【机器学习】037_暂退法


    一、实现原理

    具有输入噪音的训练,等价于Tikhonov正则化

    核心方法:在前向传播的过程中,计算每一内部层的同时注入噪声

    · 从作用上来看,表面上来说是在训练过程中丢弃一些神经元

    · 假设x是某一层神经网络层的输出,是下一层的输入,我们希望对x加入一些噪音,使得:

    E[x^`]=x

      ※x`的期望为x,也就是说平均上来说输出值还是x

    · 暂退法对每个元素进行了如下扰动:

            有p的概率下取值:x^`_i=0

            其它情况(1-p概率):x^`_i = \frac{x_i}{1-p}

    实践中使用暂退法:

    · 通常将暂退法作用在全连接隐藏层的输出上

    如图所示,在第一个隐藏层的输出上,有些神经元有p的概率使输出值置零。

    非置零的输出值,即有1-p的概率被施加了一个较小的扰动值使其略微增大。

    ※暂退法只在训练中使用,dropout是正则项,在推理过程中不会使用,这样也会保证输出值确定

    ※每次执行暂退法的时候,实际上是每次随机采样了一些子神经网络

    总结:

    ①暂退法将一些输出项随机置零来控制模型的复杂度

    ②暂退法的作用效果和正则化等价

    ③常应用在多层感知机的隐藏层输出上

    ④丢弃概率p是控制模型复杂度的超参数

    二、代码实现

    从零实现代码:

    1. import torch
    2. from torch import nn
    3. from d2l import torch as d2l
    4. def dropout_layer(X, dropout):
    5. # assert用于选择dropout符合范围的情况,不符合则报错
    6. assert 0 <= dropout <= 1, "不符合范围!"
    7. # 在本情况中,所有元素都被丢弃
    8. if dropout == 1:
    9. return torch.zeros_like(X)
    10. # 在本情况中,所有元素都被保留
    11. if dropout == 0:
    12. return X
    13. # 在这一步操作中,首先定义一个和X张量形状相同但元素值均为随机数的张量
    14. # 将这个张量里每个元素与dropout比较,如果大于就置为True,小于等于就置为False
    15. # 再调用float将True和False转化为1和0
    16. # 这样,mask就是一个仅含1与0的张量了
    17. # 最后将mask里的每个元素与X里的每个元素做数乘
    18. mask = (torch.rand(X.shape) > dropout).float()
    19. return mask * X / (1.0 - dropout)
    20. # 生成X来测试暂退法
    21. X= torch.arange(16, dtype = torch.float32).reshape((2, 8))
    22. print(X)
    23. print(dropout_layer(X, 0.))
    24. print(dropout_layer(X, 0.5))
    25. print(dropout_layer(X, 1.))
    26. # 定义模型参数
    27. num_inputs, num_outputs, num_hiddens1, num_hiddens2 = 784, 10, 256, 256
    28. # 定义模型
    29. dropout1, dropout2 = 0.2, 0.5
    30. # is_training用来表示当前是在测试还是在训练
    31. class Net(nn.Module):
    32. def __init__(self, num_inputs, num_outputs, num_hiddens1, num_hiddens2,
    33. is_training = True):
    34. super(Net, self).__init__()
    35. self.num_inputs = num_inputs
    36. self.training = is_training
    37. self.lin1 = nn.Linear(num_inputs, num_hiddens1)
    38. self.lin2 = nn.Linear(num_hiddens1, num_hiddens2)
    39. self.lin3 = nn.Linear(num_hiddens2, num_outputs)
    40. self.relu = nn.ReLU()
    41. def forward(self, X):
    42. H1 = self.relu(self.lin1(X.reshape((-1, self.num_inputs))))
    43. # 只有在训练模型时才使用dropout
    44. if self.training == True:
    45. # 在第一个全连接层之后添加一个dropout层
    46. H1 = dropout_layer(H1, dropout1)
    47. H2 = self.relu(self.lin2(H1))
    48. if self.training == True:
    49. # 在第二个全连接层之后添加一个dropout层
    50. H2 = dropout_layer(H2, dropout2)
    51. # 输出不需要dropout作用
    52. out = self.lin3(H2)
    53. return out
    54. net = Net(num_inputs, num_outputs, num_hiddens1, num_hiddens2)
    55. # 训练、测试模型
    56. num_epochs, lr, batch_size = 10, 0.5, 256
    57. loss = nn.CrossEntropyLoss(reduction='none')
    58. train_iter, test_iter = d2l.load_data_fashion_mnist(batch_size)
    59. trainer = torch.optim.SGD(net.parameters(), lr=lr)
    60. d2l.train_ch3(net, train_iter, test_iter, loss, num_epochs, trainer)

    简洁实现代码:

    1. import torch
    2. from torch import nn
    3. from d2l import torch as d2l
    4. # 定义概率参数
    5. dropout1, dropout2 = 0.2, 0.5
    6. net = nn.Sequential(nn.Flatten(),
    7. nn.Linear(784, 256),
    8. nn.ReLU(),
    9. # 在第一个全连接层之后添加一个dropout层
    10. nn.Dropout(dropout1),
    11. nn.Linear(256, 256),
    12. nn.ReLU(),
    13. # 在第二个全连接层之后添加一个dropout层
    14. nn.Dropout(dropout2),
    15. nn.Linear(256, 10))
    16. def init_weights(m):
    17. if type(m) == nn.Linear:
    18. nn.init.normal_(m.weight, std=0.01)
    19. net.apply(init_weights);
    20. # 训练、测试模型
    21. num_epochs, lr, batch_size = 10, 0.5, 256
    22. loss = nn.CrossEntropyLoss(reduction='none')
    23. train_iter, test_iter = d2l.load_data_fashion_mnist(batch_size)
    24. trainer = torch.optim.SGD(net.parameters(), lr=lr)
    25. d2l.train_ch3(net, train_iter, test_iter, loss, num_epochs, trainer)

  • 相关阅读:
    【LeetCode75】第六十四题 最长公共子序列
    “干翻”GPT-3,Meta 用开源发起攻势
    uniapp——实现聊天室功能——技能提升
    js中循环判断找到满足条件的单项后结束循环
    【无标题】
    SpringBoot 集成 AKKA
    EN 12259-2固定消防系统湿式报警阀组件—CE认证
    Ansible 的脚本 --- playbook 剧本
    五月集训(第28天) —— 动态规划
    零知识证明: Tornado Cash 项目学习
  • 原文地址:https://blog.csdn.net/Yukiice/article/details/134497165