学习资源站

YOLOv11改进-Conv篇-2024.1月最新成果可变形卷积DCNv4(适用检测,Seg,分类,Pose,OBB)

一、本文介绍

本文给大家带来的改进机制是2024-1月的最新成果 DCNv4 ,其是DCNv3的升级版本,效果可以说是在目前的卷积中名列前茅了, 同时该卷积具有轻量化的效果 !一个DCNv4参数量下降越15Wparameters左右,。它主要通过 两个方面 对前一版本DCNv3进行改进:首先,它 移除了空间聚合中的softmax归一化 ,这样做增强了其动态特性和表达能力;其次,它 优化了内存访问过程 ,以减少冗余操作,从而加快处理速度。DCNv4的表现可以说是非常的全面,同时该网络为新发目前存在大量使用Bug我均已修复。 欢迎大家订阅本专栏,本专栏每周更新3-5篇最新机制,更有包含我所有改进的文件和交流群提供给大家。

欢迎大家订阅我的专栏一起学习YOLO!

二、DCNv4原理

论文地址: 论文官方地址

代码地址: 代码官方地址


2.1 DCNv4的基本原理

DCNv4(Deformable Convolution v4) 是一种改进的 卷积网络 操作符,专为提高 计算机视觉 应用的效率和效果而设计。它主要通过 两个方面 对前一版本DCNv3进行改进:首先,它 移除了空间聚合中的softmax归一化 ,这样做增强了其动态特性和表达能力;其次,它 优化了内存访问过程 ,以减少冗余操作,从而加快处理速度。这些改进使DCNv4在各种视觉任务中,例如图像分类和生成,表现出更快的收敛速度和更高的处理效率。

DCNv4的 主要特点 可以分为以下两点:

1. 动态特性增强: 通过移除空间聚合中的 softmax 归一化,DCNv4增强了其动态特性和表达能力。
2. 内存访问优化: DCNv4优化了内存访问过程,减少了冗余操作,从而提高了处理速度。

下图展示了 DCNv4与其他空间聚合核心操作符的比较:

在处理同一通道内不同位置的查询像素时:

a) DCNv3使用有界(0到1范围内)动态权重聚合空间特征,采样点集(注意力集)对于每个位置是相同的。
b) DCNv3对于每个位置使用一个专用的窗口。
c) 传统卷积提供更灵活的无界值范围以聚合权重,并使用固定形状和大小的窗口。
d) DCNv4结合了动态形状和大小的窗口、自适应感受野以及无界值范围的动态聚合权重,对于每个位置都是输入依赖的。


2.2 动态特性增强

在DCNv4中, 动态特性增强 的实现对于计算机视觉 模型 的适应性和灵活性是一个重要进步。这一特性使得DCNv4能够更加有效地处理图像中的形变和动态变化,尤其是在那些 需要精细调整感受野以识别不同尺寸和形状对象的场景中 。通过 去除之前版本中的softmax归一化 ,DCNv4的每个卷积核都能自适应地调整其聚合权重,对每个像素的响应不再受到固定范围的限制。这种灵活性的增加对于提升模型在处理不同分辨率、不同尺度的物体时的性能至关重要,尤其是在图像生成和高级图像理解任务中表现得更为明显。


2.3 内存访问优化

下图展示了 DCNv4在优化内存访问方面的改进:

与DCNv3相比, DCNv4 使用一个线程来处理同一组中的多个通道,这些通道共享采样偏移和聚合权重。这种方法减少了像内存读取和双线性插值系数计算这样的工作量,并且可以合并多个内存访问指令。结果是,DCNv4减少了内存访问需求,如图所示,与DCNv3相比,它能够减少一半的内存访问输出张量。这种优化提高了处理速度,并减少了计算资源的使用。


三、DCNv4的核心代码

3.1 DCNv4的核心代码

该网络目前涉及到编译处理操作,代码结构无法进行重构组成,只能提供文件的形式给大家,修改后的代码我以上传Qq群内,大家可以入群获取即可,不想入群的小伙伴也可以通过如下的百度云链接进行下载。

百度网盘链接:https://pan.baidu.com/s/13qjv9TzVGsC4ranCkHMCHg
提取码:Snu7


3.2 DynamicHead_DCNv4代码

  1. import torch.nn as nn
  2. import torch
  3. from ultralytics.nn.DCNv4_op.DCNv4.modules.dcnv4 import DCNv4
  4. def autopad(k, p=None, d=1): # kernel, padding, dilation
  5. """Pad to 'same' shape outputs."""
  6. if d > 1:
  7. k = d * (k - 1) + 1 if isinstance(k, int) else [d * (x - 1) + 1 for x in k] # actual kernel-size
  8. if p is None:
  9. p = k // 2 if isinstance(k, int) else [x // 2 for x in k] # auto-pad
  10. return p
  11. class Conv(nn.Module):
  12. """Standard convolution with args(ch_in, ch_out, kernel, stride, padding, groups, dilation, activation)."""
  13. default_act = nn.SiLU() # default activation
  14. def __init__(self, c1, c2, k=1, s=1, p=None, g=1, d=1, act=True):
  15. """Initialize Conv layer with given arguments including activation."""
  16. super().__init__()
  17. self.conv = nn.Conv2d(c1, c2, k, s, autopad(k, p, d), groups=g, dilation=d, bias=False)
  18. self.bn = nn.BatchNorm2d(c2)
  19. self.act = self.default_act if act is True else act if isinstance(act, nn.Module) else nn.Identity()
  20. def forward(self, x):
  21. """Apply convolution, batch normalization and activation to input tensor."""
  22. return self.act(self.bn(self.conv(x)))
  23. def forward_fuse(self, x):
  24. """Perform transposed convolution of 2D data."""
  25. return self.act(self.conv(x))
  26. class Bottleneck(nn.Module):
  27. """Standard bottleneck."""
  28. def __init__(self, c1, c2, shortcut=True, g=1, k=(3, 3), e=0.5):
  29. """Initializes a bottleneck module with given input/output channels, shortcut option, group, kernels, and
  30. expansion.
  31. """
  32. super().__init__()
  33. c_ = int(c2 * e) # hidden channels
  34. self.cv1 = Conv(c1, c_, k[0], 1)
  35. self.cv2 = DCNv4(c2)
  36. self.add = shortcut and c1 == c2
  37. def forward(self, x):
  38. """'forward()' applies the YOLO FPN to input data."""
  39. return x + self.cv2(self.cv1(x)) if self.add else self.cv2(self.cv1(x))
  40. class C2f_DCNv4(nn.Module):
  41. """Faster Implementation of CSP Bottleneck with 2 convolutions."""
  42. def __init__(self, c1, c2, n=1, shortcut=False, g=1, e=0.5):
  43. """Initialize CSP bottleneck layer with two convolutions with arguments ch_in, ch_out, number, shortcut, groups,
  44. expansion.
  45. """
  46. super().__init__()
  47. self.c = int(c2 * e) # hidden channels
  48. self.cv1 = Conv(c1, 2 * self.c, 1, 1)
  49. self.cv2 = Conv((2 + n) * self.c, c2, 1) # optional act=FReLU(c2)
  50. self.m = nn.ModuleList(Bottleneck(self.c, self.c, shortcut, g, k=(3, 3), e=1.0) for _ in range(n))
  51. def forward(self, x):
  52. """Forward pass through C2f layer."""
  53. x = self.cv1(x)
  54. x = x.chunk(2, 1)
  55. y = list(x)
  56. # y = list(self.cv1(x).chunk(2, 1))
  57. y.extend(m(y[-1]) for m in self.m)
  58. return self.cv2(torch.cat(y, 1))
  59. def forward_split(self, x):
  60. """Forward pass using split() instead of chunk()."""
  61. y = list(self.cv1(x).split((self.c, self.c), 1))
  62. y.extend(m(y[-1]) for m in self.m)
  63. return self.cv2(torch.cat(y, 1))


四、手把手教你使用DCNv4

首先需要注意的是,目前该卷积尚未提供Pytorch版本,发布的版本需要编译才可以运行, Windows 系统很难编译可能20个里面能编译成功一个就不错了,所以建议大家使用该结构的时候利用Linux系统来使用,进行租用服务器,不会租用服务器训练的读者,可以入群看群内视频,我已发布如何利用服务器训练的教程视频。

我这里的教程是以Windows示例,群内视频是Linux系统教程自己使用那种自行选择就行。


4.1 修改一

我们将下载好的文件复制粘贴到如下的目录文件下' ultralytics /nn'。粘贴后的文件目录构造如下图所示->


4.2 修改二

第二步我门中到我们复制进去文件里的如下文件进行编译'ultralytics/nn/DCNv4_op/make.sh',

之后弹出窗口进行编译,如果成功最后会有成功的提示的。


4.3 修改三

第三步我们编译成功之后,我们就可以使用了,需要注意的是该模块无法替换主干上的普通的卷积,只能搭配C2f来使用(步长为2的时候会报错)。

我们将章节3.2 中的提到的DynamicHead_DCNv4的代码进行如下的步骤添加即可。

我们找到如下ultralytics/nn/modules文件夹下建立一个目录名字呢就是'Addmodules'文件夹 ( 用群内的文件的话已经有了无需新建) !然后在其内部建立一个新的py文件将核心代码复制粘贴进去即可。


4.4 修改四

第二步我们在该目录下创建一个新的py文件名字为'__init__.py'( 用群内的文件的话已经有了无需新建) ,然后在其内部导入我们的检测头如下图所示。


4.5 修改五

第三步我门中到如下文件'ultralytics/nn/tasks.py'进行导入和注册我们的模块 ( 用群内的文件的话已经有了无需重新导入直接开始第四步即可)


4.6 修改六

按照我的添加在parse_model里添加即可。


4.7 修改七

此步需要仔细修改大家不修改此步会报模型不能够定义在CPU上的错误,还是这个文件'ultralytics/nn/tasks.py''然后我们按照图片进行修改。

  1. try:
  2. m.stride = torch.tensor([s / x.shape[-2] for x in forward(torch.zeros(1, ch, s, s))]) # forward on CPU
  3. except RuntimeError:
  4. try:
  5. self.model.to(torch.device('cuda'))
  6. m.stride = torch.tensor([s / x.shape[-2] for x in forward(
  7. torch.zeros(1, ch, s, s).to(torch.device('cuda')))]) # forward on CUDA
  8. except RuntimeError as error:
  9. raise error

替换完成的样子如下图所示

到此就修改完成了,大家可以复制下面的yaml文件运行。


五、C2f_DCNv4的yaml文件和运行记录

5.1 C2f_DCNv4的yaml文件

下面的添加 C2f_DCNv4 是我实验结果的版本。

C2f该还不能够替换第一个C2f也会报错,其中的原因我还没来得及去看。

  1. # Ultralytics YOLO 🚀, AGPL-3.0 license
  2. # YOLOv10 object detection model. For Usage examples see https://docs.ultralytics.com/tasks/detect
  3. # Parameters
  4. nc: 80 # number of classes
  5. scales: # model compound scaling constants, i.e. 'model=yolov10n.yaml' will call yolov10.yaml with scale 'n'
  6. # [depth, width, max_channels]
  7. n: [0.33, 0.25, 1024]
  8. backbone:
  9. # [from, repeats, module, args]
  10. - [-1, 1, Conv, [64, 3, 2]] # 0-P1/2
  11. - [-1, 1, Conv, [128, 3, 2]] # 1-P2/4
  12. - [-1, 3, C2f, [128, True]]
  13. - [-1, 1, Conv, [256, 3, 2]] # 3-P3/8
  14. - [-1, 6, C2f_DCNv4, [256, True]]
  15. - [-1, 1, SCDown, [512, 3, 2]] # 5-P4/16
  16. - [-1, 6, C2f_DCNv4, [512, True]]
  17. - [-1, 1, SCDown, [1024, 3, 2]] # 7-P5/32
  18. - [-1, 3, C2f_DCNv4, [1024, True]]
  19. - [-1, 1, SPPF, [1024, 5]] # 9
  20. - [-1, 1, PSA, [1024]] # 10
  21. # YOLOv10.0n head
  22. head:
  23. - [-1, 1, nn.Upsample, [None, 2, "nearest"]]
  24. - [[-1, 6], 1, Concat, [1]] # cat backbone P4
  25. - [-1, 3, C2f_DCNv4, [512]] # 13
  26. - [-1, 1, nn.Upsample, [None, 2, "nearest"]]
  27. - [[-1, 4], 1, Concat, [1]] # cat backbone P3
  28. - [-1, 3, C2f_DCNv4, [256]] # 16 (P3/8-small)
  29. - [-1, 1, Conv, [256, 3, 2]]
  30. - [[-1, 13], 1, Concat, [1]] # cat head P4
  31. - [-1, 3, C2f_DCNv4, [512]] # 19 (P4/16-medium)
  32. - [-1, 1, SCDown, [512, 3, 2]]
  33. - [[-1, 10], 1, Concat, [1]] # cat head P5
  34. - [-1, 3, C2fCIB, [1024, True, True]] # 22 (P5/32-large)
  35. - [[16, 19, 22], 1, v10Detect, [nc]] # Detect(P3, P4, P5)


5.2 C2f_DCNv4的训练过程截图


五、本文总结

到此本文的正式分享内容就结束了,在这里给大家推荐我的YOLOv11改进有效涨点专栏,本专栏目前为新开的平均质量分98分,后期我会根据各种最新的前沿顶会进行论文复现,也会对一些老的改进机制进行补充,如果大家觉得本文帮助到你了,订阅本专栏,关注后续更多的更新~