《博主简介》
小伙伴们好,我是阿旭。专注于人工智能、AIGC、python、计算机视觉相关分享研究。
✌更多学习资源,可关注公-仲-hao:【阿旭算法与机器学习】,共同学习交流~
👍感谢小伙伴们点赞、关注!
《------往期经典推荐------》
一、AI应用软件开发实战专栏【链接】
项目名称项目名称1.【人脸识别与管理系统开发】2.【车牌识别与自动收费管理系统开发】3.【手势识别系统开发】4.【人脸面部活体检测系统开发】5.【图片风格快速迁移软件开发】6.【人脸表表情识别系统】7.【YOLOv8多目标识别与自动标注软件开发】8.【基于YOLOv8深度学习的行人跌倒检测系统】9.【基于YOLOv8深度学习的PCB板缺陷检测系统】10.【基于YOLOv8深度学习的生活垃圾分类目标检测系统】11.【基于YOLOv8深度学习的安全帽目标检测系统】12.【基于YOLOv8深度学习的120种犬类检测与识别系统】13.【基于YOLOv8深度学习的路面坑洞检测系统】14.【基于YOLOv8深度学习的火焰烟雾检测系统】15.【基于YOLOv8深度学习的钢材表面缺陷检测系统】16.【基于YOLOv8深度学习的舰船目标分类检测系统】17.【基于YOLOv8深度学习的西红柿成熟度检测系统】18.【基于YOLOv8深度学习的血细胞检测与计数系统】19.【基于YOLOv8深度学习的吸烟/抽烟行为检测系统】20.【基于YOLOv8深度学习的水稻害虫检测与识别系统】21.【基于YOLOv8深度学习的高精度车辆行人检测与计数系统】22.【基于YOLOv8深度学习的路面标志线检测与识别系统】22.【基于YOLOv8深度学习的智能小麦害虫检测识别系统】23.【基于YOLOv8深度学习的智能玉米害虫检测识别系统】24.【基于YOLOv8深度学习的200种鸟类智能检测与识别系统】25.【基于YOLOv8深度学习的45种交通标志智能检测与识别系统】26.【基于YOLOv8深度学习的人脸面部表情识别系统】
二、机器学习实战专栏【链接】,已更新31期,欢迎关注,持续更新中~~
三、深度学习【Pytorch】专栏【链接】
四、【Stable Diffusion绘画系列】专栏【链接】
《------正文------》
前言
EfficientViT
是一种新的高分辨率视觉模型家族,具有新颖的
多尺度线性注意机制
。本文详细介绍了如何使用
efficientViT
网络替换
YOLOV8的主干网络结构
,并且使用修改后的yolov8进行目标检测训练与推理。本文提供了所有源码免费供小伙伴们学习参考,需要的可以通过文末方式自行下载。
本文使用的ultralytics版本为:ultralytics == 8.0.227。
目录
1. efficientViT简介
论文发表时间:2023.09.27
github地址:https://github.com/mit-han-lab/efficientvit
paper地址:https://arxiv.org/abs/2205.14756
摘要:
高分辨率密集预测技术能够实现许多吸引人的实际应用,比如计算摄影、自动驾驶等。然而,巨大的计算成本使得在硬件设备上部署最先进的高分辨率密集预测模型变得困难。本研究提出了
EfficientViT
,一种新的高分辨率视觉模型家族,具有新颖的
多尺度线性注意机制
。与先前依赖于重型softmax注意力、硬件效率低下的大卷积核卷积或复杂的拓扑结构来获得良好性能的高分辨率密集预测模型不同,我们的多尺度线性注意力通过轻量级而且硬件高效的操作实现了全局感受野和多尺度学习(这对高分辨率密集预测是两个理想的特性)。因此,
EfficientViT在各种硬件平台上实现了显著的性能提升,并且具有显著的加速能力,包括移动CPU、边缘GPU等
。
论文亮点如下:
• 我们引入了一种新的多尺度线性注意力模块,用于高效的高分辨率密集预测。它在保持硬件效率的同时实现了全局感知域和多尺度学习。据我们所知,我们的工作是首次展示线性注意力对于高分辨率密集预测的有效性。
• 我们基于提出的多尺度线性注意力模块设计了一种新型的高分辨率视觉模型——EfficientViT。
• 我们的模型在语义分割、超分辨率、任意分割和ImageNet分类等各种硬件平台(移动CPU、边缘GPU和云GPU)上相对于先前的SOTA模型展现出了显著的加速效果。
1.1 efficientViT网络结构
1.2 性能对比
2.使用efficientViT替换YOLOV8主干网络结构
首先,在yolov8官网下载代码并解压,地址如下:
解压后,如下图所示:
第1步–添加efficientVit.py文件,并导入
在
ultralytics/nn/backbone
目录下,新建backbone网络文件
efficientVit.py
,内容如下:
import torch
import torch.nn as nn
import torch.nn.functional as F
import torch.utils.checkpoint as checkpoint
import itertools
from timm.models.layers import SqueezeExcite
import numpy as np
import itertools
__all__ =['EfficientViT_M0','EfficientViT_M1','EfficientViT_M2','EfficientViT_M3','EfficientViT_M4','EfficientViT_M5']classConv2d_BN(torch.nn.Sequential):def__init__(self, a, b, ks=1, stride=1, pad=0, dilation=1,
groups=1, bn_weight_init=1, resolution=-10000):super().__init__()
self.add_module('c', torch.nn.Conv2d(
a, b, ks, stride, pad, dilation, groups, bias=False))
self.add_module('bn', torch.nn.BatchNorm2d(b))
torch.nn.init.constant_(self.bn.weight, bn_weight_init)
torch.nn.init.constant_(self.bn.bias,0)@torch.no_grad()defswitch_to_deploy(self):
c, bn = self._modules.values()
w = bn.weight /(bn.running_var + bn.eps)**0.5
w = c.weight * w[:,None,None,None]
b = bn.bias - bn.running_mean * bn.weight / \
(bn.running_var + bn.eps)**0.5
m = torch.nn.Conv2d(w.size(1)* self.c.groups, w.size(0), w.shape[2:], stride=self.c.stride, padding=self.c.padding, dilation=self.c.dilation, groups=self.c.groups)
m.weight.data.copy_(w)
m.bias.data.copy_(b)return m
defreplace_batchnorm(net):for child_name, child in net.named_children():ifhasattr(child,'fuse'):setattr(net, child_name, child.fuse())elifisinstance(child, torch.nn.BatchNorm2d):setattr(net, child_name, torch.nn.Identity())else:
replace_batchnorm(child)classPatchMerging(torch.nn.Module):def__init__(self, dim, out_dim, input_resolution):super().__init__()
hid_dim =int(dim *4)
self.conv1 = Conv2d_BN(dim, hid_dim,1,1,0, resolution=input_resolution)
self.act = torch.nn.ReLU()
self.conv2 = Conv2d_BN(hid_dim, hid_dim,3,2,1, groups=hid_dim, resolution=input_resolution)
self.se = SqueezeExcite(hid_dim,.25)
self.conv3 = Conv2d_BN(hid_dim, out_dim,1,1,0, resolution=input_resolution //2)defforward(self, x):
x = self.conv3(self.se(self.act(self.conv2(self.act(self.conv1(x))))))return x
classResidual(torch.nn.Module):def__init__(self, m, drop=0.):super().__init__()
self.m = m
self.drop = drop
defforward(self, x):if self.training and self.drop >0:return x + self.m(x)* torch.rand(x.size(0),1,1,1,
device=x.device).ge_(self.drop).div(1- self.drop).detach()else:return x + self.m(x)classFFN(torch.nn.Module):def__init__(self, ed, h, resolution):super().__init__()
self.pw1 = Conv2d_BN(ed, h, resolution=resolution)
self.act = torch.nn.ReLU()
self.pw2 = Conv2d_BN(h, ed, bn_weight_init=0, resolution=resolution)defforward(self, x):
x = self.pw2(self.act(self.pw1(x)))return x
classCascadedGroupAttention(torch.nn.Module):r""" Cascaded Group Attention.
Args:
dim (int): Number of input channels.
key_dim (int): The dimension for query and key.
num_heads (int): Number of attention heads.
attn_ratio (int): Multiplier for the query dim for value dimension.
resolution (int): Input resolution, correspond to the window size.
kernels (List[int]): The kernel size of the dw conv on query.
"""def__init__(self, dim, key_dim, num_heads=8,
attn_ratio=4,
resolution=14,
kernels=[5,5,5,5],):super().__init__()
self.num_heads = num_heads
self.scale = key_dim **-0.5
self.key_dim = key_dim
self.d =int(attn_ratio * key_dim)
self.attn_ratio = attn_ratio
qkvs =[]
dws =[]for i inrange(num_heads):
qkvs.append(Conv2d_BN(dim //(num_heads), self.key_dim *2+ self.d, resolution=resolution))
dws.append(Conv2d_BN(self.key_dim, self.key_dim, kernels[i],1, kernels[i]//2, groups=self.key_dim, resolution=resolution))
self.qkvs = torch.nn.ModuleList(qkvs)
self.dws = torch.nn.ModuleList(dws)
self.proj = torch.nn.Sequential(torch.nn.ReLU(), Conv2d_BN(
self.d * num_heads, dim, bn_weight_init=0, resolution=resolution))
points =list(itertools.product(range(resolution),range(resolution)))
N =len(points)
attention_offsets ={}
idxs =[]for p1 in points:for p2 in points:
offset =(abs(p1[0]- p2[0]),abs(p1[1]- p2[1]))if offset notin attention_offsets:
attention_offsets[offset]=len(attention_offsets)
idxs.append(attention_offsets[offset])
self.attention_biases = torch.nn.Parameter(
torch.zeros(num_heads,len(attention_offsets)))
self.register_buffer('attention_bias_idxs',
torch.LongTensor(idxs).view(N, N))@torch.no_grad()deftrain(self, mode=True):super().train(mode)if mode andhasattr(self,'ab'):del self.ab
else:
self.ab = self.attention_biases[:, self.attention_bias_idxs]defforward(self, x):# x (B,C,H,W)
B, C, H, W = x.shape
trainingab = self.attention_biases[:, self.attention_bias_idxs]
feats_in = x.chunk(len(self.qkvs), dim=1)
feats_out =[]
feat = feats_in[0]for i, qkv inenumerate(self.qkvs):if i >0:# add the previous output to the input
feat = feat + feats_in[i]
feat = qkv(feat)
q, k, v = feat.view(B,-1, H, W).split([self.key_dim, self.key_dim, self.d], dim=1)# B, C/h, H, W
q = self.dws[i](q)
q, k, v = q.flatten(2), k.flatten(2), v.flatten(2)# B, C/h, N
attn =((q.transpose(-2,-1) @ k)* self.scale
+(trainingab[i]if self.training else self.ab[i]))
attn = attn.softmax(dim=-1)# BNN
feat =(v @ attn.transpose(-2,-1)).view(B, self.d, H, W)# BCHW
feats_out.append(feat)
x = self.proj(torch.cat(feats_out,1))return x
classLocalWindowAttention(torch.nn.Module):r""" Local Window Attention.
Args:
dim (int): Number of input channels.
key_dim (int): The dimension for query and key.
num_heads (int): Number of attention heads.
attn_ratio (int): Multiplier for the query dim for value dimension.
resolution (int): Input resolution.
window_resolution (int): Local window resolution.
kernels (List[int]): The kernel size of the dw conv on query.
"""def__init__(self, dim, key_dim, num_heads=8,
attn_ratio=4,
resolution=14,
window_resolution=7,
kernels=[5,5,5,5],):super().__init__()
self.dim = dim
self.num_heads = num_heads
self.resolution = resolution
assert window_resolution >0,'window_size must be greater than 0'
self.window_resolution = window_resolution
self.attn = CascadedGroupAttention(dim, key_dim, num_heads,
attn_ratio=attn_ratio,
resolution=window_resolution,
kernels=kernels,)defforward(self, x):
B, C, H, W = x.shape
if H <= self.window_resolution and W <= self.window_resolution:
x = self.attn(x)else:
x = x.permute(0,2,3,1)
pad_b =(self.window_resolution - H %
self.window_resolution)% self.window_resolution
pad_r =(self.window_resolution - W %
self.window_resolution)% self.window_resolution
padding = pad_b >0or pad_r >0if padding:
x = torch.nn.functional.pad(x,(0,0,0, pad_r,0, pad_b))
pH, pW = H + pad_b, W + pad_r
nH = pH // self.window_resolution
nW = pW // self.window_resolution
# window partition, BHWC -> B(nHh)(nWw)C -> BnHnWhwC -> (BnHnW)hwC -> (BnHnW)Chw
x = x.view(B, nH, self.window_resolution, nW, self.window_resolution, C).transpose(2,3).reshape(
B * nH * nW, self.window_resolution, self.window_resolution, C
).permute(0,3,1,2)
x = self.attn(x)# window reverse, (BnHnW)Chw -> (BnHnW)hwC -> BnHnWhwC -> B(nHh)(nWw)C -> BHWC
x = x.permute(0,2,3,1).view(B, nH, nW, self.window_resolution, self.window_resolution,
C).transpose(2,3).reshape(B, pH, pW, C)if padding:
x = x[:,:H,:W].contiguous()
x = x.permute(0,3,1,2)return x
classEfficientViTBlock(torch.nn.Module):""" A basic EfficientViT building block.
Args:
type (str): Type for token mixer. Default: 's' for self-attention.
ed (int): Number of input channels.
kd (int): Dimension for query and key in the token mixer.
nh (int): Number of attention heads.
ar (int): Multiplier for the query dim for value dimension.
resolution (int): Input resolution.
window_resolution (int): Local window resolution.
kernels (List[int]): The kernel size of the dw conv on query.
"""def__init__(self,type,
ed, kd, nh=8,
ar=4,
resolution=14,
window_resolution=7,
kernels=[5,5,5,5],):super().__init__()
self.dw0 = Residual(Conv2d_BN(ed, ed,3,1,1, groups=ed, bn_weight_init=0., resolution=resolution))
self.ffn0 = Residual(FFN(ed,int(ed *2), resolution))iftype=='s':
self.mixer = Residual(LocalWindowAttention(ed, kd, nh, attn_ratio=ar, \
resolution=resolution, window_resolution=window_resolution, kernels=kernels))
self.dw1 = Residual(Conv2d_BN(ed, ed,3,1,1, groups=ed, bn_weight_init=0., resolution=resolution))
self.ffn1 = Residual(FFN(ed,int(ed *2), resolution))defforward(self, x):return self.ffn1(self.dw1(self.mixer(self.ffn0(self.dw0(x)))))classEfficientViT(torch.nn.Module):def__init__(self, img_size=400,
patch_size=16,
frozen_stages=0,
in_chans=3,
stages=['s','s','s'],
embed_dim=[64,128,192],
key_dim=[16,16,16],
depth=[1,2,3],
num_heads=[4,4,4],
window_size=[7,7,7],
kernels=[5,5,5,5],
down_ops=[['subsample',2],['subsample',2],['']],
pretrained=None,
distillation=False,):super().__init__()
resolution = img_size
self.patch_embed = torch.nn.Sequential(Conv2d_BN(in_chans, embed_dim[0]//8,3,2,1, resolution=resolution), torch.nn.ReLU(),
Conv2d_BN(embed_dim[0]//8, embed_dim[0]//4,3,2,1, resolution=resolution //2), torch.nn.ReLU(),
Conv2d_BN(embed_dim[0]//4, embed_dim[0]//2,3,2,1, resolution=resolution //4), torch.nn.ReLU(),
Conv2d_BN(embed_dim[0]//2, embed_dim[0],3,1,1, resolution=resolution //8))
resolution = img_size // patch_size
attn_ratio =[embed_dim[i]/(key_dim[i]* num_heads[i])for i inrange(len(embed_dim))]
self.blocks1 =[]
self.blocks2 =[]
self.blocks3 =[]for i,(stg, ed, kd, dpth, nh, ar, wd, do)inenumerate(zip(stages, embed_dim, key_dim, depth, num_heads, attn_ratio, window_size, down_ops)):for d inrange(dpth):eval('self.blocks'+str(i+1)).append(EfficientViTBlock(stg, ed, kd, nh, ar, resolution, wd, kernels))if do[0]=='subsample':#('Subsample' stride)
blk =eval('self.blocks'+str(i+2))
resolution_ =(resolution -1)// do[1]+1
blk.append(torch.nn.Sequential(Residual(Conv2d_BN(embed_dim[i], embed_dim[i],3,1,1, groups=embed_dim[i], resolution=resolution)),
Residual(FFN(embed_dim[i],int(embed_dim[i]*2), resolution)),))
blk.append(PatchMerging(*embed_dim[i:i +2], resolution))
resolution = resolution_
blk.append(torch.nn.Sequential(Residual(Conv2d_BN(embed_dim[i +1], embed_dim[i +1],3,1,1, groups=embed_dim[i +1], resolution=resolution)),
Residual(FFN(embed_dim[i +1],int(embed_dim[i +1]*2), resolution)),))
self.blocks1 = torch.nn.Sequential(*self.blocks1)
self.blocks2 = torch.nn.Sequential(*self.blocks2)
self.blocks3 = torch.nn.Sequential(*self.blocks3)
self.channel =[i.size(1)for i in self.forward(torch.randn(1,3,640,640))]defforward(self, x):
outs =[]
x = self.patch_embed(x)
x = self.blocks1(x)
outs.append(x)
x = self.blocks2(x)
outs.append(x)
x = self.blocks3(x)
outs.append(x)return outs
EfficientViT_m0 ={'img_size':224,'patch_size':16,'embed_dim':[64,128,192],'depth':[1,2,3],'num_heads':[4,4,4],'window_size':[7,7,7],'kernels':[7,5,3,3],}
EfficientViT_m1 ={'img_size':224,'patch_size':16,'embed_dim':[128,144,192],'depth':[1,2,3],'num_heads':[2,3,3],'window_size':[7,7,7],'kernels':[7,5,3,3],}
EfficientViT_m2 ={'img_size':224,'patch_size':16,'embed_dim':[128,192,224],'depth':[1,2,3],'num_heads':[4,3,2],'window_size':[7,7,7],'kernels':[7,5,3,3],}
EfficientViT_m3 ={'img_size':224,'patch_size':16,'embed_dim':[128,240,320],'depth':[1,2,3],'num_heads':[4,3,4],'window_size':[7,7,7],'kernels':[5,5,5,5],}
EfficientViT_m4 ={'img_size':224,'patch_size':16,'embed_dim':[128,256,384],'depth':[1,2,3],'num_heads':[4,4,4],'window_size':[7,7,7],'kernels':[7,5,3,3],}
EfficientViT_m5 ={'img_size':224,'patch_size':16,'embed_dim':[192,288,384],'depth':[1,3,4],'num_heads':[3,3,4],'window_size':[7,7,7],'kernels':[7,5,3,3],}defEfficientViT_M0(pretrained='', frozen_stages=0, distillation=False, fuse=False, pretrained_cfg=None, model_cfg=EfficientViT_m0):
model = EfficientViT(frozen_stages=frozen_stages, distillation=distillation, pretrained=pretrained,**model_cfg)if pretrained:
model.load_state_dict(update_weight(model.state_dict(), torch.load(pretrained)['model']))if fuse:
replace_batchnorm(model)return model
defEfficientViT_M1(pretrained='', frozen_stages=0, distillation=False, fuse=False, pretrained_cfg=None, model_cfg=EfficientViT_m1):
model = EfficientViT(frozen_stages=frozen_stages, distillation=distillation, pretrained=pretrained,**model_cfg)if pretrained:
model.load_state_dict(update_weight(model.state_dict(), torch.load(pretrained)['model']))if fuse:
replace_batchnorm(model)return model
defEfficientViT_M2(pretrained='', frozen_stages=0, distillation=False, fuse=False, pretrained_cfg=None, model_cfg=EfficientViT_m2):
model = EfficientViT(frozen_stages=frozen_stages, distillation=distillation, pretrained=pretrained,**model_cfg)if pretrained:
model.load_state_dict(update_weight(model.state_dict(), torch.load(pretrained)['model']))if fuse:
replace_batchnorm(model)return model
defEfficientViT_M3(pretrained='', frozen_stages=0, distillation=False, fuse=False, pretrained_cfg=None, model_cfg=EfficientViT_m3):
model = EfficientViT(frozen_stages=frozen_stages, distillation=distillation, pretrained=pretrained,**model_cfg)if pretrained:
model.load_state_dict(update_weight(model.state_dict(), torch.load(pretrained)['model']))if fuse:
replace_batchnorm(model)return model
defEfficientViT_M4(pretrained='', frozen_stages=0, distillation=False, fuse=False, pretrained_cfg=None, model_cfg=EfficientViT_m4):
model = EfficientViT(frozen_stages=frozen_stages, distillation=distillation, pretrained=pretrained,**model_cfg)if pretrained:
model.load_state_dict(update_weight(model.state_dict(), torch.load(pretrained)['model']))if fuse:
replace_batchnorm(model)return model
defEfficientViT_M5(pretrained='', frozen_stages=0, distillation=False, fuse=False, pretrained_cfg=None, model_cfg=EfficientViT_m5):
model = EfficientViT(frozen_stages=frozen_stages, distillation=distillation, pretrained=pretrained,**model_cfg)if pretrained:
model.load_state_dict(update_weight(model.state_dict(), torch.load(pretrained)['model']))if fuse:
replace_batchnorm(model)return model
defupdate_weight(model_dict, weight_dict):
idx, temp_dict =0,{}for k, v in weight_dict.items():# k = k[9:]if k in model_dict.keys()and np.shape(model_dict[k])== np.shape(v):
temp_dict[k]= v
idx +=1
model_dict.update(temp_dict)print(f'loading weights... {idx}/{len(model_dict)} items')return model_dict
在
ultralytics/nn/tasks.py
中导入刚才的
efficientVit
模块:
# 主干网络from ultralytics.nn.backbone.efficientViT import*
第2步–修改tasks.py中的相关内容
parse_model函数修改
修改
ultralytics/nn/tasks.py
中的
parse_model
函数,修改后完整代码如下:
defparse_model(d, ch, verbose=True):# model_dict, input_channels(3)"""Parse a YOLO model.yaml dictionary into a PyTorch model."""import ast
# Args
max_channels =float('inf')
nc, act, scales =(d.get(x)for x in('nc','activation','scales'))
depth, width, kpt_shape =(d.get(x,1.0)for x in('depth_multiple','width_multiple','kpt_shape'))if scales:
scale = d.get('scale')ifnot scale:
scale =tuple(scales.keys())[0]
LOGGER.warning(f"WARNING ⚠️ no model scale passed. Assuming scale='{scale}'.")
depth, width, max_channels = scales[scale]if act:
Conv.default_act =eval(act)# redefine default activation, i.e. Conv.default_act = nn.SiLU()if verbose:
LOGGER.info(f"{colorstr('activation:')}{act}")# printif verbose:
LOGGER.info(f"\n{'':>3}{'from':>20}{'n':>3}{'params':>10}{'module':<45}{'arguments':<30}")
ch =[ch]
layers, save, c2 =[],[], ch[-1]# layers, savelist, ch out
is_backbone =Falsefor i,(f, n, m, args)inenumerate(d['backbone']+ d['head']):# from, number, module, argstry:if m =='node_mode':
m = d[m]iflen(args)>0:if args[0]=='head_channel':
args[0]=int(d[args[0]])
t = m
m =getattr(torch.nn, m[3:])if'nn.'in m elseglobals()[m]# get moduleexcept:passfor j, a inenumerate(args):ifisinstance(a,str):with contextlib.suppress(ValueError):try:
args[j]=locals()[a]if a inlocals()else ast.literal_eval(a)except:
args[j]= a
n = n_ =max(round(n * depth),1)if n >1else n # depth gainif m in(Classify, Conv, ConvTranspose, GhostConv, Bottleneck, GhostBottleneck, SPP, SPPF, DWConv, Focus,
BottleneckCSP, C1, C2, C2f, C3, C3TR, C3Ghost, nn.ConvTranspose2d, DWConvTranspose2d, C3x, RepC3):
c1, c2 = ch[f], args[0]if c2 != nc:# if c2 not equal to number of classes (i.e. for Classify() output)
c2 = make_divisible(min(c2, max_channels)* width,8)
args =[c1, c2,*args[1:]]if m in(BottleneckCSP, C1, C2, C2f, C3, C3TR, C3Ghost, C3x, RepC3):
args.insert(2, n)# number of repeats
n =1elif m is AIFI:
args =[ch[f],*args]elif m in(HGStem, HGBlock):
c1, cm, c2 = ch[f], args[0], args[1]
args =[c1, cm, c2,*args[2:]]if m is HGBlock:
args.insert(4, n)# number of repeats
n =1elif m is ResNetLayer:
c2 = args[1]if args[3]else args[1]*4elif m is nn.BatchNorm2d:
args =[ch[f]]elif m is Concat:
c2 =sum(ch[x]for x in f)elif m in(Detect, Segment, Pose):
args.append([ch[x]for x in f])if m is Segment:
args[2]= make_divisible(min(args[2], max_channels)* width,8)elif m is RTDETRDecoder:# special case, channels arg must be passed in index 1
args.insert(1,[ch[x]for x in f])elif m in{MHSA, ShuffleAttention}:
args =[ch[f],*args]elif m in{EfficientViT_M0, EfficientViT_M1, EfficientViT_M2, EfficientViT_M3, EfficientViT_M4, EfficientViT_M5}:
m = m(*args)
c2 = m.channel
else:
c2 = ch[f]ifisinstance(c2,list):
is_backbone =True
m_ = m
m_.backbone =Trueelse:
m_ = nn.Sequential(*(m(*args)for _ inrange(n)))if n >1else m(*args)# module
t =str(m)[8:-2].replace('__main__.','')# module type
m.np =sum(x.numel()for x in m_.parameters())# number params
m_.i, m_.f, m_.type= i +4if is_backbone else i, f, t # attach index, 'from' index, typeif verbose:
LOGGER.info(f'{i:>3}{str(f):>20}{n_:>3}{m.np:10.0f}{t:<45}{str(args):<30}')# print
save.extend(x %(i +4if is_backbone else i)for x in([f]ifisinstance(f,int)else f)if
x !=-1)# append to savelist
layers.append(m_)if i ==0:
ch =[]ifisinstance(c2,list):
ch.extend(c2)for _ inrange(5-len(ch)):
ch.insert(0,0)else:
ch.append(c2)return nn.Sequential(*layers),sorted(save)
parse_model修改的详细内容对比
- 将
efficientVit.py
中的all
参数中的函数名,写入tasks.py
的判断分支中。 新建if判断分支,添加如下内容:
elif m in{efficientvit_b0, efficientvit_b1, efficientvit_b2, efficientvit_b3}:
m = m(*args)
c2 = m.channel
2.修改下图解析部分代码1,如下图:
修改前:
修改后:
代码如下:
is_backbone =Falsefor i,(f, n, m, args)inenumerate(d['backbone']+ d['head']):# from, number, module, argstry:if m =='node_mode':
m = d[m]iflen(args)>0:if args[0]=='head_channel':
args[0]=int(d[args[0]])
t = m
m =getattr(torch.nn, m[3:])if'nn.'in m elseglobals()[m]# get moduleexcept:passfor j, a inenumerate(args):ifisinstance(a,str):with contextlib.suppress(ValueError):try:
args[j]=locals()[a]if a inlocals()else ast.literal_eval(a)except:
args[j]= a
3.修改下面截图中的部分代码2
修改前:
修改后:
代码如下:
ifisinstance(c2,list):
is_backbone =True
m_ = m
m_.backbone =Trueelse:
m_ = nn.Sequential(*(m(*args)for _ inrange(n)))if n >1else m(*args)# module
t =str(m)[8:-2].replace('__main__.','')# module type
m.np =sum(x.numel()for x in m_.parameters())# number params
m_.i, m_.f, m_.type= i +4if is_backbone else i, f, t # attach index, 'from' index, typeif verbose:
LOGGER.info(f'{i:>3}{str(f):>20}{n_:>3}{m.np:10.0f}{t:<45}{str(args):<30}')# print
save.extend(x %(i +4if is_backbone else i)for x in([f]ifisinstance(f,int)else f)if x !=-1)# append to savelist
layers.append(m_)
4.修改下面截图部分代码
修改前:
修改后:
修改代码为:
ifisinstance(c2,list):
ch.extend(c2)for _ inrange(5-len(ch)):
ch.insert(0,0)else:
ch.append(c2)
_predict_once函数修改
替换
ultralytics/nn/tasks.py
中的
BaseModel
类的
_predict_once
函数,代码如下:
def_predict_once(self, x, profile=False, visualize=False):"""
Perform a forward pass through the network.
Args:
x (torch.Tensor): The input tensor to the model.
profile (bool): Print the computation time of each layer if True, defaults to False.
visualize (bool): Save the feature maps of the model if True, defaults to False.
Returns:
(torch.Tensor): The last output of the model.
"""
y, dt =[],[]# outputsfor m in self.model:if m.f !=-1:# if not from previous layer
x = y[m.f]ifisinstance(m.f,int)else[x if j ==-1else y[j]for j in m.f]# from earlier layersif profile:
self._profile_one_layer(m, x, dt)ifhasattr(m,'backbone'):
x = m(x)for _ inrange(5-len(x)):
x.insert(0,None)for i_idx, i inenumerate(x):if i_idx in self.save:
y.append(i)else:
y.append(None)# for i in x:# if i is not None:# print(i.size())
x = x[-1]else:
x = m(x)# run
y.append(x if m.i in self.save elseNone)# save outputif visualize:
feature_visualization(x, m.type, m.i, save_dir=visualize)return x
第3步:创建配置文件–yolov8-efficientViT.yaml
在
ultralytics/cfg/models/v8
目录下,创建新的配置文件
yolov8-efficientViT.yaml
,内容如下:
注:可以使用EfficientViT_M0, EfficientViT_M1, EfficientViT_M2, EfficientViT_M3, EfficientViT_M4, EfficientViT_M5中的任何一个,参数量不同。
# Ultralytics YOLO 🚀, AGPL-3.0 license# YOLOv8 object detection model with P3-P5 outputs. For Usage examples see https://docs.ultralytics.com/tasks/detect# Parameters
nc:80# number of classes
scales:# model compound scaling constants, i.e. 'model=yolov8n.yaml' will call yolov8.yaml with scale 'n'# [depth, width, max_channels]
n:[0.33,0.25,1024]# YOLOv8n summary: 225 layers, 3157200 parameters, 3157184 gradients, 8.9 GFLOPs
s:[0.33,0.50,1024]# YOLOv8s summary: 225 layers, 11166560 parameters, 11166544 gradients, 28.8 GFLOPs
m:[0.67,0.75,768]# YOLOv8m summary: 295 layers, 25902640 parameters, 25902624 gradients, 79.3 GFLOPs
l:[1.00,1.00,512]# YOLOv8l summary: 365 layers, 43691520 parameters, 43691504 gradients, 165.7 GFLOPs
x:[1.00,1.25,512]# YOLOv8x summary: 365 layers, 68229648 parameters, 68229632 gradients, 258.5 GFLOPs# 0-P1/2# 1-P2/4# 2-P3/8# 3-P4/16# 4-P5/32# YOLOv8.0n backbone
backbone:# [from, repeats, module, args]-[-1,1, EfficientViT_M0,[]]# 4-[-1,1, SPPF,[1024,5]]# 5# YOLOv8.0n head
head:-[-1,1, nn.Upsample,[None,2,'nearest']]# 6-[[-1,3],1, Concat,[1]]# 7 cat backbone P4-[-1,3, C2f,[512]]# 8-[-1,1, nn.Upsample,[None,2,'nearest']]# 9-[[-1,2],1, Concat,[1]]# 10 cat backbone P3-[-1,3, C2f,[256]]# 11 (P3/8-small)-[-1,1, Conv,[256,3,2]]# 12-[[-1,8],1, Concat,[1]]# 13 cat head P4-[-1,3, C2f,[512]]# 14 (P4/16-medium)-[-1,1, Conv,[512,3,2]]# 15-[[-1,5],1, Concat,[1]]# 16 cat head P5-[-1,3, C2f,[1024]]# 17 (P5/32-large)-[[11,14,17],1, Detect,[nc]]# Detect(P3, P4, P5)
yolov8.yaml
与
yolov8-efficientViT.yaml
对比
backbone部分:
yolov8.yaml
与
yolov8-efficientViT.yaml
对比:
head部分:
yolov8.yaml
与
yolov8-efficientViT.yaml
对比:【注意层数的变化,所以要修改对应的层数数字部分】
第4步:加载配置文件训练模型
运行训练代码
train.py
文件,内容如下:
#coding:utf-8# 替换主干网络,训练from ultralytics import YOLO
if __name__ =='__main__':
model = YOLO('ultralytics/cfg/models/v8/yolov8-efficientViT.yaml')
model.load('yolov8n.pt')# loading pretrain weights
model.train(data='datasets/TomatoData/data.yaml', epochs=250, batch=4)
第5步:模型推理
模型训练完成后,我们使用训练好的模型对图片进行检测:
#coding:utf-8from ultralytics import YOLO
import cv2
# 所需加载的模型目录# path = 'models/best2.pt'
path ='runs/detect/train9/weights/best.pt'# 需要检测的图片地址
img_path ="TestFiles/Riped tomato_31.jpeg"# 加载预训练模型# conf 0.25 object confidence threshold for detection# iou 0.7 intersection over union (IoU) threshold for NMS
model = YOLO(path, task='detect')# 检测图片
results = model(img_path)
res = results[0].plot()# res = cv2.resize(res,dsize=None,fx=2,fy=2,interpolation=cv2.INTER_LINEAR)
cv2.imshow("YOLOv8 Detection", res)
cv2.waitKey(0)
【源码获取】
为了小伙伴们能够,更好的学习实践,本文已将所有代码、数据集、论文等相关内容打包上传,供小伙伴们学习。获取方式如下:
结束语
关于本篇文章大家有任何建议或意见,欢迎在评论区留言交流!
觉得不错的小伙伴,感谢点赞、关注加收藏哦!
版权归原作者 阿_旭 所有, 如有侵权,请联系我们删除。