改进YOLO:YOLOv5结合swin transformer

文章参考于芒果大神,在自己的数据集上跑了一下,改了一些出现的错误。

一、配置yolov5_swin_transfomrer.yaml

# Parametersnc: 10  # number of classesdepth_multiple: 0.33  # model depth multiplewidth_multiple: 0.50  # layer channel multipleanchors:  - [10,13, 16,30, 33,23]  # P3/8  - [30,61, 62,45, 59,119]  # P4/16  - [116,90, 156,198, 373,326]  # P5/32# YOLOv5 v6.0 backbone by yoloairbackbone:  # [from, number, module, args]  [[-1, 1, Conv, [64, 6, 2, 2]],  # 0-P1/2   [-1, 1, Conv, [128, 3, 2]],  # 1-P2/4   [-1, 3, C3STR, [128]],   [-1, 1, Conv, [256, 3, 2]],  # 3-P3/8   [-1, 6, C3STR, [256]],   [-1, 1, Conv, [512, 3, 2]],  # 5-P4/16   [-1, 9, C3STR, [512]],   [-1, 1, Conv, [1024, 3, 2]],  # 7-P5/32   [-1, 3, C3STR, [1024]],  # 9 <--- ST2CSPB() Transformer module   [-1, 1, SPPF, [512, 512]],  # 9  ]# YOLOv5 v6.0 headhead:  [[-1, 1, Conv, [512, 1, 1]],   [-1, 1, nn.Upsample, [None, 2, 'nearest']],   [[-1, 6], 1, Concat, [1]],  # cat backbone P4   [-1, 3, C3, [512, False]],  # 13   [-1, 1, Conv, [256, 1, 1]],   [-1, 1, nn.Upsample, [None, 2, 'nearest']],   [[-1, 4], 1, Concat, [1]],  # cat backbone P3   [-1, 3, C3, [256, False]],  # 17 (P3/8-small)   [-1, 1, Conv, [256, 3, 2]],   [[-1, 14], 1, Concat, [1]],  # cat head P4   [-1, 3, C3, [512, False]],  # 20 (P4/16-medium)   [-1, 1, Conv, [512, 3, 2]],   [[-1, 10], 1, Concat, [1]],  # cat head P5   [-1, 3, C3, [1024, False]],  # 23 (P5/32-large)   [[17, 20, 23], 1, Detect, [nc, anchors]],  # Detect(P3, P4, P5)  ]

二、配置common.py文件

在common.py中增加以下下代码:

class SwinTransformerBlock(nn.Module):    def __init__(self, c1, c2, num_heads, num_layers, window_size=8):        super().__init__()        self.conv = None        if c1 != c2:            self.conv = Conv(c1, c2)        # remove input_resolution        self.blocks = nn.Sequential(*[SwinTransformerLayer(dim=c2, num_heads=num_heads, window_size=window_size,                                 shift_size=0 if (i % 2 == 0) else window_size // 2) for i in range(num_layers)])    def forward(self, x):        if self.conv is not None:            x = self.conv(x)        x = self.blocks(x)        return xclass WindowAttention(nn.Module):    def __init__(self, dim, window_size, num_heads, qkv_bias=True, qk_scale=None, attn_drop=0., proj_drop=0.):        super().__init__()        self.dim = dim        self.window_size = window_size  # Wh, Ww        self.num_heads = num_heads        head_dim = dim // num_heads        self.scale = qk_scale or head_dim ** -0.5        # define a parameter table of relative position bias        self.relative_position_bias_table = nn.Parameter(            torch.zeros((2 * window_size[0] - 1) * (2 * window_size[1] - 1), num_heads))  # 2*Wh-1 * 2*Ww-1, nH        # get pair-wise relative position index for each token inside the window        coords_h = torch.arange(self.window_size[0])        coords_w = torch.arange(self.window_size[1])        coords = torch.stack(torch.meshgrid([coords_h, coords_w]))  # 2, Wh, Ww        coords_flatten = torch.flatten(coords, 1)  # 2, Wh*Ww        relative_coords = coords_flatten[:, :, None] - coords_flatten[:, None, :]  # 2, Wh*Ww, Wh*Ww        relative_coords = relative_coords.permute(1, 2, 0).contiguous()  # Wh*Ww, Wh*Ww, 2        relative_coords[:, :, 0] += self.window_size[0] - 1  # shift to start from 0        relative_coords[:, :, 1] += self.window_size[1] - 1        relative_coords[:, :, 0] *= 2 * self.window_size[1] - 1        relative_position_index = relative_coords.sum(-1)  # Wh*Ww, Wh*Ww        self.register_buffer("relative_position_index", relative_position_index)        self.qkv = nn.Linear(dim, dim * 3, bias=qkv_bias)        self.attn_drop = nn.Dropout(attn_drop)        self.proj = nn.Linear(dim, dim)        self.proj_drop = nn.Dropout(proj_drop)        nn.init.normal_(self.relative_position_bias_table, std=.02)        self.softmax = nn.Softmax(dim=-1)    def forward(self, x, mask=None):        B_, N, C = x.shape        qkv = self.qkv(x).reshape(B_, N, 3, self.num_heads, C // self.num_heads).permute(2, 0, 3, 1, 4)        q, k, v = qkv[0], qkv[1], qkv[2]  # make torchscript happy (cannot use tensor as tuple)        q = q * self.scale        attn = (q @ k.transpose(-2, -1))        relative_position_bias = self.relative_position_bias_table[self.relative_position_index.view(-1)].view(            self.window_size[0] * self.window_size[1], self.window_size[0] * self.window_size[1], -1)  # Wh*Ww,Wh*Ww,nH        relative_position_bias = relative_position_bias.permute(2, 0, 1).contiguous()  # nH, Wh*Ww, Wh*Ww        attn = attn + relative_position_bias.unsqueeze(0)        if mask is not None:            nW = mask.shape[0]            attn = attn.view(B_ // nW, nW, self.num_heads, N, N) + mask.unsqueeze(1).unsqueeze(0)            attn = attn.view(-1, self.num_heads, N, N)            attn = self.softmax(attn)        else:            attn = self.softmax(attn)        attn = self.attn_drop(attn)        # print(attn.dtype, v.dtype)        try:            x = (attn @ v).transpose(1, 2).reshape(B_, N, C)        except:            #print(attn.dtype, v.dtype)            x = (attn.half() @ v).transpose(1, 2).reshape(B_, N, C)        x = self.proj(x)        x = self.proj_drop(x)        return xclass Mlp(nn.Module):    def __init__(self, in_features, hidden_features=None, out_features=None, act_layer=nn.SiLU, drop=0.):        super().__init__()        out_features = out_features or in_features        hidden_features = hidden_features or in_features        self.fc1 = nn.Linear(in_features, hidden_features)        self.act = act_layer()        self.fc2 = nn.Linear(hidden_features, out_features)        self.drop = nn.Dropout(drop)    def forward(self, x):        x = self.fc1(x)        x = self.act(x)        x = self.drop(x)        x = self.fc2(x)        x = self.drop(x)        return xclass SwinTransformerLayer(nn.Module):    def __init__(self, dim, num_heads, window_size=8, shift_size=0,                 mlp_ratio=4., qkv_bias=True, qk_scale=None, drop=0., attn_drop=0., drop_path=0.,                 act_layer=nn.SiLU, norm_layer=nn.LayerNorm):        super().__init__()        self.dim = dim        self.num_heads = num_heads        self.window_size = window_size        self.shift_size = shift_size        self.mlp_ratio = mlp_ratio        # if min(self.input_resolution) <= self.window_size:        #     # if window size is larger than input resolution, we don't partition windows        #     self.shift_size = 0        #     self.window_size = min(self.input_resolution)        assert 0 <= self.shift_size  0. else nn.Identity()        self.norm2 = norm_layer(dim)        mlp_hidden_dim = int(dim * mlp_ratio)        self.mlp = Mlp(in_features=dim, hidden_features=mlp_hidden_dim, act_layer=act_layer, drop=drop)    def create_mask(self, H, W):        # calculate attention mask for SW-MSA        img_mask = torch.zeros((1, H, W, 1))  # 1 H W 1        h_slices = (slice(0, -self.window_size),                    slice(-self.window_size, -self.shift_size),                    slice(-self.shift_size, None))        w_slices = (slice(0, -self.window_size),                    slice(-self.window_size, -self.shift_size),                    slice(-self.shift_size, None))        cnt = 0        for h in h_slices:            for w in w_slices:                img_mask[:, h, w, :] = cnt                cnt += 1        mask_windows = window_partition(img_mask, self.window_size)  # nW, window_size, window_size, 1        mask_windows = mask_windows.view(-1, self.window_size * self.window_size)        attn_mask = mask_windows.unsqueeze(1) - mask_windows.unsqueeze(2)        attn_mask = attn_mask.masked_fill(attn_mask != 0, float(-100.0)).masked_fill(attn_mask == 0, float(0.0))        return attn_mask    def forward(self, x):        # reshape x[b c h w] to x[b l c]        _, _, H_, W_ = x.shape        Padding = False        if min(H_, W_)  0:            attn_mask = self.create_mask(H, W).to(x.device)        else:            attn_mask = None        shortcut = x        x = self.norm1(x)        x = x.view(B, H, W, C)        # cyclic shift        if self.shift_size > 0:            shifted_x = torch.roll(x, shifts=(-self.shift_size, -self.shift_size), dims=(1, 2))        else:            shifted_x = x        # partition windows        x_windows = window_partition(shifted_x, self.window_size)  # nW*B, window_size, window_size, C        x_windows = x_windows.view(-1, self.window_size * self.window_size, C)  # nW*B, window_size*window_size, C        # W-MSA/SW-MSA        attn_windows = self.attn(x_windows, mask=attn_mask)  # nW*B, window_size*window_size, C        # merge windows        attn_windows = attn_windows.view(-1, self.window_size, self.window_size, C)        shifted_x = window_reverse(attn_windows, self.window_size, H, W)  # B H' W' C        # reverse cyclic shift        if self.shift_size > 0:            x = torch.roll(shifted_x, shifts=(self.shift_size, self.shift_size), dims=(1, 2))        else:            x = shifted_x        x = x.view(B, H * W, C)        # FFN        x = shortcut + self.drop_path(x)        x = x + self.drop_path(self.mlp(self.norm2(x)))        x = x.permute(0, 2, 1).contiguous().view(-1, C, H, W)  # b c h w        if Padding:            x = x[:, :, :H_, :W_]  # reverse padding        return xclass C3STR(C3):    # C3 module with SwinTransformerBlock()    def __init__(self, c1, c2, n=1, shortcut=True, g=1, e=0.5):        super().__init__(c1, c2, n, shortcut, g, e)        c_ = int(c2 * e)        num_heads = c_ // 32        self.m = SwinTransformerBlock(c_, c_, num_heads, n)

三、yolo.py文件配置

在parse_model(d, ch)函数中增加C3STR

图片[1] - 改进YOLO:YOLOv5结合swin transformer - MaxSSL

四、train.py文件配置

在if __name__ == ‘__main__’:中更改cfg

图片[2] - 改进YOLO:YOLOv5结合swin transformer - MaxSSL

五、一些问题

1.NameError: name ‘F’ is not defined

在common.py中增加以下代码:

import torch.nn.functional as F

2.File “D:\Projects\yoloair-main\models\common.py”, line 1519, in __init__
super().__init__(c1, c2, c2, n, shortcut, g, e)
TypeError: __init__() takes from 3 to 7 positional arguments but 8 were given

图片[3] - 改进YOLO:YOLOv5结合swin transformer - MaxSSL

去掉一个c2。

3.NameError: name ‘window_partition’ is not defined

def window_partition(x, window_size):    """    Args:        x: (B, H, W, C)        window_size (int): window size    Returns:        windows: (num_windows*B, window_size, window_size, C)    """    B, H, W, C = x.shape    x = x.view(B, H // window_size, window_size, W // window_size, window_size, C)    windows = x.permute(0, 1, 3, 2, 4, 5).contiguous().view(-1, window_size, window_size, C)    return windows

4.NameError: name ‘window_reverse’ is not defined

ef window_reverse(windows, window_size, H, W):    """    Args:        windows: (num_windows*B, window_size, window_size, C)        window_size (int): Window size        H (int): Height of image        W (int): Width of image    Returns:        x: (B, H, W, C)    """    B = int(windows.shape[0] / (H * W / window_size / window_size))    x = windows.view(B, H // window_size, W // window_size, window_size, window_size, -1)    x = x.permute(0, 1, 3, 2, 4, 5).contiguous().view(B, H, W, -1)    return x
© 版权声明
THE END
喜欢就支持一下吧
点赞0 分享