飞道的博客

YOLOv5改进之十七:CNN+Transformer——融合Bottleneck Transformers

921人阅读  评论(0)

 ​前 言:作为当前先进的深度学习目标检测算法YOLOv5,已经集合了大量的trick,但是还是有提高和改进的空间,针对具体应用场景下的检测难点,可以不同的改进方法。此后的系列文章,将重点对YOLOv5的如何改进进行详细的介绍,目的是为了给那些搞科研的同学需要创新点或者搞工程项目的朋友需要达到更好的效果提供自己的微薄帮助和参考。

需要更多程序资料以及答疑欢迎大家关注——微信公众号:人工智能AI算法工程师 

解决问题:YOLOv5主干特征提取网络为CNN网络,CNN具有平移不变性和局部性,缺乏全局建模长距离建模的能力,引入自然语言处理领域的框架Transformer来形成CNN+Transformer架构,充分两者的优点,提高目标检测效果,本人经过实验,对小目标以及密集预测任务会有一定的提升效果。

原理:

作者单位:UC Berkeley, 谷歌
论文:https://arxiv.org/abs/2101.1160https://link.zhihu.com/?target=https%3A//arxiv.org/abs/2101.11605

GitHub:https://github.com/leaderj1001/BottleneckTransformers

     BoTNet是一种简单却功能强大的backbone,该架构将自注意力纳入了多种计算机视觉任务,包括图像分类,目标检测和实例分割。通过仅在ResNet的最后三个bottleneck blocks中用全局自注意力替换空间卷积,并且不进行其他任何更改,在目标检测方面显著改善了基线,同时还减少了参数,从而使延迟最小化。

Transformer中的MHSA和BoTNet中的MHSA的区别:

归一化,Transformer使用 Layer Normalization,而BoTNet使用 Batch Normalization。
非线性激活,Transformer仅仅使用一个非线性激活在FPN block模块中,BoTNet使用了3个非线性激活。
输出投影,Transformer中的MHSA包含一个输出投影,BoTNet则没有。
优化器,Transformer使用Adam优化器训练,BoTNet使用sgd+ momentum 

方 法:

第一步修改common.py,增加CTR3模块。


   
  1. class CTR3(nn.Module):
  2. # CSP Bottleneck with 3 convolutions
  3. def __init__( self, c1, c2, n=1, e=0.5, e2=1, w=20, h=20): # ch_in, ch_out, number, , expansion,w,h
  4. super(CTR3, self).__init__()
  5. c_ = int(c2 * e) # hidden channels
  6. self.cv1 = Conv(c1, c_, 1, 1)
  7. self.cv2 = Conv(c1, c_, 1, 1)
  8. self.cv3 = Conv( 2 * c_, c2, 1) # act=FReLU(c2)
  9. self.m = nn.Sequential(
  10. *[BottleneckTransformer(c_, c_, stride= 1, heads= 4, mhsa= True, resolution=(w, h), expansion=e2) for _ in
  11. range(n)])
  12. # self.m = nn.Sequential(*[CrossConv(c_, c_, 3, 1, g, 1.0, shortcut) for _ in range(n)])
  13. def forward( self, x):
  14. # print("CTR3-INPUT:",x.shape)
  15. # return self.cv3
  16. return self.cv3(torch.cat((self.m(self.cv1(x)), self.cv2(x)), dim= 1))

第二步:将yolo.py中注册CTR3模块。


   
  1. if m in [Conv,MobileNetV3_InvertedResidual,ShuffleNetV2_InvertedResidual,ghostc3,DepthSepConv, CTR3
  2. ]:

第三步:进行修改yaml文件


   
  1. backbone :
  2. # [from, number, module, args]
  3. [[ - 1 , 1 , Conv , [ 64 , 6 , 2 , 2 ] ] , # 0-P1/2
  4. [ - 1 , 1 , Conv , [ 128 , 3 , 2 ] ] , # 1-P2/4
  5. [ - 1 , 3 , C3 , [ 128 ] ] ,
  6. [ - 1 , 1 , Conv , [ 256 , 3 , 2 ] ] , # 3-P3/8
  7. [ - 1 , 6 , C3 , [ 256 ] ] ,
  8. [ - 1 , 1 , Conv , [ 512 , 3 , 2 ] ] , # 5-P4/16
  9. [ - 1 , 9 , C3 , [ 512 ] ] ,
  10. [ - 1 , 1 , Conv , [ 1024 , 3 , 2 ] ] , # 7-P5/32
  11. [ - 1 , 3 , CTR3 , [ 1024 ] ] ,
  12. [ - 1 , 1 , SPPF , [ 1024 , 5 ] ] , # 9
  13. ]

结 果:本人在多个数据集上做了大量实验,针对不同的数据集效果不同,以及不同位置添加,都会有一定的差异。

预告一下:下一篇内容将继续分享其他Transformer模块的融合。有兴趣的朋友可以关注一下我,有问题可以留言或者私聊我哦

PS:Transformer不仅仅是适用改进YOLOv5,也可以改进其他的YOLO网络以及目标检测网络,比如YOLOv3、v4、v6、v7等。

最后,希望能互粉一下,做个朋友,一起学习交流。


转载:https://blog.csdn.net/m0_70388905/article/details/125691455
查看评论
* 以上用户言论只代表其个人观点,不代表本网站的观点或立场