一文搞懂卷积网络之四(空间注意力Non-local)

本文介绍CNN注意力机制开篇之作Non-local,其解决传统CNN长距离特征提取不足问题,通过学习特征图点间相关性实现全局联系。文中实现了Embedded Gaussian等三种模块结构,在Cifar10上与ResNet18基线对比实验,发现BottleNeck结构和模块位置对效果影响大,不同版本Non-local性能有差异。

☞☞☞AI 智能聊天, 问答助手, AI 智能搜索, 免费无限量使用 DeepSeek R1 模型☜☜☜

一文搞懂卷积网络之四(空间注意力non-local) - 创想鸟

一、闲聊

上个项目里,我们介绍了灰太狼和他的亲戚们……啊,不,是 ResNet 和它的变体们,包括 ResNet 本尊、ResNetV2、ResNeXt 等。其实,当时还出了一个号称“灰太狼最强亲戚”的 ResNeSt,这个家伙涨点的绝技就是在 ResNet 模型里加入了 Split-Attention 注意力模块(详情可以参考大佬的项目:ResNet最强变体ResNeSt —— 实现篇(Paddle动态图版本))。这个项目我们就来了解CNN的注意力机制,先从CV注意力的开篇之作 Non-local 走起~

二、Non-local 实现空间注意力的原理

在传统的CNN、DNN模型中,卷积层的计算只是将周围的特征加权加和,且一般当前层的计算只依赖前一层的结果,而现在的网络又大多使用1×1、3×3尺寸的小卷积核,对长距离相关特征的提取不足。(… a convolutional operation sums up the weighted input in a local neighborhood, and a recurrent operation at time i is often based only on the current and the latest time steps.)

全连接层虽然连接了相邻层的全部神经元,但只对单个神经元进行了权重学习,并未学习神经元之间的联系。(The non-local operation is also different from a fully-connected (fc) layer. Eq.(1) computes responses based on relationships between different locations, whereas fc uses learned weights. In other words, the relationship between xj and xi is not a function of the input data in fc, unlike in non-local layers.)

Non-local 注意力模块是借鉴了 Non-local 图片滤镜算法(Non-local image processing)、序列化处理的前馈神经网络(Feedforward modeling for sequences)和 自注意力机制(Self-attention)等工作,提出的一种提取特征图全局联系的通用模型结构,着力于学习特征图中的点与点之间的相关程度特征,公式如下:

一文搞懂卷积网络之四(空间注意力Non-local) - 创想鸟        

上式中,一文搞懂卷积网络之四(空间注意力Non-local) - 创想鸟计算特征图x中代表i,j两个点相关关系的标量。(A pairwise func-tion f computes a scalar (representing relationship such as affinity) between i and all j.)一文搞懂卷积网络之四(空间注意力Non-local) - 创想鸟计算的是代表特征图x中j点的值。(The unary function g computes a representation of the input signal at the position j.)最后一文搞懂卷积网络之四(空间注意力Non-local) - 创想鸟计算出的特征图所有点之间的响应值通过一文搞懂卷积网络之四(空间注意力Non-local) - 创想鸟进行标准化。(The response is normalized by a factor C(x).)

如文章中所说,这种 Non-local 机制是一种通用(generic)的注意力实现方法,所以上式中的一文搞懂卷积网络之四(空间注意力Non-local) - 创想鸟可以使用不同的方式实现相关性计算。这就有了通过 Embedded Gaussion、Vanilla Gaussion、Dot product 和 Concatenation 几种方式实现的 Non-local 模块。后面我们会实现其中的前三种结构,并测试其对网络性能的提升作用。

文章地址:https://arxiv.org/abs/1711.07971作者源码地址:https://github.com/facebookresearch/video-nonlocal-net

三、Non-local 结构的实现

文章中对 Non-local 模块的结构总结如下图:

一文搞懂卷积网络之四(空间注意力Non-local) - 创想鸟        

如上图所示,先将输入的特征图降维(降到1维)后逐次嵌入(embed)到 theta、phi 和 g 三个向量中。然后,将向量 theta 和向量 phi 的转置相乘,做 softmax 激活后再与向量 g 相乘。最后,再将这个 Non-local 操作包裹一层,这通过一个1×1的卷积核和一个跨层连接实现,以方便嵌入此注意力模块到现有系统中(We wrap the non-local operation in Eq.(1) into a non-local block that can be incorporated into many existing architectures.)。

在实现的过程中还要注意几个地方:

再将降维后的特征图嵌入三个向量中时,可以进行通道缩减(如上图所示缩减比例为一半),然后在最后经过1×1卷积时再升回来。这样类似 BottleNeck 的结构可以节省大约一半参数。(We set the number of channels represented by Wg, Wθ, and Wφ to be half of the number of channels in x. This follows the bottleneck design of [21] and reduces the computation of a block by about a half.)模块最后的1×1卷积后面加了一个 BN 层,这个 BN 层的放大系数(也就是权重参数)全部初始化为 0,以确保此模块的初始状态为恒等映射,使得其可以被插入到使用预训练权重的模型中去。(We add a BN layer right after the last 1×1×1 layer that represents Wz; we do not add BN to other layers in a non-local block. The scale parameter of this BN layer is initialized as zero, following [17]. This ensures that the initial state of the entire non-local block is an identity mapping, so it can be inserted into any pre-trained networks while maintaining its initial behavior.)文章中实现的是视频分类的 Non-local 模块,使用的是 T×H×W 的3D卷积,我们这里实现图片分类的 Non-local 模块时要去掉卷积的 T(时间)维度,采用2D卷积。

实现 Non-local 模块前先做好依赖项导入、参数设置、数据集处理等准备工作:

In [5]

import paddleimport paddle.nn as nnfrom paddle.io import DataLoaderimport numpy as npimport osimport paddle.vision.transforms as Tfrom paddle.vision.datasets import Cifar10import matplotlib.pyplot as plt%matplotlib inlineimport warningswarnings.filterwarnings("ignore", category=Warning) # 过滤报警信息BATCH_SIZE = 32PIC_SIZE = 96EPOCH_NUM = 30CLASS_DIM = 10PLACE = paddle.CPUPlace() # 在cpu上训练# PLACE = paddle.CUDAPlace(0)  # 在gpu上训练# 数据集处理transform = T.Compose([    T.Resize(PIC_SIZE),    T.Transpose(),    T.Normalize([127.5, 127.5, 127.5], [127.5, 127.5, 127.5]),])train_dataset = Cifar10(mode='train', transform=transform)val_dataset = Cifar10(mode='test', transform=transform)train_loader = DataLoader(train_dataset, places=PLACE, shuffle=True, batch_size=BATCH_SIZE, drop_last=True, num_workers=0, use_shared_memory=False)valid_loader = DataLoader(val_dataset, places=PLACE, shuffle=False, batch_size=BATCH_SIZE, drop_last=True, num_workers=0, use_shared_memory=False)def save_show_pics(pics, file_name='tmp', save_path='./output/pics/', save_root_path='./output/'):    if not os.path.exists(save_root_path):        os.makedirs(save_root_path)    if not os.path.exists(save_path):        os.makedirs(save_path)    shape = pics.shape    pic = pics.transpose((0,2,3,1)).reshape([-1,8,PIC_SIZE,PIC_SIZE,3])    pic = np.concatenate(tuple(pic), axis=1)    pic = np.concatenate(tuple(pic), axis=1)    pic = (pic + 1.) / 2.    plt.imsave(save_path+file_name+'.jpg', pic)    # plt.figure(figsize=(8,8), dpi=80)    plt.imshow(pic)    plt.xticks([])    plt.yticks([])    plt.show()test_loader = DataLoader(train_dataset, places=PLACE, shuffle=True, batch_size=BATCH_SIZE, drop_last=True, num_workers=0, use_shared_memory=False)data, label = next(test_loader())save_show_pics(data.numpy())

       

               

1.Embedded Gaussian 实现 Non-local 模块

如以上所描述的,我们实现的就是最常用的用 Embedded Gaussian 实现的 Non-local 模块:

In [ ]

class EmbeddedGaussion(nn.Layer):    def __init__(self, shape):        super(EmbeddedGaussion, self).__init__()        input_dim = shape[1]        self.theta = nn.Conv2D(input_dim, input_dim // 2, 1)        self.phi = nn.Conv2D(input_dim, input_dim // 2, 1)        self.g = nn.Conv2D(input_dim, input_dim // 2, 1)        self.conv = nn.Conv2D(input_dim // 2, input_dim, 1)        self.bn = nn.BatchNorm2D(input_dim, weight_attr=paddle.ParamAttr(initializer=nn.initializer.Constant(0)))    def forward(self, x):        shape = x.shape        theta = paddle.flatten(self.theta(x), start_axis=2, stop_axis=-1)        phi = paddle.flatten(self.phi(x), start_axis=2, stop_axis=-1)        g = paddle.flatten(self.g(x), start_axis=2, stop_axis=-1)        non_local = paddle.matmul(theta, phi, transpose_y=True)        non_local = nn.functional.softmax(non_local)        non_local = paddle.matmul(non_local, g)        non_local = paddle.reshape(non_local, [shape[0], shape[1] // 2, shape[2], shape[3]])        non_local = self.bn(self.conv(non_local))        return non_local + xnl = EmbeddedGaussion([16, 16, 8, 8])x = paddle.to_tensor(np.random.uniform(-1, 1, [16, 16, 8, 8]).astype('float32'))y = nl(x)print(y.shape)

       

[16, 16, 8, 8]

       

2.Vanilla Gaussian 实现 Non-local 模块

Embedded Gaussian 实现的 Non-local 模块如果去掉特征图嵌入向量 theta 和向量 g 的操作,就是普通的用 Vanilla Gaussian 实现的 Non-local 版本了。当然,没有了前面的1×1卷积,通道缩减也就无从谈起了。

In [ ]

class VanillaGaussion(nn.Layer):    def __init__(self, shape):        super(VanillaGaussion, self).__init__()        input_dim = shape[1]        self.g = nn.Conv2D(input_dim, input_dim, 1)        self.conv = nn.Conv2D(input_dim, input_dim, 1)        self.bn = nn.BatchNorm(input_dim)    def forward(self, x):        shape = x.shape        theta = paddle.flatten(x, start_axis=2, stop_axis=-1)        phi = paddle.flatten(x, start_axis=2, stop_axis=-1)        g = paddle.flatten(self.g(x), start_axis=2, stop_axis=-1)        non_local = paddle.matmul(theta, phi, transpose_y=True)        non_local = nn.functional.softmax(non_local)        non_local = paddle.matmul(non_local, g)        non_local = paddle.reshape(non_local, shape)        non_local = self.bn(self.conv(non_local))        return non_local + xnl = VanillaGaussion([16, 16, 8, 8])x = paddle.to_tensor(np.random.uniform(-1, 1, [16, 16, 8, 8]).astype('float32'))y = nl(x)print(y.shape)

       

[16, 16, 8, 8]

       

3.Dot Production 实现 Non-local 模块

Embedded Gaussian 实现 Non-local 的模块如果去掉 SoftMax 激活操作,通过除以 N(N 为特征图的位置数量)进行缩放代替,就是 Dot Production 实现的 Non-local 版本了。

In [ ]

class DotProduction(nn.Layer):    def __init__(self, shape):        super(DotProduction, self).__init__()        input_dim = shape[1]        self.theta = nn.Conv2D(input_dim, input_dim // 2, 1)        self.phi = nn.Conv2D(input_dim, input_dim // 2, 1)        self.g = nn.Conv2D(input_dim, input_dim // 2, 1)        self.conv = nn.Conv2D(input_dim // 2, input_dim, 1)        self.bn = nn.BatchNorm(input_dim)    def forward(self, x):        shape = x.shape        theta = paddle.flatten(self.theta(x), start_axis=2, stop_axis=-1)        phi = paddle.flatten(self.phi(x), start_axis=2, stop_axis=-1)        g = paddle.flatten(self.g(x), start_axis=2, stop_axis=-1)        non_local = paddle.matmul(theta, phi, transpose_y=True)        non_local = non_local / shape[2]        non_local = paddle.matmul(non_local, g)        non_local = paddle.reshape(non_local, [shape[0], shape[1] // 2, shape[2], shape[3]])        non_local = self.bn(self.conv(non_local))        return non_local + xnl = DotProduction([16, 16, 8, 8])x = paddle.to_tensor(np.random.uniform(-1, 1, [16, 16, 8, 8]).astype('float32'))y = nl(x)print(y.shape)

       

[16, 16, 8, 8]

       

四、Non-local 的运行对比效果

下面我们就来实验下刚才实现的三个版本的 Non-local 模块的效果。原文是在视频分类数据集上做的实验,这里我们用 Paddle 内置的 Cifar10 图片分类数据集做下实验。

1.运行 ResNet18 基线版本

在 ResNet18 模型结构上加上一个残差块作为基线版本,后面的 Non-local 模块就替换这个残差块。这样能确认效果的提升来自 Non-Local 结构,而非增加的参数。

In [ ]

class Residual(nn.Layer):    def __init__(self, num_channels, num_filters, use_1x1conv=False, stride=1):        super(Residual, self).__init__()        self.use_1x1conv = use_1x1conv        model = [            nn.Conv2D(num_channels, num_filters, 3, stride=stride, padding=1),            nn.BatchNorm2D(num_filters),            nn.ReLU(),            nn.Conv2D(num_filters, num_filters, 3, stride=1, padding=1),            nn.BatchNorm2D(num_filters),        ]        self.model = nn.Sequential(*model)        if use_1x1conv:            model_1x1 = [nn.Conv2D(num_channels, num_filters, 1, stride=stride)]            self.model_1x1 = nn.Sequential(*model_1x1)    def forward(self, X):        Y = self.model(X)        if self.use_1x1conv:            X = self.model_1x1(X)        return paddle.nn.functional.relu(X + Y)class ResnetBlock(nn.Layer):    def __init__(self, num_channels, num_filters, num_residuals, first_block=False):        super(ResnetBlock, self).__init__()        model = []        for i in range(num_residuals):            if i == 0:                if not first_block:                    model += [Residual(num_channels, num_filters, use_1x1conv=True, stride=2)]                else:                    model += [Residual(num_channels, num_filters)]            else:                model += [Residual(num_filters, num_filters)]        self.model = nn.Sequential(*model)    def forward(self, X):        return self.model(X)class ResNet(nn.Layer):    def __init__(self, num_classes=10):        super(ResNet, self).__init__()        model = [            nn.Conv2D(3, 64, 7, stride=2, padding=3),            nn.BatchNorm2D(64),            nn.ReLU(),            nn.MaxPool2D(kernel_size=3, stride=2, padding=1)        ]        model += [            ResnetBlock(64, 64, 2, first_block=True),            ResnetBlock(64, 128, 2),            # ResnetBlock(128, 256, 2),            ResnetBlock(128, 256, 2 + 1),            ResnetBlock(256, 512, 2)        ]        model += [            nn.AdaptiveAvgPool2D(output_size=1),            nn.Flatten(start_axis=1, stop_axis=-1),            nn.Linear(512, num_classes),        ]        self.model = nn.Sequential(*model)    def forward(self, X):        Y = self.model(X)        return Y# 模型定义model = paddle.Model(ResNet(num_classes=CLASS_DIM))# 设置训练模型所需的optimizer, loss, metricmodel.prepare(    paddle.optimizer.Adam(learning_rate=1e-4, parameters=model.parameters()),    paddle.nn.CrossEntropyLoss(),    paddle.metric.Accuracy(topk=(1, 5)))# 启动训练、评估model.fit(train_loader, valid_loader, epochs=EPOCH_NUM, log_freq=500,     callbacks=paddle.callbacks.VisualDL(log_dir='./log/BLResNet18+1'))

       

The loss value printed in the log is the current step, and the metric is the average value of previous step.Epoch 1/30

       

2.测试加入 Non-local 模块的版本

分别测试用 Embedded Gaussian、Vanilla Gaussian 和 Dot Production 方法实现的 Non-local 模块的效果。

In [ ]

class Residual(nn.Layer):    def __init__(self, num_channels, num_filters, use_1x1conv=False, stride=1):        super(Residual, self).__init__()        self.use_1x1conv = use_1x1conv        model = [            nn.Conv2D(num_channels, num_filters, 3, stride=stride, padding=1),            nn.BatchNorm2D(num_filters),            nn.ReLU(),            nn.Conv2D(num_filters, num_filters, 3, stride=1, padding=1),            nn.BatchNorm2D(num_filters),        ]        self.model = nn.Sequential(*model)        if use_1x1conv:            model_1x1 = [nn.Conv2D(num_channels, num_filters, 1, stride=stride)]            self.model_1x1 = nn.Sequential(*model_1x1)    def forward(self, X):        Y = self.model(X)        if self.use_1x1conv:            X = self.model_1x1(X)        return paddle.nn.functional.relu(X + Y)class ResnetBlock(nn.Layer):    def __init__(self, num_channels, num_filters, num_residuals, first_block=False):        super(ResnetBlock, self).__init__()        model = []        for i in range(num_residuals):            if i == 0:                if not first_block:                    model += [Residual(num_channels, num_filters, use_1x1conv=True, stride=2)]                else:                    model += [Residual(num_channels, num_filters)]            else:                model += [Residual(num_filters, num_filters)]        self.model = nn.Sequential(*model)    def forward(self, X):        return self.model(X)class ResNetNonLocal(nn.Layer):    def __init__(self, num_classes=10):        super(ResNetNonLocal, self).__init__()        model = [            nn.Conv2D(3, 64, 7, stride=2, padding=3),            nn.BatchNorm2D(64),            nn.ReLU(),            nn.MaxPool2D(kernel_size=3, stride=2, padding=1)        ]        model += [            ResnetBlock(64, 64, 2, first_block=True),            ResnetBlock(64, 128, 2),            ResnetBlock(128, 256, 2),            EmbeddedGaussion([BATCH_SIZE, 256, 14, 14]),            # VanillaGaussion([BATCH_SIZE, 256, 14, 14]),            # DotProduction([BATCH_SIZE, 256, 14, 14]),            # # EmbeddedGaussionNoBottleNeck([BATCH_SIZE, 256, 14, 14]),            ResnetBlock(256, 512, 2),        ]        model += [            nn.AdaptiveAvgPool2D(output_size=1),            nn.Flatten(start_axis=1, stop_axis=-1),            nn.Linear(512, num_classes),        ]        self.model = nn.Sequential(*model)    def forward(self, X):        Y = self.model(X)        return Y# 模型定义model = paddle.Model(ResNetNonLocal(num_classes=CLASS_DIM))# 设置训练模型所需的optimizer, loss, metricmodel.prepare(    paddle.optimizer.Adam(learning_rate=1e-4, parameters=model.parameters()),    paddle.nn.CrossEntropyLoss(),    paddle.metric.Accuracy(topk=(1, 5)))# 启动训练、评估model.fit(train_loader, valid_loader, epochs=EPOCH_NUM, log_freq=500,     callbacks=paddle.callbacks.VisualDL(log_dir='./log/EmbeddedGaussion'))# model.fit(train_loader, valid_loader, epochs=EPOCH_NUM, log_freq=500, #     callbacks=paddle.callbacks.VisualDL(log_dir='./log/VanillaGaussion'))# model.fit(train_loader, valid_loader, epochs=EPOCH_NUM, log_freq=500, #     callbacks=paddle.callbacks.VisualDL(log_dir='./log/DotProduction'))

       

The loss value printed in the log is the current step, and the metric is the average value of previous step.Epoch 1/30

       

上面的代码需要运行三次,每次需要注释掉 ResNetNonLocal 类的 forward() 方法里不同版本的 Non-local 模块,并且在 model.fit 写入VisualDL 的 log 文件时用不同的名称。

接下来,我们对比下运行结果的验证集准确率:

1)测试 Vanilla Gaussian 版本 Non-local 模块

一文搞懂卷积网络之四(空间注意力Non-local) - 创想鸟 上图中,蓝色线为 ResNet18 加一个残差块的基线版本的验证集准确率曲线,紫色线为加入Vanilla Gaussian 版本 Non-local 模块后模型的验证集准确率曲线。改进的模型准曲率提高了0.3%。

2)测试 Dot Production 版本 Non-local 模块

一文搞懂卷积网络之四(空间注意力Non-local) - 创想鸟 蓝色线仍为基线模型准确率,绿线为加入Dot Production 版本 Non-local 模块后模型的准确率。改进的模型准曲率提高了0.6%。

3)测试 Embedded Gaussian 版本 Non-local 模块

最后来测试下“顶配”版本的。 一文搞懂卷积网络之四(空间注意力Non-local) - 创想鸟 仍然蓝色线为基线模型数据…我去,这么好的装备怎么出现了这么差的结果,加了 Embedded Gaussian 版本 Non-local 模块的模型精度甚至低于基线模型的精度,这是肿么回事?!&@%

一顿修改猛如虎之后(换模型结构、换数据集、换数据增强、换超参),似乎找到一点儿线索。

4)测试不缩减通道数的 Embedded Gaussian 版本 Non-local 模块

在下面这个 Embedded Gaussian 版本 Non-local 模块中,我们不在1×1卷积上采用类似 BottleNeck 的结构缩减通道数。

In [4]

class EmbeddedGaussionNoBottleNeck(nn.Layer):    def __init__(self, shape):        super(EmbeddedGaussionNoBottleNeck, self).__init__()        input_dim = shape[1]        self.theta = nn.Conv2D(input_dim, input_dim, 1)        self.phi = nn.Conv2D(input_dim, input_dim, 1)        self.g = nn.Conv2D(input_dim, input_dim, 1)        self.conv = nn.Conv2D(input_dim, input_dim, 1)        self.bn = nn.BatchNorm(input_dim)    def forward(self, x):        shape = x.shape        theta = paddle.flatten(self.theta(x), start_axis=2, stop_axis=-1)        phi = paddle.flatten(self.phi(x), start_axis=2, stop_axis=-1)        g = paddle.flatten(self.g(x), start_axis=2, stop_axis=-1)        non_local = paddle.matmul(theta, phi, transpose_y=True)        non_local = nn.functional.softmax(non_local)        non_local = paddle.matmul(non_local, g)        non_local = paddle.reshape(non_local, shape)        non_local = self.bn(self.conv(non_local))        return non_local + xnl = EmbeddedGaussionNoBottleNeck([16, 16, 8, 8])x = paddle.to_tensor(np.random.uniform(-1, 1, [16, 16, 8, 8]).astype('float32'))y = nl(x)print(y.shape)

       

[16, 16, 8, 8]

       

训练后与基线模型对照下: 一文搞懂卷积网络之四(空间注意力Non-local) - 创想鸟 蓝色为基线模型版本曲线。在增加了 Non-local 模块的宽度后,性能和基线版本差不多,虽然没有 Vanilla Gaussian 和 Dot Production 实现的版本提升的精度多,但已经比原来的降低通道数的 Embedded Gaussian 版本好了不少。

五、总结

各种消融实验完毕,总结学习体会。

无论是在 Non-local 结构还是在 ResNet 的残差块中,BottleNeck 结构只适合在深层网络中用于减少参数量。在原版的 ResNet 的结构里,也只有 50 层以上的配置才会使用 BottleNeck 结构。这也解释了在前面项目实验(ResNet一族)中,后面出的一些 ResNet 魔该版本,基本都是在 50 层以上挑战大规模数据集。这些改动都是建立在 BottelNeck 基础上的,所以才用在我们的浅层网络中会效果欠佳。这可能是因为深层的网络中,被激活的神经元更稀疏,才能通过 BottelNeck 使用更窄的网络来减少参数。而在较浅的网络中,网络的宽度对性能影响还是挺大的。Non-local 模块的使用位置是很重要的,在模型的不同层加入注意力模块的效果大相径庭,在不同的数据集上效果也不同(在224×224尺寸的OxfordFlower102 和 CalTech201 数据集上,找这个合适的位置更难,放弃了。我想也许作者当年在视频数据集上做实验也是有原因的吧。),加注意力模块的位置不好反而会使模型性能下降。有的信息说注意力快适合用在前面的层,但是在这个实验里,还是放在后面用来提取更大粒度特征图的位置相关特征时效果更好。

以上就是一文搞懂卷积网络之四(空间注意力Non-local)的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/63414.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
mysql如何设置多用户多账号
上一篇 2025年11月11日 12:17:36
小红书官网网页版登录 小红书网址入口地址
下一篇 2025年11月11日 12:20:39

相关推荐

  • 开源免费PHP工具 PHP开发效率提升利器

    推荐开源免费PHP开发工具以提升效率:VS Code、Sublime Text轻量高效,PhpStorm专业强大;调试用Xdebug、Kint、Ray;依赖管理选Composer;代码质量工具包括PHPStan、Psalm、PHP_CodeSniffer;数据库管理可用%ignore_a_1%MyA…

    2026年5月10日
    000
  • Matplotlib 地图中多类型图例的创建与优化

    Matplotlib 地图中多类型图例的创建与优化Matplotlib 地图中多类型图例的创建与优化Matplotlib 地图中多类型图例的创建与优化Matplotlib 地图中多类型图例的创建与优化

    本教程旨在解决matplotlib地图可视化中,如何在一个图例中同时展示颜色块(如区域分类)和自定义标记(如特定兴趣点)的问题。文章详细介绍了当传统`patch`对象无法正确显示标记时,如何利用`matplotlib.lines.line2d`创建标记图例句柄,并将其与颜色块图例句柄合并,从而生成一…

    2026年5月10日 用户投稿
    100
  • Golang JSON序列化:控制敏感字段暴露的最佳实践

    本教程探讨golang中如何高效控制结构体字段在json序列化时的可见性。当需要将包含敏感信息的结构体数组转换为json响应时,通过利用`encoding/json`包提供的结构体标签,特别是`json:”-“`,可以轻松实现对特定字段的忽略,从而避免敏感数据泄露,确保api…

    2026年5月10日
    000
  • 比特币新手教程 比特币交易平台有哪些

    比特币是一种去中心化的数字货币,基于区块链技术实现点对点交易,具有匿名性、有限发行和不可篡改等特点;新手可通过交易所购买,P2P交易获得比特币,常用平台包括Binance、OKX和Huobi;交易流程包括注册账户、实名认证、绑定支付方式、充值法币并下单购买,可选择市价单或限价单;比特币存储方式有交易…

    2026年5月10日
    000
  • c++中的SFINAE技术是什么_c++模板编程中的SFINAE原理与应用

    SFINAE 是“替换失败不是错误”的原则,指模板实例化时若参数替换导致错误,只要存在其他合法候选,编译器不报错而是继续重载决议。它用于条件启用模板、类型检测等场景,如通过 decltype 或 enable_if 控制函数重载,实现类型特征判断。尽管 C++20 引入 Concepts 简化了部分…

    2026年5月10日
    000
  • Go语言mgo查询构建:深入理解bson.M与日期范围查询的正确实践

    本文旨在解决go语言mgo库中构建复杂查询时,特别是涉及嵌套`bson.m`和日期范围筛选的常见错误。我们将深入剖析`bson.m`的类型特性,解释为何直接索引`interface{}`会导致“invalid operation”错误,并提供一种推荐的、结构清晰的代码重构方案,以确保查询条件能够正确…

    2026年5月10日
    100
  • RichHandler与Rich Progress集成:解决显示冲突的教程

    在使用rich库的`richhandler`进行日志输出并同时使用`progress`组件时,可能会遇到显示错乱或溢出问题。这通常是由于为`richhandler`和`progress`分别创建了独立的`console`实例导致的。解决方案是确保日志处理器和进度条组件共享同一个`console`实例…

    2026年5月10日
    000
  • Golang goroutine与channel调试技巧

    使用go run -race检测数据竞争,结合runtime.NumGoroutine监控协程数量,通过pprof分析阻塞调用栈,利用select超时避免永久阻塞,有效排查goroutine泄漏、死锁和数据竞争问题。 Go语言的goroutine和channel是并发编程的核心,但它们也带来了调试上…

    2026年5月10日
    000
  • 《魔兽世界》将于6月11日开启国服回归技术测试

    《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试《魔兽世界》将于6月11日开启国服回归技术测试

    《%ign%ignore_a_1%re_a_1%》官方宣布,将于6月11日开启国服回归技术测试,时间为7天,并称可以在6月内正式开服,玩家们可以访问官网下载战网客户端并预下载“巫妖王之怒”客户端,技术测试详情见下图。 WordAi WordAI是一个AI驱动的内容重写平台 53 查看详情 以上就是《…

    2026年5月10日 用户投稿
    200
  • 使用 Jupyter Notebook 进行探索性数据分析

    Jupyter Notebook通过单元格实现代码与Markdown结合,支持数据导入(pandas)、清洗(fillna)、探索(matplotlib/seaborn可视化)、统计分析(describe/corr)和特征工程,便于记录与分享分析过程。 Jupyter Notebook 是进行探索性…

    2026年5月10日
    000
  • 如何在HTML中插入表单元素_HTML表单控件与输入类型使用指南

    HTML表单通过标签构建,包含action和method属性定义数据提交目标与方式,常用input类型如text、password、email等适配不同输入需求,配合label、required、placeholder提升可用性,结合textarea、select、button等控件实现完整交互,是…

    2026年5月10日
    000
  • 创建指定大小并填充特定数据的Golang文件教程

    本文将介绍如何使用Golang创建一个指定大小的文件,并用特定数据填充它。我们将使用 `os` 包提供的函数来创建和截断文件,从而实现快速生成大文件的目的。示例代码展示了如何创建一个10MB的文件,并将其填充为全零数据。掌握这些方法,可以方便地在例如日志系统或磁盘队列等场景中,预先创建测试文件或初始…

    2026年5月10日
    000
  • Python命令怎样使用profile分析脚本性能 Python命令性能分析的基础教程

    使用Python的cProfile模块分析脚本性能最直接的方式是通过命令行执行python -m cProfile your_script.py,它会输出每个函数的调用次数、总耗时、累积耗时等关键指标,帮助定位性能瓶颈;为进一步分析,可将结果保存为文件python -m cProfile -o ou…

    2026年5月10日
    000
  • 如何插入查询结果数据_SQL插入Select查询结果方法

    如何插入查询结果数据_SQL插入Select查询结果方法如何插入查询结果数据_SQL插入Select查询结果方法如何插入查询结果数据_SQL插入Select查询结果方法如何插入查询结果数据_SQL插入Select查询结果方法

    使用INSERT INTO…SELECT语句可高效插入数据,通过NOT EXISTS、LEFT JOIN、MERGE语句或唯一约束避免重复;表结构不一致时可通过别名、类型转换、默认值或计算字段处理;结合存储过程可提升可维护性,支持参数化与动态SQL。 将查询结果数据插入到另一个表中,可以…

    2026年5月10日 用户投稿
    000
  • 使用 WebCodecs VideoDecoder 实现精确逐帧回退

    本文档旨在解决在使用 WebCodecs VideoDecoder 进行视频解码时,实现精确逐帧回退的问题。通过比较帧的时间戳与目标帧的时间戳,可以避免渲染中间帧,从而提高用户体验。本文将提供详细的解决方案和示例代码,帮助开发者实现精确的视频帧控制。 在使用 WebCodecs VideoDecod…

    2026年5月10日
    000
  • Discord.py 交互按钮超时与持久化解决方案

    本教程旨在解决Discord.py中交互按钮在一段时间后出现“This Interaction Failed”错误的问题。我们将深入探讨视图(View)的超时机制,并提供通过正确设置timeout参数以及利用bot.add_view()方法实现按钮持久化的具体方案,确保您的机器人交互功能稳定可靠,即…

    2026年5月10日
    000
  • Debian Copilot的社区活跃度如何

    debian copilot是codeberg社区维护的ai助手,旨在为debian用户提供服务。尽管搜索结果中没有直接提供关于debian copilot社区支持活跃度的具体数据,但我们可以通过debian社区的整体活跃度和特点来推断其活跃性。 Debian社区的一般情况: Debian拥有详尽的…

    2026年5月10日
    000
  • JavaScript 动态菜单点击高亮效果实现教程

    本教程详细介绍了如何使用 JavaScript 实现动态菜单的点击高亮功能。通过事件委托和状态管理,当用户点击菜单项时,被点击项会高亮显示(绿色),同时其他菜单项恢复默认样式(白色)。这种方法避免了不必要的DOM操作,提高了性能和代码可维护性,确保了无论点击方向如何,功能都能稳定运行。 动态菜单高亮…

    2026年5月10日
    200
  • c++如何实现UDP通信_c++基于UDP的网络通信示例

    UDP通信基于套接字实现,适用于实时性要求高的场景。1. 流程包括创建套接字、绑定地址(接收方)、发送(sendto)与接收(recvfrom)数据、关闭套接字;2. 服务端监听指定端口,接收客户端消息并回传;3. 客户端发送消息至服务端并接收响应;4. 跨平台需处理Winsock初始化与库链接,编…

    2026年5月10日
    000
  • html5怎么画实线_HTML5用CSS border-style:solid画元素实线边框【绘制】

    可通过CSS的border-style属性设为solid添加实线边框:一、内联样式用border:2px solid #000;二、内部样式表统一设置如div{border:1px solid #333};三、外部CSS文件定义.my-box{border:3px solid red}并引入;四、单…

    2026年5月10日
    000

发表回复

登录后才能评论
关注微信