pytorch中的BN层简介

简介

BN层在训练过程中,会将一个Batch的中的数据转变成正太分布,在推理过程中使用训练过程中的参数对数据进行处理,然而网络并不知道你是在训练还是测试阶段,因此,需要手动的加上,需要在测试和训练阶段使用如下函数。

model.train() or model.eval()

在Pytorch中,BN层的类的参数有:

torch.nn.BatchNorm2d(num_features, eps=1e-05, momentum=0.1, affine=True, track_running_stats=True)

其中affine定义了BN层的参数γ和β是否是可学习的(不可学习默认是常数1和0). BN层的统计数据更新是在每一次训练阶段model.train()后的forward()方法中自动实现的,而不是在梯度计算与反向传播中更新optim.step()中完成。

pytorch里BN层的具体实现过程

BN层的输出Y与输入X之间的关系是:
Y = ( X − r u n n i n g m e a n ) / s q r t ( r u n n i n g v a r + e p s ) ∗ g a m m a + b e t a Y = (X - running_mean) / sqrt(running_var + eps) * gamma + beta Y=(Xrunningmean)/sqrt(runningvar+eps)gamma+beta

  • 其中gamma、beta为可学习参数(在pytorch中分别改叫weight和bias),训练时通过反向传播更新;
  • 而running_mean、running_var则是在前向时先由X计算出mean和var,再由mean和var以动量momentum来更新running_mean和running_var。
  • 所以在训练阶段,running_mean和running_var在每次前向时更新一次;在测试阶段,则通过net.eval()固定该BN层的running_mean和running_var,此时这两个值即为训练阶段最后一次前向时确定的值,并在整个测试阶段保持不变。

momentum的定义

Pytorch中的BN层的动量平滑和常见的动量法计算方式是相反的,默认的momentum=0.1

冻结BN及其统计数据

冻结BN的方式是在模型训练时,把BN单独挑出来,重新设置其状态为eval (在model.train()之后覆盖training状态)

  • 方法一
def set_bn_eval(m):
    classname = m.__class__.__name__
    if classname.find('BatchNorm') != -1:
      m.eval()

model.apply(set_bn_eval)
  • 方法二,重写module中的train()方法
def train(self, mode=True):
        """  Override the default train() to freeze the BN parameters  """
        super(MyNet, self).train(mode)
        if self.freeze_bn:
            print("Freezing Mean/Var of BatchNorm2D.")
            if self.freeze_bn_affine:
                print("Freezing Weight/Bias of BatchNorm2D.")
        if self.freeze_bn:
            for m in self.backbone.modules():
                if isinstance(m, nn.BatchNorm2d):
                    m.eval()
                    if self.freeze_bn_affine:
                        m.weight.requires_grad = False
                        m.bias.requires_grad = False
PytorchBN层是Batch Normalization的缩写,用于在深度学习模型对输入数据进行归一化处理。BN层的作用是通过对每个小批量的输入数据进行归一化,使得模型在训练过程更加稳定和快速收敛。\[1\] 在Pytorch,使用BN层的方法如下所示: ```python from torch import nn # 创建一个BN层对象,需要传入特征的通道数num_features作为参数 bn = nn.BatchNorm2d(num_features) # 输入数据 input = torch.randn(batch_size, num_features, height, width) # 将输入数据传入BN层进行处理 output = bn(input) ``` 其,`num_features`表示输入数据的通道数,`batch_size`表示输入数据的批量大小,`height`和`width`表示输入数据的高度和宽度。\[1\] 在BN层的类,还有一些其他的参数可以进行设置,例如`eps`表示用于数值稳定性的小值,默认为1e-5;`momentum`表示用于计算移动平均的动量,默认为0.1;`affine`表示是否学习BN层的参数γ和β,默认为True;`track_running_stats`表示是否跟踪训练过程的统计数据,默认为True。\[2\] 需要注意的是,BN层的参数γ和β是否可学习是由`affine`参数控制的,默认情况下是可学习的,即可通过反向传播进行更新。而BN层的统计数据更新是在每一次训练阶段的`model.train()`后的`forward()`方法自动实现的,而不是在梯度计算与反向传播更新`optim.step()`完成。\[3\] #### 引用[.reference_title] - *1* [一起来学PyTorch——神经网络(BN层)](https://blog.csdn.net/TomorrowZoo/article/details/129531658)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down1,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* *3* [pytorchBN层简介](https://blog.csdn.net/lpj822/article/details/109772094)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insert_down1,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值