深度学习——残差网络(ResNet)

news2025/1/18 7:29:43

深度学习——残差网络(ResNet)

文章目录

  • 前言
  • 一、函数类
  • 二、残差块
  • 三、ResNet模型
  • 四、模型训练
  • 五、小结
  • 总结


前言

随着设计越来越深的网络,深刻理解“新添加的层如何提升神经网络的性能”变得至关重要。更重要的是设计网络的能力,在这种网络中,添加层会使网络更具表现力, 为了取得质的突破,我们需要一些数学基础知识。
本章主要学习残差网络

参考书:
《动手学深度学习》


一、函数类

首先,假设有一类特定的神经网络架构 F \mathcal{F} F,它包括学习速率和其他超参数设置。对于所有 f ∈ F f \in \mathcal{F} fF,存在一些参数集(例如权重和偏置),这些参数可以通过在合适的数据集上进行训练而获得。

现在假设 f ∗ f^* f是我们真正想要找到的函数,如果是 f ∗ ∈ F f^* \in \mathcal{F} fF,那我们可以轻而易举的训练得到它,但通常我们不会那么幸运。

相反,我们将尝试找到一个函数 f F ∗ f^*_\mathcal{F} fF,这是我们在 F \mathcal{F} F中的最佳选择。

例如,给定一个具有 X \mathbf{X} X特性和 y \mathbf{y} y标签的数据集,我们可以尝试通过解决以下优化问题来找到它:

f F ∗ : = a r g m i n f L ( X , y , f )  subject to  f ∈ F . f^*_\mathcal{F} := \mathop{\mathrm{argmin}}_f L(\mathbf{X}, \mathbf{y}, f) \text{ subject to } f \in \mathcal{F}. fF:=argminfL(X,y,f) subject to fF.

那么,怎样得到更近似真正 f ∗ f^* f的函数呢?

唯一合理的可能性是,我们需要设计一个更强大的架构 F ′ \mathcal{F}' F
换句话说,我们预计 f F ′ ∗ f^*_{\mathcal{F}'} fF f F ∗ f^*_{\mathcal{F}} fF“更近似”。
然而,如果 F ⊈ F ′ \mathcal{F} \not\subseteq \mathcal{F}' FF,则无法保证新的体系“更近似”。事实上, f F ′ ∗ f^*_{\mathcal{F}'} fF可能更糟:

如图中所示,对于非嵌套函数类,较复杂的函数类并不总是向“真”函数 f ∗ f^* f靠拢(复杂度由 F 1 \mathcal{F}_1 F1 F 6 \mathcal{F}_6 F6递增)。虽然 F 3 \mathcal{F}_3 F3 F 1 \mathcal{F}_1 F1更接近 f ∗ f^* f,但 F 6 \mathcal{F}_6 F6却离的更远了。
相反对于 图中右侧的嵌套函数类 F 1 ⊆ … ⊆ F 6 \mathcal{F}_1 \subseteq \ldots \subseteq \mathcal{F}_6 F1F6,可以避免上述问题。

在这里插入图片描述

因此,只有当较复杂的函数类包含较小的函数类时,我们才能确保提高它们的性能。

对于深度神经网络,如果我们能将新添加的层训练成恒等映射 f ( x ) = x f(\mathbf{x}) = \mathbf{x} f(x)=x,新模型和原模型将同样有效
同时,由于新模型可能得出更优的解来拟合训练数据集,因此添加层似乎更容易降低训练误差。

针对这一问题,何恺明等人提出了残差网络(ResNet),其核心思想是:每个附加层都应该更容易地包含原始函数作为其元素之一

于是,残差块(residual blocks)便诞生了,
也就是说:在残差块中,每个层的输入不仅包含前一层的输出,还包含了原始输入。这样做的目的是让网络学习到残差(即当前层的输出与原始输入之间的差异)。

二、残差块

右图是ResNet的基础架构–残差块。 在残差块中,输入可通过跨层数据线路更快地向前传播。

在这里插入图片描述

ResNet沿用了VGG完整的 3 × 3 3\times 3 3×3卷积层设计。 残差块里首先有2个有相同输出通道数的 3 × 3 3\times 3 3×3卷积层。
每个卷积层后接一个批量规范化层和ReLU激活函数。 然后我们通过跨层数据通路,跳过这2个卷积运算,将输入直接加在最后的ReLU激活函数前。

这样的设计要求2个卷积层的输出与输入形状一样,从而使它们可以相加。
如果想改变通道数,就需要引入一个额外的 1 × 1 1\times 1 1×1卷积层来将输入变换成需要的形状后再做相加运算。

残差块的实现如下:

import torch
from torch import nn
from d2l import torch as d2l
from torch.nn import functional as F

#残差块的实现
class Residual(nn.Module):
    def __init__(self,input_channels,num_channels,use_1x1conv= False,strides =1):

        super().__init__()
        self.conv1 = nn.Conv2d(input_channels,num_channels,kernel_size=3,padding=1,stride=strides)
        self.conv2 = nn.Conv2d(num_channels, num_channels,
                               kernel_size=3, padding=1)
        if use_1x1conv:
            self.conv3 = nn.Conv2d(input_channels,num_channels,kernel_size=1,stride=strides)
        else:
            self.conv3 = None
        self.bn1 = nn.BatchNorm2d(num_channels)
        self.bn2 = nn.BatchNorm2d(num_channels)
    def forward(self,X):
        Y = F.relu(self.bn1(self.conv1(X)))
        Y = self.bn2(self.conv2(Y))
        if self.conv3:
            X = self.conv3(X)
        Y +=X
        return F.relu(Y)

#查看输入和输出形状一致的情况
blk = Residual(3,3)
X = torch.rand(size=(4,3,6,6))
Y = blk(X)
print(Y.shape)

#也可以在增加输出通道数的同时,减半输出的高度和宽度
blk = Residual(3,6,use_1x1conv=True,strides=2)
print(blk(X).shape)

此代码生成两种类型的网络: 一种是当use_1x1conv=False时,应用ReLU非线性函数之前,将输入添加到输出。 另一种是当use_1x1conv=True时,添加通过1×1卷积调整通道和分辨率。

在这里插入图片描述

三、ResNet模型

ResNet的前两层跟之前介绍的GoogLeNet中的一样: 在输出通道数为64、步幅为2的7×7卷积层后,接步幅为2的3×3的最大汇聚层。 不同之处在于ResNet每个卷积层后增加了批量规范化层。

GoogLeNet在后面接了4个由Inception块组成的模块。 ResNet则使用4个由残差块组成的模块,每个模块使用若干个同样输出通道数的残差块。

在这里插入图片描述

"""
ResNet模型
"""

b1 = nn.Sequential(nn.Conv2d(1,64,kernel_size=7,stride=2,padding=3),
                   nn.BatchNorm2d(64),nn.ReLU(),
                   nn.MaxPool2d(kernel_size=3,stride=2,padding=1)
                   )

#注意,我们对第一个模块做了特别处理。         
def resnet_block(input_channels,num_channels,num_residuals,first_block = False):
    blk = []
    for i in range(num_residuals):
        if i ==0 and not first_block:
            blk.append(Residual(input_channels,num_channels,use_1x1conv=True,strides=2))
        else:
            blk.append(Residual(num_residuals,num_channels))
    return blk

#接着在ResNet加入所有残差块,这里每个模块使用2个残差块
b2 = nn.Sequential(*resnet_block(64,64,2,first_block=True))
b3 = nn.Sequential(*resnet_block(64,128,2))
b4 = nn.Sequential(*resnet_block(128, 256, 2))
b5 = nn.Sequential(*resnet_block(256, 512, 2))

#最后,与GoogLeNet一样,在ResNet中加入全局平均汇聚层,以及全连接层输出。
net = nn.Sequential(b1,b2,b3,b4,b5,nn.AdaptiveAvgPool2d((1,1)),
                    nn.Flatten(),nn.Linear(512,10))


#查看每个模块的输出形状
X = torch.rand(size=(1,1,224,224))
for layer in net:
    X = layer(X)
    print(layer.__class__.__name__,"输出形状为:\t",X.shape)

在这里插入图片描述

四、模型训练

#模型训练
lr,num_epochs,batch_size = 0.05,10,256
train_iter,test_iter = d2l.load_data_fashion_mnist(batch_size,resize=96)
d2l.train_ch6(net,train_iter,test_iter,num_epochs,lr,device=d2l.try_gpu())
d2l.plt.show()

在这里插入图片描述

五、小结

  1. 学习嵌套函数是训练神经网络的理想情况。在深层神经网络中,学习另一层作为恒等映射较容易(尽管这是一个极端情况)。
  2. 残差映射可以更容易地学习同一函数,例如将权重层中的参数近似为零。
  3. 利用残差块可以训练出一个有效的深层神经网络:输入可以通过层间的残余连接更快地向前传播。

总结

总之,残差网络通过残差映射学习到了输入与输出之间的差异;通过将输入与输出直接相加,实现了信息的跳跃连接;通过堆叠多个残差块来构建更深的网络,提高了网络的表达能力。

既以为人己愈有,既以与人己愈多。

–2023-10-15 进阶篇

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1104180.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Compose Desktop 使用中的几个问题(分平台加载资源、编写Gradle 任务下载平台资源、桌面特有组件、鼠标键盘事件)

前言 在我之前的文章 Compose For Desktop 实践:使用 Compose-jb 做一个时间水印助手 中,我们使用 Compose For Desktop 写了一个用于读取照片 EXIF 中的拍摄日期参数并以文字水印的方式添加到照片上的桌面程序。 但是事实上,这个程序的名字…

C语言 输入输出

输出 printf 发送格式化输出到标准输出 stdout 调用格式 printf("<格式化字符串>",<参数表>); 头文件 stdio.h 声明 int printf(const char *format,...) 参数 format -- 是字符串&#xff0c;包含要被写入到标准输出 stdout 的文本。可以包含…

Pycharm的安装和使用

目录 环境安装 环境安装 下载并安装 打开网站&#xff1a;https://www.jetbrains.com/pycharm/download/#sectionwindows 下拉到最下面 开始安装 运行pycharm

人工智能时代大模型算法之文心大模型4.0

大家好&#xff0c;我是爱编程的喵喵。双985硕士毕业&#xff0c;现担任全栈工程师一职&#xff0c;热衷于将数据思维应用到工作与生活中。从事机器学习以及相关的前后端开发工作。曾在阿里云、科大讯飞、CCF等比赛获得多次Top名次。现为CSDN博客专家、人工智能领域优质创作者。…

VsCode通过Git History插件查看某个页面的版本修改记录

首先需要安装插件Git History 方式一&#xff1a;通过 点击File History 查看某个文件变更&#xff1b;即通过commit的提交记录去查看某个文件的修改 方式二&#xff1a;通过点击选择toggle File Blame 查看当前页面每一行所有提交修改记录

吃鸡达人必备:分享顶级干货+作图工具推荐+账号安全查询!

吃鸡达人们&#xff0c;你们好&#xff01;今天我来给大家介绍一些炙手可热的吃鸡话题&#xff0c;以及一些让你实力飙升的独家干货&#xff01; 首先&#xff0c;让我们说一下如何提高自己的游戏战斗力。作为一名专业吃鸡行家&#xff0c;我将与你们分享一些顶级游戏作战干货&…

Python数据分析实战-dataframe分组提取每一组的首条记录(附源码和实现效果)

实现功能 使用pandas库来进行DataFrame的分组和提取每一组的第一条记录 实现代码 import pandas as pd# 创建一个示例DataFrame data {Group: [A, A, B, B, C],Value: [1, 2, 3, 4, 5]} df pd.DataFrame(data)# 按照 Group 列进行分组&#xff0c;并提取每一组的第一条记录…

C语言标准库函数使用的参考方式

联网后&#xff0c;登录https://cplusplus.com/ 以string.h库中的函数为例 点击后直接跳转&#xff0c;可点击右下角的按钮&#xff0c;显示实际运行输出

c刷题[6]

目录 杨辉三角 杨氏矩阵 字符串左旋 判断字符串 公务员面试 改数字 数字三角形 输出不重复序列 杨辉三角 在屏幕上打印杨辉三角。 先转化成二维数组&#xff1a; 1 1 1 1 2 1 1 3 3 1 规律&#xff1a;每行第一个元素和最后一个元素为1(主对角),从第三行开始&#xff…

【算法挨揍日记】day15——560. 和为 K 的子数组、974. 和可被 K 整除的子数组

560. 和为 K 的子数组 560. 和为 K 的子数组 题目描述&#xff1a; 给你一个整数数组 nums 和一个整数 k &#xff0c;请你统计并返回 该数组中和为 k 的连续子数组的个数 。 子数组是数组中元素的连续非空序列。 解题思路&#xff1a; 我们可以很容易想到暴力解法&#xf…

C++对象模型(11)-- 虚基类

1、虚基类的引入 我们再来复习一下“多重继承的对象布局”&#xff0c;假设继承的类结构是这样的&#xff1a; 相应的代码&#xff1a; class W { public:int i_w; }; class X : public W {}; class Y : public W {}; class Z : public X, public Y {}; 我们在main()函数中加…

BUUCTF题解之[极客大挑战 2019]EasySQL 1

1.题目分析 考查sql注入的基本使用。 1.sql注入的定义 SQL注入是一种针对Web应用程序的攻击技术&#xff0c;通过在应用程序的用户输入参数中嵌入SQL代码&#xff0c;进而攻击应用程序的数据库。 攻击者可以通过SQL注入来获取敏感信息、执行无权执行的操作、甚至完全控制数据…

tomcat的部署以及优化

tomcat的介绍 Tomcat的简介 Tomcat 是 Java 语言开发的&#xff0c;Tomcat 服务器是一个免费的开放源代码的 Web 应用服务器&#xff0c;是 Apache 软件基金会的 Jakarta 项目中的一个核心项目&#xff0c;由 Apache、Sun 和其他一些公司及个人共同开发而成。 Tomcat 属于轻量…

Go项目踩坑:go get下载超时,goFrame框架下的go项目里将vue项目的dist同步打包发布,go项目打包并压缩

Go项目踩坑&#xff1a;go get下载超时&#xff0c;goFrame框架下的go项目里将vue项目的dist同步打包发布&#xff0c;go项目打包并压缩 go get下载超时goFrame打包静态资源vue项目打包gf pack生成go文件 静态资源使用打包发布go项目交叉编译&#xff0c;省略一些不必要的信息通…

黑客利用人工智能窃取医疗数据的 7 种方式

人工智能被描述为医疗保健行业的一把双刃剑。基于人工智能的系统可以分析大量数据并在早期和可治疗的阶段检测疾病&#xff0c;它们可以比任何人类更快地诊断症状&#xff0c;并且人工智能正在帮助药物开发&#xff0c;使新的救命药物得以识别并将其推向市场速度更快且成本显着…

三集合容斥原理整理

三集合容斥原理的三个公式都是怎么被推导出来的&#xff1f; - 刘明哲的回答 - 知乎 https://www.zhihu.com/question/465008307/answer/2251909478 你必须知道的行测数量知识&#xff08;七&#xff09;容斥问题 - 跟我考公吧的文章 - 知乎 https://zhuanlan.zhihu.com/p/4271…

基于若依框架的药品管理系统

若依框架每张表共有的五个属性&#xff1a; 更改若依后端代码时创建新的module&#xff0c;选择maven&#xff0c;继承ruoyi。 创建实体类时继承BaseEntity&#xff0c;这个类有创建人&#xff0c;创建时间等五个字段&#xff0c;这个类在其它模块中&#xff0c;需要在depende…

在.Core中用EF添加数据库实体类

首先安装dotnet-ef工具&#xff0c;否则提示&#xff1a; *无法执行&#xff0c;因为找不到指定的命令或文件。 可能的原因包括: *你拼错了内置的 dotnet 命令。 *你打算执行 .NET Core 程序&#xff0c;但 dotnet-ef 不存在。 你打算运行全局工具&#xff0c;但在路径上找不到…

SNAP对Sentinel-1预处理

SNAP对Sentinel-1预处理 一、导入数据 二、轨道校正 点击run开始处理 三、噪声去除 打开S-1 Thermal Noise Removal工具 如果选中了VH&#xff0c;就只会输出一个VH极化结果 四、辐射定标 Run 五、滤波处理 六、地形校正 这边的dem需要自己下载 dem下载地址 如果一格…

Orleans的成员管理和故障检测故障检测

Orleans的成员管理和故障检测故障检测 简介 Orleans框架是一个基于.NET平台的开源分布式系统框架&#xff0c;用于开发可扩展&#xff0c;高可用&#xff0c;高性能的云服务应用程序。它采用了Actor模型&#xff0c;将分布式系统中的各个节点抽象成为Actor&#xff0c;使得开…