(动手学习深度学习)第7章 稠密连接网络---DenseNet
目录
- DenseNet
- DenseNet的优点:
- DenseNet的改进思路
- 总结
- DenseNet代码实现
DenseNet




DenseNet的优点:
- 省参数。在 ImageNet 分类数据集上达到同样的准确率,DenseNet 所需的参数量不到 ResNet 的一半。对于工业界而言,小模型可以显著地节省带宽,降低存储开销。
- 省计算。达到与 ResNet 相当的精度,DenseNet 所需的计算量也只有 ResNet 的一半左右。计算效率在深度学习实际应用中的需求非常强烈,从本次 CVPR 会上大家对模型压缩以及 MobileNet 和 ShuffleNet 这些工作的关注就可以看得出来。最近我们也在搭建更高效的 DenseNet,初步结果表明 DenseNet 对于这类应用具有非常大的潜力,即使不用 Depth Separable Convolution 也能达到比现有方法更好的结果,预计在近期我们会公开相应的方法和模型。
- 抗过拟合。DenseNet 具有非常好的抗过拟合性能,尤其适合于训练数据相对匮乏的应用。对于 DenseNet 抗过拟合的原因有一个比较直观的解释:神经网络每一层提取的特征都相当于对输入数据的一个非线性变换,而随着深度的增加,变换的复杂度也逐渐增加(更多非线性函数的复合)。相比于一般神经网络的分类器直接依赖于网络最后一层(复杂度最高)的特征,DenseNet 可以综合利用浅层复杂度低的特征,因而更容易得到一个光滑的具有更好泛化性能的决策函数。实际上,DenseNet 的泛化性能优于其他网络是可以从理论上证明的:去年的一篇几乎与 DenseNet 同期发布在 arXiv 上的论文(AdaNet: Adaptive Structural Learning of Artificial Neural Networks)所证明的结论(见原文中 Theorem 1)表明类似于 DenseNet 的网络结构具有更小的泛化误差界。
- 泛化性能更强。如果没有data augmention,CIFAR-100下,ResNet表现下降很多,DenseNet下降不多,说明DenseNet泛化性能更强。
DenseNet的改进思路
- 每层开始的瓶颈层(1x1 卷积)对于减少参数量和计算量非常有用。
- 像 VGG 和 ResNet 那样每做一次下采样(down-sampling)之后都把层宽度(growth rate) 增加一倍,可以提高 DenseNet 的计算效率(FLOPS efficiency)。
- 与其他网络一样,DenseNet 的深度和宽度应该均衡的变化,当然 DenseNet 每层的宽度要远小于其他模型。
- 每一层设计得较窄会降低 DenseNet 在 GPU 上的运算效率,但可能会提高在 CPU 上的运算效率。
总结
- 在跨层连接上,不同于ResNet中将输入与输出相加,稠密连接网络(DenseNet)在通道维度上连接(concat)输入和输出。
- DenseNet的主要构建模块是稠密块和过渡层。
- 在构建DenseNet时,我们需要通过添加过渡层来控制网络的维数,从而再次减少通道数。
DenseNet代码实现
- 导入相关库
import torch
from torch import nn
from d2l import torch as d2l
- 定义网络模型
- DenseNet Block由稠密块和过渡层构成:
- 稠密块:
- 由多个卷积块(BN+RelU+Conv)构成。
- 过渡层:
- 由卷积块(BN+Conv)和池化层构成。
# 定义卷积块(BN+RelU+Conv): 输入和输出的长和宽不变
def conv_block(input_channels, num_channels):return nn.Sequential(nn.BatchNorm2d(input_channels), nn.ReLU(),nn.Conv2d(input_channels, num_channels, kernel_size=3, padding=1))
# 定义稠密块
class DensBlock(nn.Module):def __init__(self, num_convs, input_channels, num_channels):""":param num_convs: 卷积层个数:param input_channels: 输入通道数:param num_channels: 输出通道数"""super().__init__()layer = []for i in range(num_convs):layer.append(conv_block(num_channels * i + input_channels, num_channels))self.net = nn.Sequential(*layer)def forward(self, X):for blk in self.net:Y = blk(X)# 连接通道维度上每个块的输入和输出X = torch.cat((X, Y), dim=1)return X
查看稠密块
blk = DensBlock(2, 3 ,10)
X = torch.randn(4, 3, 8, 8)
Y = blk(X)
Y.shape # [4, 23, 8, 8] 23 = 3 + 2*10

- DenseNet Block由稠密块和过渡层构成
- 稠密块:
- 由多个卷积块(BN+RelU+Conv)构成。
- 过渡层:
- 由卷积块(BN+Conv)和池化层构成。
# 定义过渡层(卷积块(BN+Conv)和池化层)
def transition_block(input_channels, num_channels):return nn.Sequential(nn.BatchNorm2d(input_channels), nn.ReLU(),nn.Conv2d(input_channels, num_channels, kernel_size=1),nn.AvgPool2d(kernel_size=2, stride=2))
查看过渡层
blk = transition_block(23, 10)
X = torch.randn(4, 23, 8, 8)
Y = blk(X)
Y.shape # [4, 23, 8, 8] ---> [3, 10, 8, 8]

- DenseNet Block由稠密块和过渡层构成
- 稠密块:
- 由多个卷积块(BN+RelU+Conv)构成。
- 过渡层:
- 由卷积块(BN+Conv)和池化层构成。
# 首先使用同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)
)
# 接下来,类似于ResNet使用的4个残差块,DenseNet使用的是4个稠密块。
# 稠密块里的卷积层通道数(即增长率)设为32,所以每个稠密块将增加128个通道。
# 在每个模块之间,ResNet通过步幅为2的残差块减小高和宽,DenseNet则使用过渡层来减半高和宽,并减半通道数
num_channels, growth_rate = 64, 32 # 当前的通道数, 增长率
num_convs_in_dense_blocks = [4, 4, 4, 4] # 4个稠密块内卷积层的个数
blks = []
for i, num_convs in enumerate(num_convs_in_dense_blocks):blks.append(DensBlock(num_convs, num_channels, growth_rate))num_channels +=num_convs * growth_rate # 上一个稠密块的输出通道数if i != len(num_convs_in_dense_blocks) -1 :# 在稠密块之间添加一个过渡层,使通道数减半。blks.append(transition_block(num_channels, num_channels // 2))num_channels = num_channels // 2
net = nn.Sequential(b1,*blks,nn.BatchNorm2d(num_channels), nn.ReLU(),nn.AdaptiveAvgPool2d((1, 1)),nn.Flatten(),nn.Linear(num_channels, 10)
)
- 查看网络模型
X = torch.randn(1, 1, 96, 96)
for layer in net :X = layer(X)print(layer.__class__.__name__, 'output shape:\t', X.shape)

- 加载数据集
batch_size = 256
train_iter, test_iter = d2l.load_data_fashion_mnist(batch_size, resize=96)
- 训练模型
import time
lr, num_epochs = 0.1, 10
start = time.perf_counter()
d2l.train_ch6(net, train_iter, test_iter, num_epochs, lr, d2l.try_gpu())
end = time.perf_counter()
print("运行耗时 %.4f s" % (end-start))

相关文章:
(动手学习深度学习)第7章 稠密连接网络---DenseNet
目录 DenseNetDenseNet的优点:DenseNet的改进思路总结 DenseNet代码实现 DenseNet DenseNet的优点: 省参数。在 ImageNet 分类数据集上达到同样的准确率,DenseNet 所需的参数量不到 ResNet 的一半。对于工业界而言,小模型可以显著…...
UART编程(寄存器)
1. 串口编程步骤 1.1 看原理图确定引脚 有很多串口,使用哪一个?看原理图确定 1.2 配置引脚为UART功能 至少用到发送、接收引脚:txd、rxd 需要把这些引脚配置为UART功能,并使能UART模块 1.3 设置串口参数 有哪些参数…...
事务码增删查改表数据
常用事务码 SE11 SE14 SE16 SE16N SM30 SE11:查看数据库表/修改表中字段数量_类型/查看表中数据/设置表为可维护或不可维护 SE14:查看数据库表的创建日期创建用户名/查看表中字段/删除表中全部数据(只能全部删) SE16:查看数据库表/对可维护数据库表进行数据维护/SE16通过调试…...
vue开发环境搭建部署(mac版)
前言 目前后端工作越来越少了,年底了,为了先过验收。项目负责人、产品、需求制定的方案就是先做假页面,所以前端的活多点。 其实现在不喜欢搞前端,原因很多,但是感觉现在似乎流行的码林绝学又是九九归一的瓶颈期…...
Java【算法 05】通过时间获取8位验证码(每两个小时生成一个)源码分享
通过时间获取验证码 1.需求2.代码实现2.1 依赖2.2 时间参数处理方法2.3 截取验证码方法2.4 验证方法 3.总结 1.需求 要求是很简单的,每个验证码的有效时间是2小时,这个并不是收到验证码开始计时的,而是每个两小时的时间段使用的是相同的验证…...
微服务 Spring Cloud 5,一图说透Spring Cloud微服务架构
目录 一、域名系统DNS二、LVS(Linux Virtual Server),Linux虚拟服务器三、CDN静态资源四、Nginx反向代理服务器1、Nginx的主要作用体现在以下几个方面:2、Nginx静态资源服务和CDN静态资源服务,如何选择? 五、Gateway网…...
conda清华源安装cuda12.1的pytorch
使用pytorch官方提供的conda command奇慢无比,根本装不下来(科学的情况下也这样) 配置一下清华源使用清华源装就好了 清华源:https://mirrors.tuna.tsinghua.edu.cn/anaconda/cloud/pytorch/ 配置方法:conda config --…...
安徽首届道医传承十八绝技发布会在合肥成功举办
近日,在安徽合肥举行了首届道医传承十八绝技发布会,本次会议由安徽渡罗门生物科技有限公司、北京道武易医文化传播有限公司、楼观台道医文化研究院联合举办。现场吸引了来自全国各地民族医学领域的专家学者参与讨论与交流。本次会议旨在促进道医的交流与…...
一款功能强大的web目录扫描器专业版
dirpro 简介 dirpro 是一款由 python 编写的目录扫描器,操作简单,功能强大,高度自动化。 自动根据返回状态码和返回长度,对扫描结果进行二次整理和判断,准确性非常高。 已实现功能 可自定义扫描线程 导入url文件进…...
【Linux网络】网卡配置与修改主机名,做好基础系统配置
目录 一、网络配置命令 1、查看网卡信息ifconfig Linux永久修改ip地址 2、主机名修改 ①hostname 临时修改主机名 ②永久修改主机名 第一种,使用命令修改 第二种:修改配置文件 3、路由信息 再来拓展一下,永久修改路由表信息 4、检查…...
三大基础排序 -选择排序、冒泡排序、插入排序
排序算法 文章目录 冒泡排序算法步骤动图代码优化总结 选择排序算法步骤动图代码总结 插入排序算法步骤动图代码总结 排序算法,就是使一串记录,按照其中的某个或某些关键字的大小,递增或递减的排列起来的操作。一般默认排序是按照由小到大即…...
el-form添加自定义校验规则校验el-input只能输入数字
0 效果 1 代码 {1,5}是用来限制小数点后几位的 addFormRules: {investAmount: [{ validator: checkInvestAmount, trigger: blur }], }, const checkInvestAmount (rule, value, callback) > {if (value ! && value ! null && value ! undefined) {if (/…...
ios 开发问题小集 [持续更新]
文章目录 一、如何给列表上的UITableViewCell添加手势二、获取NSIndexPath的方式2.1 根据row, section 来创建2.2 根据point 的位置来找到 indexPath三、tableView在Grouped样式下,设置表头表尾空白一、如何给列表上的UITableViewCell添加手势 给cell添加手势,大家都会这么做…...
idea Plugins 搜索不到插件
Settings — System Settings — HTTP Proxy,打开HTTP Proxy 页面,设置自动发现代理: 勾选Atuto-detect proxy settings,勾选Automatic proxy configuration URL,输入: https://plugins.jetbrains.com/id…...
三相电机的某些实测特性曲线
三相电机参数: 0.75KW,额定电流是2A,功率因数0.71,效率78.9%。制式S1. 1.负载不变时的线电压与线电流的关系 1.1相关数据与python代码: 这里记录了一系列的实验: 第一组实验:近乎空载…...
Essential C++ 面向对象4.1 ~ 5.4
个人认为,结合网上对《Essential c》的评论,它不适合初学者: (1)过于精炼,很多内容不会细讲 (2)中文版翻译较生硬,逻辑不够连贯清晰 (3)课后作业有…...
数组【数据结构与算法】
什么是线性表?什么是非线性表?什么是数组?数组如何实现根据下标随机访问数组元素?为什么数组从下标0开始,不从下标1开始? 什么是线性表? 数据结构元素只有前后关系。 线性表包括:数…...
Python克隆单个网页
网上所有代码都无法完全克隆单个网页,不是Css,Js下载不下来就是下载下来也不能正常显示,只能自己写了,记得点赞~ 效果如图: 源码与所需的依赖: pip install requests pip install requests beautifulsoup4…...
电脑硬盘数据恢复哪个好?值得考虑的 8 个硬盘恢复软件解决方案
借助硬盘恢复软件,任何人都可以在家中恢复丢失的文件,而无需任何特殊技能。事实上,最困难的一步是选择最佳解决方案,因为可用选项的数量可能有点多。幸运的是,这篇文章可以为您提供帮助。 8 款顶级硬盘数据恢复软件解决…...
第二十三节——路由守卫
一、概念 提供的导航守卫主要用来通过跳转或取消的方式守卫导航。有多种机会植入路由导航过程中:全局的, 单个路由独享的, 或者组件级的。简单理解:导航守卫就是路由跳转过程中的一些钩子函数,再直白点路由跳转是一个大的过程,这…...
从零构建人脸识别系统:OpenCV与dlib实战
1. 项目概述人脸识别系统是计算机视觉领域最具实用价值的技术之一。从手机解锁到机场安检,这项技术已经深入到我们生活的方方面面。但大多数人只把它当作黑箱使用,很少了解背后的实现原理。今天我想分享如何从零开始构建一个基础但完整的人脸识别系统&am…...
告别Embedded API:手把手教你用Neo4j Java Driver 1.7连接社区版(附3.5与4.x版本差异说明)
Neo4j Java驱动开发实战:从3.5到4.x的迁移指南 当Java开发者首次接触Neo4j时,往往会面临一个关键选择:是使用传统的Embedded API还是现代的Driver API?这个决定不仅影响开发效率,更关系到系统的可维护性和扩展性。本文…...
告别Anchor Boxes:用PyTorch从零实现FCOS目标检测(附37.2AP代码详解)
从零构建FCOS目标检测器:PyTorch实战指南与37.2AP调优秘籍 当目标检测领域还在与Anchor Boxes的复杂参数纠缠时,FCOS(Fully Convolutional One-Stage)像一阵清风拂过计算机视觉的战场。这个完全基于像素级预测的架构,…...
从‘掩膜膨胀’到特征重建:深入浅出图解Partial Convolutions如何‘脑补’图像缺失部分
从‘掩膜膨胀’到特征重建:深入浅出图解Partial Convolutions如何‘脑补’图像缺失部分 想象一下,你正在修复一张老照片——照片的角落被撕掉了一块,或者某个区域因为年代久远而模糊不清。传统的方法可能需要你手动绘制缺失的部分,…...
除了改UUID,PowerShell还能这样玩转Hyper-V:从批量管理到自动化配置
PowerShell在Hyper-V自动化管理中的高阶应用:从批量操作到智能运维 当大多数管理员还在使用图形界面逐个点击配置Hyper-V虚拟机时,掌握PowerShell脚本技术的工程师已经实现了批量创建200台虚拟机并完成网络配置的自动化流程。这种效率差距正是现代IT运维…...
有哪些数字人制作软件,支持短视频和实时对话直播的
PioneerX human数字人凭借强大的技术支撑,实现了国内外主流平台的全域覆盖,适配各类场景的传播与运营需求。依托前沿AI技术,PioneerX human为企业量身打造虚拟数字人定制、AI短视频智能生产、全天候数字人直播、IP孵化培育及IP交易流通等全链…...
苹果芯片未来五年的发展路线图
苹果芯片(Apple Silicon)正迎来一段重要的发展历程。据报道,台积电(TSMC)正致力于在2029年之前研发出1纳米以下的芯片,而苹果作为台积电最重要的客户,很可能成为全球首家在自家设备中搭载1.4纳米…...
别再模拟IIC了!用STM32F103C8T6的硬件IIC驱动AT24C64,CubeMX配置+避坑指南
从模拟IIC到硬件IIC:STM32F103C8T6驱动AT24C64的实战进阶指南 在嵌入式开发中,IIC总线因其简洁的两线制设计(SCL时钟线和SDA数据线)而广受欢迎。然而,许多开发者习惯使用GPIO模拟IIC时序,这种方式虽然灵活&…...
CUDA 11.7 自定义安装避坑指南:如何把临时文件和核心组件都请出C盘
CUDA 11.7 自定义安装避坑指南:如何把临时文件和核心组件都请出C盘 每次安装CUDA工具包时,C盘空间总会神秘消失几个GB——这几乎是所有深度学习开发者的共同困扰。尤其当你的C盘是256GB甚至更小的SSD时,这种"空间吞噬"现象足以让人…...
终极指南:深入解析Godot PCK文件解包器的完整工作流程
终极指南:深入解析Godot PCK文件解包器的完整工作流程 【免费下载链接】godot-unpacker godot .pck unpacker 项目地址: https://gitcode.com/gh_mirrors/go/godot-unpacker godot-unpacker是一款专业的Godot游戏资源解包工具,专门用于提取Godot引…...
