【图像分割】【深度学习】PFNet官方Pytorch代码-PFNet网络PM定位模块解析

【图像分割】【深度学习】PFNet官方Pytorch代码-PFNet网络PM定位模块解析

文章目录

  • 【图像分割】【深度学习】PFNet官方Pytorch代码-PFNet网络PM定位模块解析
  • 前言
  • PFNet网络简述
  • 主干网络
  • 定位模块 Positioning Module
    • 通道注意力模块 Channel Attention
    • 空间注意力模块 Spatial Attention
  • 总结


前言

在详细解析PFNet代码之前,首要任务是成功运行PFNet代码【win10下参考教程】,后续学习才有意义。本博客讲解PFNet神经网络模块的PM定位模块代码,不涉及其他功能模块代码。

博主将各功能模块的代码在不同的博文中进行了详细的解析,点击【win10下参考教程】,博文的目录链接放在前言部分。


PFNet网络简述

论文给出的PFNet整体架构如图所示:

输入一张RGB三通道彩色图像,先将其送入ResNet-50的backbone提取多尺度特征,然后将四个尺度的特征(layer1~4)分别通过四个卷积层(CBR)进行通道缩减。在最深层特征上使用定位模块PM来粗略定位潜在目标,然后再逐层通过聚焦模块FM细化分割结果,消除FP(假阳)和FN(假阴)的干扰,最终经过上采样后得到预测分割结果。
代码位置:PFNet.py

class PFNet(nn.Module):
    def __init__(self, backbone_path=None):
        super(PFNet, self).__init__()
        # params

        # backbone 主干网络
        resnet50 = resnet.resnet50(backbone_path)
        self.layer0 = nn.Sequential(resnet50.conv1, resnet50.bn1, resnet50.relu)
        self.layer1 = nn.Sequential(resnet50.maxpool, resnet50.layer1)
        self.layer2 = resnet50.layer2
        self.layer3 = resnet50.layer3
        self.layer4 = resnet50.layer4

        # channel reduction
        self.cr4 = nn.Sequential(nn.Conv2d(2048, 512, 3, 1, 1), nn.BatchNorm2d(512), nn.ReLU())
        self.cr3 = nn.Sequential(nn.Conv2d(1024, 256, 3, 1, 1), nn.BatchNorm2d(256), nn.ReLU())
        self.cr2 = nn.Sequential(nn.Conv2d(512, 128, 3, 1, 1), nn.BatchNorm2d(128), nn.ReLU())
        self.cr1 = nn.Sequential(nn.Conv2d(256, 64, 3, 1, 1), nn.BatchNorm2d(64), nn.ReLU())

        # positioning
        self.positioning = Positioning(512)

        # focus
        self.focus3 = Focus(256, 512)
        self.focus2 = Focus(128, 256)
        self.focus1 = Focus(64, 128)

        for m in self.modules():
            if isinstance(m, nn.ReLU):
                m.inplace = True

    def forward(self, x):
        # x: [batch_size, channel=3, h, w]
        layer0 = self.layer0(x)  # [-1, 64, h/2, w/2]
        layer1 = self.layer1(layer0)  # [-1, 256, h/4, w/4]
        layer2 = self.layer2(layer1)  # [-1, 512, h/8, w/8]
        layer3 = self.layer3(layer2)  # [-1, 1024, h/16, w/16]
        layer4 = self.layer4(layer3)  # [-1, 2048, h/32, w/32]

        # channel reduction
        cr4 = self.cr4(layer4)
        cr3 = self.cr3(layer3)
        cr2 = self.cr2(layer2)
        cr1 = self.cr1(layer1)

        # positioning
        positioning, predict4 = self.positioning(cr4)

        # focus
        focus3, predict3 = self.focus3(cr3, positioning, predict4)
        focus2, predict2 = self.focus2(cr2, focus3, predict3)
        focus1, predict1 = self.focus1(cr1, focus2, predict2)

        # rescale
        predict4 = F.interpolate(predict4, size=x.size()[2:], mode='bilinear', align_corners=True)
        predict3 = F.interpolate(predict3, size=x.size()[2:], mode='bilinear', align_corners=True)
        predict2 = F.interpolate(predict2, size=x.size()[2:], mode='bilinear', align_corners=True)
        predict1 = F.interpolate(predict1, size=x.size()[2:], mode='bilinear', align_corners=True)

        if self.training:
            return predict4, predict3, predict2, predict1

        return torch.sigmoid(predict4), torch.sigmoid(predict3), torch.sigmoid(predict2), torch.sigmoid(
            predict1)

主干网络

resnet网络结构讲解可以参看博主的博文【ResNet模型算法详解】,这里只讲解论文中使用到的主干网络相关的代码,其他主干网络的代码其实也大同小异,就不再赘述。
主干网络选择的是resnet50:
在这里插入图片描述
代码位置:backbone/resnet/resnet.py

def resnet50(backbone_path, pretrained=True, **kwargs):
    return _resnet('resnet50', Bottleneck, [3, 4, 6, 3], backbone_path, pretrained, **kwargs)
def _resnet(arch, block, layers, backbone_path, pretrained, **kwargs):
    # 初始化网络结构
    model = ResNet(block, layers, **kwargs)
    # 是否加载预训练网络
    if pretrained:
        # 通过提供的预训练权重存放路径加载预训练权重
        state_dict = torch.load(backbone_path)
        model.load_state_dict(state_dict)
        print("From {} Load {} Weights Succeed!".format(backbone_path, arch))
    return model    

1×1卷积和3×3卷积是组成resnet网络最基本的结构单元。

# 3×3卷积构成
def conv3x3(in_planes, out_planes, stride=1, groups=1, dilation=1):
    """3x3 convolution with padding"""
    # pfnet的主干网络不涉及分组卷积和空洞卷积
    return nn.Conv2d(in_planes, out_planes, kernel_size=3, stride=stride,
                     padding=dilation, groups=groups, bias=False, dilation=dilation)
# 1×1卷积
def conv1x1(in_planes, out_planes, stride=1):
    """1x1 convolution"""
    return nn.Conv2d(in_planes, out_planes, kernel_size=1, stride=stride, bias=False)

强调一点,pfnet的主干网络不涉及分组卷积和空洞卷积,只用到了常规卷积,因此groups和dilation默认值都是1,因此读者阅读resnet.py代码时候不必过多纠结。

resnet50的是由基础卷积块Bottleneck(block)搭建而成,只是不同深度的卷积块的channel有所不同:

class Bottleneck(nn.Module):
    expansion = 4
    __constants__ = ['downsample']

    def __init__(self, inplanes, planes, stride=1, downsample=None, groups=1,
                 base_width=64, dilation=1, norm_layer=None):
        super(Bottleneck, self).__init__()
        if norm_layer is None:
            norm_layer = nn.BatchNorm2d
        width = int(planes * (base_width / 64.)) * groups
        # 1×1卷积
        self.conv1 = conv1x1(inplanes, width)
        self.bn1 = norm_layer(width)
        # 3×3卷积
        self.conv2 = conv3x3(width, width, stride, groups, dilation)
        self.bn2 = norm_layer(width)
        # 1×1卷积
        self.conv3 = conv1x1(width, planes * self.expansion)
        self.bn3 = norm_layer(planes * self.expansion)

        self.relu = nn.ReLU(inplace=True)
        # shortcut连接
        self.downsample = downsample
        self.stride = stride

    def forward(self, x):
        identity = x

        out = self.conv1(x)
        out = self.bn1(out)
        out = self.relu(out)

        out = self.conv2(out)
        out = self.bn2(out)
        out = self.relu(out)

        out = self.conv3(out)
        out = self.bn3(out)

        if self.downsample is not None:
            identity = self.downsample(x)

        out += identity
        out = self.relu(out)

        return out

resnet50主干网络搭建

class ResNet(nn.Module):
    def __init__(self, block, layers, num_classes=1000, zero_init_residual=False,
                 groups=1, width_per_group=64, replace_stride_with_dilation=None,
                 norm_layer=None):
        super(ResNet, self).__init__()
        if norm_layer is None:
            norm_layer = nn.BatchNorm2d
        self._norm_layer = norm_layer

        self.inplanes = 64
        self.dilation = 1

        # 分组卷积相关可以忽略
        if replace_stride_with_dilation is None:
            replace_stride_with_dilation = [False, False, False]
        # 分组卷积相关可以忽略

        if len(replace_stride_with_dilation) != 3:
            raise ValueError("replace_stride_with_dilation should be None "
                             "or a 3-element tuple, got {}".format(replace_stride_with_dilation))
        self.groups = groups
        self.base_width = width_per_group

        self.conv1 = nn.Conv2d(3, self.inplanes, kernel_size=7, stride=2, padding=3,
                               bias=False)
        self.bn1 = norm_layer(self.inplanes)
        self.relu = nn.ReLU(inplace=True)

        self.maxpool = nn.MaxPool2d(kernel_size=3, stride=2, padding=1)
        self.layer1 = self._make_layer(block, 64, layers[0])

        self.layer2 = self._make_layer(block, 128, layers[1], stride=2,
                                       dilate=replace_stride_with_dilation[0])

        self.layer3 = self._make_layer(block, 256, layers[2], stride=2,
                                       dilate=replace_stride_with_dilation[1])

        self.layer4 = self._make_layer(block, 512, layers[3], stride=2,
                                       dilate=replace_stride_with_dilation[2])

        # 图像分类器部分可以忽略
        self.avgpool = nn.AdaptiveAvgPool2d((1, 1))
        self.fc = nn.Linear(512 * block.expansion, num_classes)
        # 图像分类器部分可以忽略

        # 模型初始化
        for m in self.modules():
            if isinstance(m, nn.Conv2d):
                nn.init.kaiming_normal_(m.weight, mode='fan_out', nonlinearity='relu')
            elif isinstance(m, (nn.BatchNorm2d, nn.GroupNorm)):
                nn.init.constant_(m.weight, 1)
                nn.init.constant_(m.bias, 0)

        # 对部分模块进行零初始化
        if zero_init_residual:
            for m in self.modules():
                if isinstance(m, Bottleneck):
                    nn.init.constant_(m.bn3.weight, 0)
                elif isinstance(m, BasicBlock):
                    nn.init.constant_(m.bn2.weight, 0)

    def _make_layer(self, block, planes, blocks, stride=1, dilate=False):
        norm_layer = self._norm_layer
        downsample = None
        previous_dilation = self.dilation

        # 空洞卷积相关可以忽略
        if dilate:
            self.dilation *= stride
        # 空洞卷积相关可以忽略

            stride = 1
        if stride != 1 or self.inplanes != planes * block.expansion:
            downsample = nn.Sequential(
                conv1x1(self.inplanes, planes * block.expansion, stride),
                norm_layer(planes * block.expansion),
            )

        layers = []
        layers.append(block(self.inplanes, planes, stride, downsample, self.groups,
                            self.base_width, previous_dilation, norm_layer))
        self.inplanes = planes * block.expansion
        for _ in range(1, blocks):
            layers.append(block(self.inplanes, planes, groups=self.groups,
                                base_width=self.base_width, dilation=self.dilation,
                                norm_layer=norm_layer))

        return nn.Sequential(*layers)

    def _forward_impl(self, x):
        # See note [TorchScript super()]
        x = self.conv1(x)
        x = self.bn1(x)
        x = self.relu(x)
        x = self.maxpool(x)

        x = self.layer1(x)
        x = self.layer2(x)
        x = self.layer3(x)
        x = self.layer4(x)

        x = self.avgpool(x)
        x = torch.flatten(x, 1)
        x = self.fc(x)

        return x

    def forward(self, x):
        return self._forward_impl(x)

定位模块 Positioning Module

原论文中定位模块(Positioning Module,PM)的结构如下图所示:

在输入深层特征 F F F后,经过通道注意力模块(Channel Attention)空间注意力模块(Spatial Attention),捕捉通道和空间位置上的远距离依赖关系。
代码位置:PFNet.py

class Positioning(nn.Module):
    def __init__(self, channel):
        super(Positioning, self).__init__()
        self.channel = channel
        # Channel Attention
        self.cab = CA_Block(self.channel)
        # Spatial Attention
        self.sab = SA_Block(self.channel)
        # 7×7卷积
        self.map = nn.Conv2d(self.channel, 1, 7, 1, 3)

    def forward(self, x):
        cab = self.cab(x)
        sab = self.sab(cab)
        map = self.map(sab)
        return sab, map

通道注意力模块 Channel Attention

捕捉通道之间的依赖关系, 输入特征图 F ∈ R C × H × W F \in {R^{C \times H \times W}} FRC×H×W C C C代表输入特征图通道数目, H H H表示特征图高度, W W W表示特征图宽度,对 F F F进行reshape获得query Q Q Q、key K K K和value V V V,其中 { Q , K , V } ∈ R C × N , N = H × W \left\{ {{\rm{Q}},{\rm{K}},{\rm{V}}} \right\} \in {R^{C \times N}},N = H \times W {Q,K,V}RC×N,N=H×W为特征图像素数量:

  1. Q Q Q K T {K^T} KT( K K K的转置)使用矩阵乘法而后进行softmax归一化计算出通道注意力图 X ∈ R C × C X \in {R^{C \times C}} XRC×C
    x i j = exp ⁡ ( Q i : ⋅ K : j T ) ∑ j = 1 C exp ⁡ ( Q i : ⋅ K : j T ) {x_{ij}} = \frac{{\exp \left( {{Q_{i:}} \cdot K_{\rm{:j}}^T} \right)}}{{\sum\limits_{j = 1}^C {\exp \left( {{Q_{i:}} \cdot K_{\rm{:j}}^T} \right)} }} xij=j=1Cexp(Qi:K:jT)exp(Qi:K:jT)
    其中, Q i : Q_{i:} Qi:表示矩阵 Q Q Q的第 i i i行, K : j T K_{\rm{:j}}^T K:jT表示矩阵 K K K的第 j j j行, x i j x_{ij} xij表示特征图中第j个通道对第i个通道的影响。
  2. x i j x_{ij} xij V j : V_{j:} Vj:矩阵进行矩阵乘法,并将结果reshape成 R C × H × W {R^{C \times H \times W}} RC×H×W,为了增强容错能力,将结果乘以可学习的尺度参数 γ \gamma γ,得到最终输出 F ′ ∈ R C × H × W {F^\prime} \in{R^{C \times H \times W}} FRC×H×W
    F i : ′ = γ ∑ j = 1 C ( x i j V j : ) + F i : F_{\rm{i:}}^\prime{\rm{ }} = \gamma \sum\limits_{j= 1}^C {\left( {{{\rm{x}}_{ij}}{V_{j:}}} \right)} + {F_{\rm{i:}}}{\rm{ }} Fi:=γj=1C(xijVj:)+Fi:
    其中, γ \gamma γ从初始值1逐渐学习权重。


代码位置:PFNet.py

class CA_Block(nn.Module):
    def __init__(self, in_dim):
        super(CA_Block, self).__init__()
        self.chanel_in = in_dim
        # 可训练参数γ
        self.gamma = nn.Parameter(torch.ones(1))
        self.softmax = nn.Softmax(dim=-1)

    def forward(self, x):
        """
            inputs :
                x : input feature maps (B X C X H X W)
            returns :
                out : channel attentive features
        """
        # [B,C,H,W]
        m_batchsize, C, height, width = x.size()
        # [B,C,H X W]
        proj_query = x.view(m_batchsize, C, -1)
        # [B,H X W,C]
        proj_key = x.view(m_batchsize, C, -1).permute(0, 2, 1)
        # [B,C,C]
        energy = torch.bmm(proj_query, proj_key)
        # [B,C,C]
        attention = self.softmax(energy)
        # [B,C,H X W]
        proj_value = x.view(m_batchsize, C, -1)
        # [B,C,H X W]
        out = torch.bmm(attention, proj_value)
        # [B,C,H,W]
        out = out.view(m_batchsize, C, height, width)
        # [B,C,H,W]
        out = self.gamma * out + x
        return out

可能上述代码理解起来有点困难,博主将以上代码绘制成以下流程示意图:

如图所示,query的第4行和key的第1列矩阵相乘计算出了attention的第4行1列的值,表示特征图中第4个通道与第1个通道的"亲疏关系";那么attention的第4行整体就表示第4个通道对包括其自身的所有通道的亲疏关系。

计算特征图通道 m m m n n n的亲疏关系:首先计算俩个通道中每个空间位置 N i m N_i^m Nim N i n N_i^n Nin的乘积,而后再将所有乘积相加,即矩阵相乘,再进行softmax计算而,因此"亲疏关系"总和为1,亲疏值越大关系越紧密。通道注意力图attention是对称矩阵, i i i j j j列表示通道 j j j对通道 i i i的影响。

attention的第3行和value的第1列矩阵相乘计算出out的第3行第1列的值,是根据输入特征图value所有通道的第1个空间位置加权attention第3个通道相关亲疏值再相加综合计算得到;那么out的第3行是根据输入特征图value的所有通道的每个空间位置分别加权相同通道亲疏值再相加得到。

空间注意力模块 Spatial Attention

捕捉空间位置的依赖关系, 将通道注意力模块的输出特征 F ′ F^\prime F作为输入,使用3个1×1的卷积层对 F ′ F^\prime{\rm{ }} F进行卷积和reshape后得到 Q ′ Q^\prime Q K ′ K^\prime K V ′ V^\prime V,其中 { Q ′ , K ′ } ∈ R N × C 8 , V ′ ∈ R C × N \left\{ {{Q^\prime},{K^\prime}} \right\} \in {R^{N\times \frac{C}{8} }},{V^\prime} \in {R^{C \times N}} {Q,K}RN×8C,VRC×N

  1. Q ′ Q^\prime Q K ′ T K^{\prime T} KT( K ′ K^\prime K的转置)使用矩阵乘法而后进行softmax归一化计算出空间注意力图 X ′ ∈ R N × N X^\prime \in {R^{N \times N}} XRN×N
    x i j = exp ⁡ ( Q i : ′ ⋅ K : j ′ T ) ∑ j = 1 N exp ⁡ ( Q i : ′ ⋅ K : j ′ T ) {x_{ij}} = \frac{{\exp \left( {{Q_{i:}^\prime} \cdot K_{\rm{:j}}^{\prime T}} \right)}}{{\sum\limits_{j = 1}^N {\exp \left( {{Q_{i:}^\prime} \cdot K_{\rm{:j}}^{\prime T}} \right)} }} xij=j=1Nexp(Qi:K:jT)exp(Qi:K:jT)
    其中, Q i : ′ Q_{i:}^\prime Qi:表示矩阵 Q Q Q的第 i i i行, K : j ′ T K_{\rm{:j}}^{\prime T} K:jT表示矩阵 K ′ K^\prime K的第 j j j行, x i j ′ x_{ij}^\prime xij表示特征图中第j个空间位置对第i个空间位置的影响。
  2. x i j ′ x_{ij}^\prime xij V i : ′ V_{i:}^\prime Vi:矩阵进行矩阵乘法,并将结果reshape成 R C × H × W {R^{C \times H \times W}} RC×H×W,为了增强容错能力,将结果乘以可学习的尺度参数 γ ′ \gamma^\prime γ,得到最终输出 F ′ ′ ∈ R C × H × W {F^{\prime \prime}} \in{R^{C \times H \times W}} F′′RC×H×W
    F i : ′ ′ = γ ′ ∑ j = 1 C ( V i : x i j ) + F i : ′ F_{\rm{i:}}^{\prime \prime}{\rm{ }} = \gamma^{\prime} \sum\limits_{j= 1}^C {\left( {{{V_{i:}}{\rm{x}}_{ij}}} \right)} + {F_{\rm{i:}}^{\prime}}{\rm{ }} Fi:′′=γj=1C(Vi:xij)+Fi:
    其中, γ \gamma γ从初始值1逐渐学习权重。


代码位置:PFNet.py

class SA_Block(nn.Module):
    def __init__(self, in_dim):
        super(SA_Block, self).__init__()
        self.chanel_in = in_dim
        # 3个1×1卷积
        self.query_conv = nn.Conv2d(in_channels=in_dim, out_channels=in_dim // 8, kernel_size=1)
        self.key_conv = nn.Conv2d(in_channels=in_dim, out_channels=in_dim // 8, kernel_size=1)
        self.value_conv = nn.Conv2d(in_channels=in_dim, out_channels=in_dim, kernel_size=1)
        # 可训练参数γ
        self.gamma = nn.Parameter(torch.ones(1))
        self.softmax = nn.Softmax(dim=-1)

    def forward(self, x):
        """
            inputs :
                x : input feature maps (B X C X H X W)
            returns :
                out : spatial attentive features
        """
        # [B,C,H,W]
        m_batchsize, C, height, width = x.size()
        # [B,H X W,C]
        proj_query = self.query_conv(x).view(m_batchsize, -1, width * height).permute(0, 2, 1)
        # [B,C,H X W]
        proj_key = self.key_conv(x).view(m_batchsize, -1, width * height)
        # [B,H X W,H X W]
        energy = torch.bmm(proj_query, proj_key)
        # [B,H X W,H X W]
        attention = self.softmax(energy)
        # [B,C,H X W]
        proj_value = self.value_conv(x).view(m_batchsize, -1, width * height)
        # [B,C,H X W]
        out = torch.bmm(proj_value, attention.permute(0, 2, 1))
        # [B,C,H,W]
        out = out.view(m_batchsize, C, height, width)
        # [B,C,H,W]
        out = self.gamma * out + x
        return out

可能上述代码理解起来有点困难,博主将以上代码绘制成以下流程示意图:

同理,如图所示,query的第4行和key的第1列矩阵相乘计算出了attention的第4行1列的值,表示特征图中第4个空间位置与第1个空间位置的"亲疏关系",那么attention的第4行整体就表示第4个空间位置对包括其自身的所有空间位置的亲疏关系。

计算特征图空间位置 m m m n n n的亲疏关系:首先计算俩个空间位置中每个通道 C i m C_i^m Cim C i n C_i^n Cin的乘积,而后再将所有乘积相加,即矩阵相乘,再进行softmax计算而,因此"亲疏关系"总和为1,亲疏值越大关系越紧密。空间位置注意力图attention是对称矩阵, i i i j j j列表示空间位置 j j j对空间位置 i i i的影响。

value的第3行矩阵和attention的第1列矩阵相乘计算出out的第3行第1列的值,是根据输入特征图value第3个通道的所有空间位置加权attention第1个空间位置相关亲疏值再相加综合计算得到,那么out的第3行是根据输入特征图value的第3个通道的所有空间位置分别加权每个空间位置亲疏值再相加得到。


总结

尽可能简单、详细的介绍PFNet网络中的主干网络模块和PM定位模块的结构和代码。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/193789.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

1-1、汇编语言概述

语雀原文链接 文章目录 1、机器语言2、汇编语言(Assembly Language)汇编语言工作过程汇编语言三类指令 3、学习资料电子PDF课件论坛视频教程 1、机器语言 机器语言是机器指令的集合。机器指令展开来讲就是一台机器可以正确执行的命令。电子计算机的机器…

古埃及金字塔的修建

从理论上说,古埃及人完全有能力设计并建造出充满各种奇妙细节的胡夫金字塔,但后世还是不断涌现出质疑之声,原因倒也简单,那就是胡夫金字塔实在太大了。据推算,整座金字塔使用大约230万块巨石,总质量可达约5…

SpringBoot+Redis编写一个抢红包雨的案例。附源码。

案例演示 SpringBootRedis编写一个抢红包雨的案例。附源码 1、案例分析,整体方案介绍 预备上线一个红包雨活动。这个红包雨的思路是活动开始前25分钟,在后台创建活动。然后前端用户进入,到点后将设置的金额拆分成多个小红包,开启倒…

数据分析实战案例:Python 分析员工为何离职(附完整代码)

大家好,今天给大家介绍一个Python数据分析项目实战,不仅包含代码,还提供分析数据集。 员工流失或是员工离开公司的比率是公司关注的一个重要问题。它不仅会导致宝贵人才的流失,还会产生成本并破坏生产力。了解员工辞职的原因对于…

勒索解密后oracle无法启动故障处理----惜分飞

客户linux平台被勒索病毒加密,其中有oracle数据库.客户联系黑客进行解密【勒索解密oracle失败】,但是数据库无法正常启动,dbv检查数据库文件报错 [oraclehisdb ~]$ dbv filesystem01.dbf DBVERIFY: Release 11.2.0.1.0 - Production on 星期一 11月 27 21:49:17 2023 Copyrig…

01-鸿蒙4.0学习之开发环境搭建 HelloWorld

HarmonyOS开发学习 1.环境配置 1.下载地址 开发工具:DevEco Studio 3.1.1 Release 下载地址 安装选择快捷方式 安装nodejs和Ohpm 安装SDK 选择同意Accept 检测8项目是否安装成功 2.创建项目 —— hello word

echarts实际开发中遇到的问题

当tooltip内容过高时,增加滚动条 enterable:true, extraCssText: height:500px;overflow-y:auto;

【JavaScript】alert的使用方法 | 超详细

alert作用效果 alert&#xff08;&#xff09;方法用于显示带有一条指定消息和一个确认的按钮的警告框。 alert使用方法 方法一&#xff1a;直接写在script标签内 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"&…

【MyBatisPlus】通俗易懂 快速入门 详细教程

目录 学习目标 一、MyBatisPlus简介 1. 入门案例 问题导入 1.1 SpringBoot整合MyBatisPlus入门程序 ①&#xff1a;创建新模块&#xff0c;选择Spring初始化&#xff0c;并配置模块相关基础信息 ②&#xff1a;选择当前模块需要使用的技术集&#xff08;仅保留JDBC&…

SSM实践基地管理系统开发mysql数据库web结构java编程计算机网页源码eclipse项目

一、源码特点 SSM 实践基地管理系统是一套完善的信息系统&#xff0c;结合springMVC框架完成本系统&#xff0c;对理解JSP java编程开发语言有帮助系统采用SSM框架&#xff08;MVC模式开发&#xff09;&#xff0c;系统具有完整的源代码和数据库 &#xff0c;系统主要采用B/…

小白备战蓝桥杯:Java基础语法

一、注释 IDEA注释快捷键&#xff1a;Ctrl / 单行注释&#xff1a; //注释信息 多行注释&#xff1a; /* 注释信息 */ 二、字面量 常用数据&#xff1a;整数、小数、字符串&#xff08;双引号&#xff09;、字符&#xff08;单引号&#xff09;、布尔值&#xff08;tr…

理解Android无埋点技术

首先什么是无埋点呢&#xff0c;其实所谓无埋点就是开发者无需再对追踪点进行埋码&#xff0c;而是脱离代码&#xff0c;只需面对应用界面圈圈点点即可追加随时生效的事件数据点。 无埋点的好处 其实无埋点并不是完全不用写代码&#xff0c;而是尽可能的少写代码。开发者将SDK集…

springboot+vue智能企业设备管理系统05k50

智能设备管理系统主要是为了提高工作人员的工作效率和更方便快捷的满足用户&#xff0c;更好存储所有数据信息及快速方便的检索功能&#xff0c;对系统的各个模块是通过许多今天的发达系统做出合理的分析来确定考虑用户的可操作性&#xff0c;遵循开发的系统优化的原则&#xf…

线性分类器--分类模型

记录学习 北京邮电大学计算机学院 鲁鹏 为什么从线性分类器开始&#xff1f;  形式简单、易于理解  通过层级结构&#xff08;神经网络&#xff09;或者高维映射&#xff08;支撑向量机&#xff09;可以 形成功能强大的非线性模型 什么是线性分类器&#xff1f; 线性分…

Linux基础项目开发1:量产工具——程序框架(一)

前言&#xff1a; 前面已经将Linux应用开发基础知识学习完了&#xff0c;现在让我们来做个小项目练练手&#xff0c;对之前的一些知识点进行一个更加具体详细的认识与了解&#xff0c;我们要进行的项目名称为&#xff1a;电子产品量产测试与烧写工具&#xff0c;这是一套软件&a…

Clion+Ubuntu(WSL)+MySQL8.0开发环境搭建

1. 下载 MySQL 源码 访问 MySQL 官方网站&#xff08;MySQL :: Download MySQL Community Server&#xff09;并下载 MySQL 8.0 的源码包&#xff08;mysql-boost-8.0.31.tar.gz&#xff09;。 2. 安装编译依赖 1&#xff09;更换镜像源 参考&#xff1a;Linux Ubuntu 修改…

实在智能荣获36氪消费新势力榜单“优选品牌服务商”

近日&#xff0c;36氪未来消费正式公布「FUTURE 2023」消费新势力名单&#xff0c;实在智能凭借专业的数字化产品力、卓越的服务力和深远的行业影响力&#xff0c;成功入选榜单并获评“优选品牌服务商”。 据悉&#xff0c;此次名单是由36氪未来消费特邀消费领域的行业专家、投…

MySQL--InnoDB引擎

InnoDB引擎 逻辑存储引擎 表空间→段→区→页→行 Tablespace 表空间&#xff08;ibd文件&#xff09;&#xff1a;一个mysql实例可以对应多个表空间&#xff0c;用于存储记录、索引等数据Segment 段&#xff1a;段分为数据段、索引段、回滚段&#xff0c;InnoDB是索引组织表…

UI自动化测试详解

1、about自动化测试 定义&#xff1a;把人为驱动的测试转化为机器执行的一种过程&#xff0c;重点在于持续集成这个概念&#xff1b; 优势&#xff1a;节约人力和时间成本&#xff1b; 测试金字塔&#xff1a; 如上图所示&#xff0c;敏捷大师Mike Cohn提出该概念&#xff0…