最新资讯

  • 如何在YOLOv8网络中添加自定义注意力机制

如何在YOLOv8网络中添加自定义注意力机制

2025-04-28 12:37:32 0 阅读

在目标检测任务中,加入注意力机制可以提升模型的检测效果。本文将介绍如何在YOLOv8模型中集成多种注意力机制,如 SimAMShuffleAttentionTripletAttentionMHSACBAMEMA,以增强模型对图像特征的提取能力。我们将展示每个注意力机制的代码示例,并讨论如何将这些模块添加到YOLOv8网络中。


目录

  • 1. 注意力机制示例代码
    • 1.1 SimAM 模块代码
    • 1.2 ShuffleAttention 模块代码
    • 1.3 TripletAttention 模块代码
    • 1.4 MHSA 模块代码
    • 1.5 CBAM 模块代码
    • 1.6 EMA 模块代码
    • 1.7 ECA 模块代码
  • 2. 添加注意力机制的步骤
    • 2.1 修改YOLOv8的配置文件
    • 2.2 编写自定义注意力机制模块
    • 2.3 训练和验证注意力机制
  • 3. 总结

1. 注意力机制示例代码

下面介绍六种常用的注意力机制模块,并提供代码示例。每种注意力机制都有其独特的优点,可以根据任务需求选择最适合的机制。

1.1 SimAM 模块代码

SimAM (Simple Attention Module) 是一种轻量级的注意力机制;通过简单的操作实现了注意力机制的效果,适用于对计算资源敏感的项目。它适合那些希望在提升模型性能的同时,尽量减少计算开销的任务,比如嵌入式设备上的实时目标检测。。

import torch
import torch.nn as nn


class SimAM(torch.nn.Module):
    def __init__(self, e_lambda=1e-4):
        super(SimAM, self).__init__()

        self.activaton = nn.Sigmoid()
        self.e_lambda = e_lambda

    def __repr__(self):
        s = self.__class__.__name__ + '('
        s += ('lambda=%f)' % self.e_lambda)
        return s

    @staticmethod
    def get_module_name():
        return "simam"

    def forward(self, x):
        b, c, h, w = x.size()

        n = w * h - 1

        x_minus_mu_square = (x - x.mean(dim=[2, 3], keepdim=True)).pow(2)
        y = x_minus_mu_square / (4 * (x_minus_mu_square.sum(dim=[2, 3], keepdim=True) / n + self.e_lambda)) + 0.5

        return x * self.activaton(y)


# if __name__ == '__main__':
#     input = torch.randn(3, 64, 7, 7)
#     model = SimAM()
#     outputs = model(input)
#     print(outputs.shape)

1.2 ShuffleAttention 模块代码

ShuffleAttention 适合需要全局特征交互的场景;通过通道洗牌操作重新排列特征,确保模型能够在不同通道间传递信息,提升特征的全局表达能力。对于需要处理复杂、具有多样性特征的图像(如交通场景、复杂的自然环境),这种机制能有效提升模型的感知能力。

import torch
from torch import nn
from torch.nn import init
from torch.nn.parameter import Parameter

class ShuffleAttention(nn.Module):

    def __init__(self, channel=512, reduction=16, G=8):
        super().__init__()
        self.G = G
        self.channel = channel
        self.avg_pool = nn.AdaptiveAvgPool2d(1)
        self.gn = nn.GroupNorm(channel // (2 * G), channel // (2 * G))
        self.cweight = Parameter(torch.zeros(1, channel // (2 * G), 1, 1))
        self.cbias = Parameter(torch.ones(1, channel // (2 * G), 1, 1))
        self.sweight = Parameter(torch.zeros(1, channel // (2 * G), 1, 1))
        self.sbias = Parameter(torch.ones(1, channel // (2 * G), 1, 1))
        self.sigmoid = nn.Sigmoid()

    def init_weights(self):
        for m in self.modules():
            if isinstance(m, nn.Conv2d):
                init.kaiming_normal_(m.weight, mode='fan_out')
                if m.bias is not None:
                    init.constant_(m.bias, 0)
            elif isinstance(m, nn.BatchNorm2d):
                init.constant_(m.weight, 1)
                init.constant_(m.bias, 0)
            elif isinstance(m, nn.Linear):
                init.normal_(m.weight, std=0.001)
                if m.bias is not None:
                    init.constant_(m.bias, 0)

    @staticmethod
    def channel_shuffle(x, groups):
        b, c, h, w = x.shape
        x = x.reshape(b, groups, -1, h, w)
        x = x.permute(0, 2, 1, 3, 4)

        # flatten
        x = x.reshape(b, -1, h, w)

        return x

    def forward(self, x):
        b, c, h, w = x.size()
        # group into subfeatures
        x = x.view(b * self.G, -1, h, w)  # bs*G,c//G,h,w

        # channel_split
        x_0, x_1 = x.chunk(2, dim=1)  # bs*G,c//(2*G),h,w

        # channel attention
        x_channel = self.avg_pool(x_0)  # bs*G,c//(2*G),1,1
        x_channel = self.cweight * x_channel + self.cbias  # bs*G,c//(2*G),1,1
        x_channel = x_0 * self.sigmoid(x_channel)

        # spatial attention
        x_spatial = self.gn(x_1)  # bs*G,c//(2*G),h,w
        x_spatial = self.sweight * x_spatial + self.sbias  # bs*G,c//(2*G),h,w
        x_spatial = x_1 * self.sigmoid(x_spatial)  # bs*G,c//(2*G),h,w

        # concatenate along channel axis
        out = torch.cat([x_channel, x_spatial], dim=1)  # bs*G,c//G,h,w
        out = out.contiguous().view(b, -1, h, w)

        # channel shuffle
        out = self.channel_shuffle(out, 2)
        return out


if __name__ == '__main__':
    input = torch.randn(50, 512, 7, 7)
    se = ShuffleAttention(channel=512, G=8)
    output = se(input)
1.3 TripletAttention 模块代码

TripletAttention 适合需要捕捉多方向特征的场景;在通道上引入了三个方向的注意力(水平、垂直、深度),能够帮助模型更好地感知多方向上的特征。这种机制特别适用于那些需要捕捉方向性信息的任务,比如道路标志检测和自然场景理解。

import torch
import torch.nn as nn


class BasicConv(nn.Module):  # https://arxiv.org/pdf/2010.03045.pdf
    def __init__(self, in_planes, out_planes, kernel_size, stride=1, padding=0, dilation=1, groups=1, relu=True,
                 bn=True, bias=False):
        super(BasicConv, self).__init__()
        self.out_channels = out_planes
        self.conv = nn.Conv2d(in_planes, out_planes, kernel_size=kernel_size, stride=stride, padding=padding,
                              dilation=dilation, groups=groups, bias=bias)
        self.bn = nn.BatchNorm2d(out_planes, eps=1e-5, momentum=0.01, affine=True) if bn else None
        self.relu = nn.ReLU() if relu else None

    def forward(self, x):
        x = self.conv(x)
        if self.bn is not None:
            x = self.bn(x)
        if self.relu is not None:
            x = self.relu(x)
        return x


class ZPool(nn.Module):
    def forward(self, x):
        return torch.cat((torch.max(x, 1)[0].unsqueeze(1), torch.mean(x, 1).unsqueeze(1)), dim=1)


class AttentionGate(nn.Module):
    def __init__(self):
        super(AttentionGate, self).__init__()
        kernel_size = 7
        self.compress = ZPool()
        self.conv = BasicConv(2, 1, kernel_size, stride=1, padding=(kernel_size - 1) // 2, relu=False)

    def forward(self, x):
        x_compress = self.compress(x)
        x_out = self.conv(x_compress)
        scale = torch.sigmoid_(x_out)
        return x * scale


class TripletAttention(nn.Module):
    def __init__(self, no_spatial=False):
        super(TripletAttention, self).__init__()
        self.cw = AttentionGate()
        self.hc = AttentionGate()
        self.no_spatial = no_spatial
        if not no_spatial:
            self.hw = AttentionGate()

    def forward(self, x):
        x_perm1 = x.permute(0, 2, 1, 3).contiguous()
        x_out1 = self.cw(x_perm1)
        x_out11 = x_out1.permute(0, 2, 1, 3).contiguous()
        x_perm2 = x.permute(0, 3, 2, 1).contiguous()
        x_out2 = self.hc(x_perm2)
        x_out21 = x_out2.permute(0, 3, 2, 1).contiguous()
        if not self.no_spatial:
            x_out = self.hw(x)
            x_out = 1 / 3 * (x_out + x_out11 + x_out21)
        else:
            x_out = 1 / 2 * (x_out11 + x_out21)
        return x_out
1.4 MHSA 模块代码

MHSA (Multi-Head Self-Attention) 是常用于Transformer模型的注意力机制,适合大规模上下文建模的场景;通过多头自注意力的机制,能够帮助模型捕捉图像中的长距离依赖关系。它在需要处理上下文信息的任务中表现出色,如自然场景中的多物体检测。对于需要全局信息并且图像内物体之间具有复杂相互关系的任务,MHSA 是理想的选择。

import torch
import torch.nn as nn

class MHSA(nn.Module):
    def __init__(self, n_dims, width=14, height=14, heads=4, pos_emb=False):
        super(MHSA, self).__init__()

        self.heads = heads
        self.query = nn.Conv2d(n_dims, n_dims, kernel_size=1)
        self.key = nn.Conv2d(n_dims, n_dims, kernel_size=1)
        self.value = nn.Conv2d(n_dims, n_dims, kernel_size=1)
        self.pos = pos_emb
        if self.pos:
            self.rel_h_weight = nn.Parameter(torch.randn([1, heads, (n_dims) // heads, 1, int(height)]),
                                             requires_grad=True)
            self.rel_w_weight = nn.Parameter(torch.randn([1, heads, (n_dims) // heads, int(width), 1]),
                                             requires_grad=True)
        self.softmax = nn.Softmax(dim=-1)

    def forward(self, x):
        n_batch, C, width, height = x.size()
        q = self.query(x).view(n_batch, self.heads, C // self.heads, -1)
        k = self.key(x).view(n_batch, self.heads, C // self.heads, -1)
        v = self.value(x).view(n_batch, self.heads, C // self.heads, -1)
        content_content = torch.matmul(q.permute(0, 1, 3, 2), k)  # 1,C,h*w,h*w
        c1, c2, c3, c4 = content_content.size()
        if self.pos:
            content_position = (self.rel_h_weight + self.rel_w_weight).view(1, self.heads, C // self.heads, -1).permute(
                0, 1, 3, 2)  # 1,4,1024,64

            content_position = torch.matmul(content_position, q)  # ([1, 4, 1024, 256])
            content_position = content_position if (
                    content_content.shape == content_position.shape) else content_position[:, :, :c3, ]
            assert (content_content.shape == content_position.shape)
            energy = content_content + content_position
        else:
            energy = content_content
        attention = self.softmax(energy)
        out = torch.matmul(v, attention.permute(0, 1, 3, 2))  # 1,4,256,64
        out = out.view(n_batch, C, width, height)
        return out

# if __name__ == '__main__':
#     input = torch.randn(50, 512, 7, 7)
#     mhsa = MHSA(n_dims=512)
#     output = mhsa(input)
#     print(output.shape)
1.5 CBAM 模块代码

CBAM(Convolutional Block Attention Module)适合需要结合通道和空间特征的场景;通过结合通道注意力和空间注意力,帮助网络更加精准地捕捉图像中的关键区域。它适用于大多数目标检测任务,特别是当需要细化某些特定物体的检测时,比如在自动驾驶中的行人检测或交通标志检测。

import torch
from torch import nn

class ChannelAttention(nn.Module):
    # Channel-attention module https://github.com/open-mmlab/mmdetection/tree/v3.0.0rc1/configs/rtmdet
    def __init__(self, channels: int) -> None:
        super().__init__()
        self.pool = nn.AdaptiveAvgPool2d(1)
        self.fc = nn.Conv2d(channels, channels, 1, 1, 0, bias=True)
        self.act = nn.Sigmoid()

    def forward(self, x: torch.Tensor) -> torch.Tensor:
        return x * self.act(self.fc(self.pool(x)))


class SpatialAttention(nn.Module):
    # Spatial-attention module
    def __init__(self, kernel_size=7):
        super().__init__()
        assert kernel_size in (3, 7), 'kernel size must be 3 or 7'
        padding = 3 if kernel_size == 7 else 1
        self.cv1 = nn.Conv2d(2, 1, kernel_size, padding=padding, bias=False)
        self.act = nn.Sigmoid()

    def forward(self, x):
        return x * self.act(self.cv1(torch.cat([torch.mean(x, 1, keepdim=True), torch.max(x, 1, keepdim=True)[0]], 1)))


class CBAM(nn.Module):
    # Convolutional Block Attention Module
    def __init__(self, c1, kernel_size=7):  # ch_in, kernels
        super().__init__()
        self.channel_attention = ChannelAttention(c1)
        self.spatial_attention = SpatialAttention(kernel_size)

    def forward(self, x):
        return self.spatial_attention(self.channel_attention(x))
1.6 EMA 模块代码

EMA(Efficient Multi-Head Attention)适合希望在多头自注意力中提升效率的场景;它通过减少计算复杂度而提升性能,适用于大规模数据集的训练。它在保持注意力机制强大的特征捕捉能力的同时,还能显著降低计算成本,适合高性能要求的任务场景。

import torch
from torch import nn

class EMA(nn.Module):
    def __init__(self, channels, c2=None, factor=32):
        super(EMA, self).__init__()
        self.groups = factor
        assert channels // self.groups > 0
        self.softmax = nn.Softmax(-1)
        self.agp = nn.AdaptiveAvgPool2d((1, 1))
        self.pool_h = nn.AdaptiveAvgPool2d((None, 1))
        self.pool_w = nn.AdaptiveAvgPool2d((1, None))
        self.gn = nn.GroupNorm(channels // self.groups, channels // self.groups)
        self.conv1x1 = nn.Conv2d(channels // self.groups, channels // self.groups, kernel_size=1, stride=1, padding=0)
        self.conv3x3 = nn.Conv2d(channels // self.groups, channels // self.groups, kernel_size=3, stride=1, padding=1)

    def forward(self, x):
        b, c, h, w = x.size()
        group_x = x.reshape(b * self.groups, -1, h, w)  # b*g,c//g,h,w
        x_h = self.pool_h(group_x)
        x_w = self.pool_w(group_x).permute(0, 1, 3, 2)
        hw = self.conv1x1(torch.cat([x_h, x_w], dim=2))
        x_h, x_w = torch.split(hw, [h, w], dim=2)
        x1 = self.gn(group_x * x_h.sigmoid() * x_w.permute(0, 1, 3, 2).sigmoid())
        x2 = self.conv3x3(group_x)
        x11 = self.softmax(self.agp(x1).reshape(b * self.groups, -1, 1).permute(0, 2, 1))
        x12 = x2.reshape(b * self.groups, c // self.groups, -1)  # b*g, c//g, hw
        x21 = self.softmax(self.agp(x2).reshape(b * self.groups, -1, 1).permute(0, 2, 1))
        x22 = x1.reshape(b * self.groups, c // self.groups, -1)  # b*g, c//g, hw
        weights = (torch.matmul(x11, x12) + torch.matmul(x21, x22)).reshape(b * self.groups, 1, h, w)
        return (group_x * weights.sigmoid()).reshape(b, c, h, w)
1.7 ECA 模块代码

ECA(Efficient Channel Attention)适合需要高效通道注意力的场景;通过消除全连接层,使用1D卷积进行局部交互,大大减少了参数量,同时仍然保留了通道注意力的能力。它适合那些对计算资源有限制的场景,比如移动设备上进行目标检测的任务。

import torch
from torch import nn

class ECA(nn.Module):
    def __init__(self, channels: int, k_size: int = 3):
        super(ECA, self).__init__()
        self.avg_pool = nn.AdaptiveAvgPool2d(1)
        self.conv = nn.Conv1d(1, 1, kernel_size=k_size, padding=(k_size - 1) // 2, bias=False)
        self.sigmoid = nn.Sigmoid()

    def forward(self, x):
        # Apply global average pooling
        y = self.avg_pool(x)

        # Reshape and apply 1D convolution
        y = self.conv(y.squeeze(-1).transpose(-1, -2)).transpose(-1, -2).unsqueeze(-1)

        # Apply sigmoid activation and element-wise multiplication
        return x * self.sigmoid(y)

2. 添加注意力机制的步骤

2.1 修改YOLOv8的配置文件

我们可以通过在YOLOv8配置文件中指定使用注意力机制,以下是如何在第10层加入注意力机制的配置示例,以ShuffleAttention注意力机制为例,用到哪个放开哪个:

# 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: 81 # 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

# YOLOv8.0n backbone
backbone:
  # [from, repeats, module, args]
  - [-1, 1, Conv, [64, 3, 2]] # 0-P1/2
  - [-1, 1, Conv, [128, 3, 2]] # 1-P2/4
  - [-1, 3, C2f, [128, True]]
  - [-1, 1, Conv, [256, 3, 2]] # 3-P3/8
  - [-1, 6, C2f, [256, True]]
  - [-1, 1, Conv, [512, 3, 2]] # 5-P4/16
  - [-1, 6, C2f, [512, True]]
  - [-1, 1, Conv, [1024, 3, 2]] # 7-P5/32
  - [-1, 3, C2f, [1024, True]]
  - [-1, 1, SPPF, [1024, 5]] # 9
#  - [-1, 1, SimAM, [1024]] # 10
  - [-1, 1, ShuffleAttention, [16, 8]] # 10
#  - [-1, 1, TripletAttention, [1024]]
#  - [-1, 1, MHSA, [14, 14, 4]] # 10
#  - [-1, 1, CBAM, [1024]] # 10
#  - [-1, 1, EMA, [1024, 8]] # 10
#  - [-1, 1, ECA, [1024]]  # 10

# YOLOv8.0n head
head:
  - [-1, 1, nn.Upsample, [None, 2, "nearest"]]
  - [[-1, 6], 1, Concat, [1]] # cat backbone P4
  - [-1, 3, C2f, [512]] # 13

  - [-1, 1, nn.Upsample, [None, 2, "nearest"]]
  - [[-1, 4], 1, Concat, [1]] # cat backbone P3
  - [-1, 3, C2f, [256]] # 16 (P3/8-small)

  - [-1, 1, Conv, [256, 3, 2]]
  - [[-1, 13], 1, Concat, [1]] # cat head P4
  - [-1, 3, C2f, [512]] # 19 (P4/16-medium)

  - [-1, 1, Conv, [512, 3, 2]]
  - [[-1, 10], 1, Concat, [1]] # cat head P5
  - [-1, 3, C2f, [1024]] # 22 (P5/32-large)

  - [[16, 19, 22], 1, Detect, [nc]] # Detect(P3, P4, P5) ShuffleAttention

2.2 编写自定义注意力机制模块

为了在YOLOv8中集成自定义的注意力机制模块,我们需要将相应的注意力机制代码文件(如 SimAM.pyShuffleAttention.pyTripletAttention.pyMHSA.pyCBAM.pyEMA.pyECA.py)放入项目中的 ultralytics/nn/ 目录下,并在 task.py 文件中进行相应的导入和修改。

步骤 1:将注意力机制模块文件放入 ultralytics/nn/ 目录

首先,确保所有注意力机制模块的代码文件都放在 ultralytics/nn/ 目录下。注意力机制模块的文件名如下:

  • SimAM.py
  • ShuffleAttention.py
  • TripletAttention.py
  • MHSA.py
  • CBAM.py
  • EMA.py
  • ECA.py

如图是我的目录:

步骤 2:在 task.py 中导入注意力机制模块

接下来,在 ultralytics/nn/tasks.py 文件的头部导入这些注意力机制模块。添加以下代码:

from ultralytics.nn.MHSA import MHSA
from ultralytics.nn.ShuffleAttention import ShuffleAttention
from ultralytics.nn.SimAM import SimAM
from ultralytics.nn.CBAM import CBAM
from ultralytics.nn.TripletAttention import TripletAttention
from ultralytics.nn.EMA import EMA
from ultralytics.nn.ECA import ECA
步骤 3:在 task.py 的 910 行到950行任意位置添加注意力机制的构建逻辑

task.py 文件中的约 910 行到950行的位置,添加代码逻辑以确保这些注意力机制模块在网络中正确初始化。可以参考以下代码:

## 添加注意力机制
elif m in {MHSA, ShuffleAttention}:
    args = [ch[f], *args]
elif m in {EMA}:
    args = [ch[f]]
elif m in (SimAM, CBAM, TripletAttention, ECA):
    c1, c2 = ch[f], args[0]
    if c2 != nc:
        c2 = make_divisible(min(c2, max_channels) * width, 8)
    args = [c1, *args[1:]]

如图是我添加的位置:

此代码块确保在网络结构中正确处理和初始化各类注意力机制。

小结

通过上述步骤,我们可以成功将自定义的注意力机制集成到YOLOv8中。在 task.py 中导入相应模块并添加初始化逻辑后,您可以通过修改配置文件来选择使用哪种注意力机制,并在训练过程中验证其效果。

2.3 训练和验证注意力机制

使用修改后的配置文件进行训练(我修改后的配置文件是yolov8_att.yaml),运行训练代码时候用yolov8s_att.yaml,其中的yolov8后面加了一个s,证明使用s网络训练;查看日志输出,能够看到类似如下红框的输出,证明 ShuffleAttention 已成功加载,其他注意力机制类似:

from ultralytics import YOLO

if __name__ == '__main__':
    model = YOLO("yolov8s_att.yaml").load("yolov8s.pt")  # build from YAML and transfer weights

    # Train the model
    results = model.train(data=r"traffic_signage.yaml", epochs=150, batch=16, imgsz=1280)


如何将注意力机制应用到项目中?

在实际项目中,不同的注意力机制具有不同的侧重点,适用于不同的任务场景。根据项目的具体需求,选择合适的注意力机制可以大大提升模型的表现。:

  1. 轻量模型优化:如果你的项目目标是提升模型的实时性并在嵌入式设备上运行,可以选择 SimAMECA 这样的轻量级注意力机制。
  2. 大规模场景建模:对于涉及复杂背景的目标检测任务,MHSACBAM 可以有效增强模型对全局特征的捕捉能力。
  3. 多目标检测:当你的项目涉及复杂的多目标检测场景(如自动驾驶中的多物体检测),可以尝试使用 ShuffleAttentionTripletAttention,它们在捕捉多方向和通道交互特征方面表现出色。
  4. 计算效率优化:如果你关注模型的计算效率,可以选择 EMA 来在保持高性能的同时减少计算开销。

通过合理选择和应用这些注意力机制,能够为目标检测、分类和语义分割等任务带来显著的性能提升。

3. 总结

本文详细介绍了六种常见的注意力机制,并展示了如何将它们集成到YOLOv8网络中。通过添加注意力机制,模型可以更好地捕捉图像中的重要信息,从而提升目标检测性能。

本文地址:https://www.vps345.com/4628.html

搜索文章

Tags

PV计算 带宽计算 流量带宽 服务器带宽 上行带宽 上行速率 什么是上行带宽? CC攻击 攻击怎么办 流量攻击 DDOS攻击 服务器被攻击怎么办 源IP 服务器 linux 运维 游戏 云计算 ssh deepseek Ollama 模型联网 API CherryStudio llama 算法 opencv 自然语言处理 神经网络 语言模型 javascript 前端 chrome edge 进程 操作系统 进程控制 Ubuntu python MCP 数据库 centos oracle 关系型 安全 分布式 阿里云 网络 网络安全 网络协议 ubuntu harmonyos 华为 开发语言 typescript 计算机网络 银河麒麟 kylin v10 麒麟 v10 spring boot websocket docker 实时音视频 自动化 蓝耘科技 元生代平台工作流 ComfyUI 人工智能 adb nginx 监控 自动化运维 笔记 C 环境变量 进程地址空间 node.js json html5 firefox gitlab numpy java 经验分享 react.js 前端面试题 持续部署 zotero WebDAV 同步失败 代理模式 科技 ai 个人开发 spring ollama llm php android c++ c语言 Dell R750XS nuxt3 vue3 sql KingBase YOLO 网络结构图 fastapi mcp mcp-proxy mcp-inspector fastapi-mcp agent sse 豆瓣 追剧助手 迅雷 nas 微信 计算机视觉 pytorch 内存 深度学习 目标检测 LDAP https aws googlecloud 服务器繁忙 备选 网站 api 调用 示例 postman mock mock server 模拟服务器 mock服务器 Postman内置变量 Postman随机数据 tomcat maven intellij idea uni-app udp 智能路由器 外网访问 内网穿透 端口映射 macos Cursor android studio 交互 后端 shell 消息队列 pycharm conda pillow spring cloud intellij-idea kafka hibernate pygame 小游戏 五子棋 统信 国产操作系统 虚拟机安装 游戏程序 windows github git ftp http 容器 apache 孤岛惊魂4 AI 爬虫 数据集 kubernetes 学习方法 程序人生 golang 串口服务器 向日葵 MQTT ssl JAVA IDEA Java DeepSeek-R1 API接口 Headless Linux pdf flash-attention 报错 多线程服务器 Linux网络编程 visualstudio kylin 银河麒麟操作系统 国产化 idm 编辑器 redis live555 rtsp rtp tcp/ip 嵌入式硬件 驱动开发 硬件工程 嵌入式实习 Hyper-V WinRM TrustedHosts Reactor 设计模式 性能优化 C++ ecmascript nextjs react reactjs flask 网络工程师 华为认证 搜索引擎 HTML audio 控件组件 vue3 audio音乐播放器 Audio标签自定义样式默认 vue3播放音频文件音效音乐 自定义audio播放器样式 播放暂停调整声音大小下载文件 MI300x DeepSeek 机器学习 架构 string模拟实现 深拷贝 浅拷贝 经典的string类问题 三个swap 开发环境 SSL证书 小程序 svn 创意 社区 Dify c# Flask FastAPI Waitress Gunicorn uWSGI Uvicorn prometheus rpc 远程过程调用 Windows环境 直播推流 ffmpeg 音视频 面试 jdk RTSP xop RTP RTSPServer 推流 视频 kvm 无桌面 命令行 佛山戴尔服务器维修 佛山三水服务器维修 flutter 联想开天P90Z装win10 matlab jenkins 云原生 ci/cd YOLOv8 NPU Atlas800 A300I pro asi_bench ecm bpm vscode ide mount挂载磁盘 wrong fs type LVM挂载磁盘 Centos7.9 rust MCP server C/S LLM agi AIGC ansible playbook gpu算力 AI编程 远程登录 telnet IIS .net core Hosting Bundle .NET Framework vs2022 华为云 物联网 fpga开发 FunASR ASR web安全 集成学习 集成测试 GaN HEMT 氮化镓 单粒子烧毁 辐射损伤 辐照效应 权限 mysql 媒体 深度优先 图论 并集查找 换根法 树上倍增 MNN Qwen 电脑 ESP32 mongodb AI agent java-ee IIS服务器 IIS性能 日志监控 next.js 部署 部署next.js 腾讯云 国标28181 视频监控 监控接入 语音广播 流程 SIP SDP qt TCP服务器 qt项目 qt项目实战 qt教程 bash Docker Hub docker pull 镜像源 daemon.json Linux 根服务器 vim 宝塔面板 同步 备份 建站 安全威胁分析 重启 排查 系统重启 日志 原因 vscode 1.86 SSH Xterminal 系统安全 k8s资源监控 annotations自动化 自动化监控 监控service 监控jvm 开源 IM即时通讯 QQ 企业微信 剪切板对通 HTML FORMAT 安全架构 gcc openEuler unity unity3d log4j embedding sqlserver web 输入法 网络穿透 云服务器 Windows Netty 即时通信 NIO jvm 远程连接 远程 命令 执行 sshpass 操作 单片机 HTTP 服务器控制 ESP32 DeepSeek can 线程池 反向代理 自动驾驶 致远OA OA服务器 服务器磁盘扩容 LORA 大语言模型 NLP 游戏机 r语言 microsoft Linux PID 学习 C语言 linux上传下载 健康医疗 互联网医院 filezilla 无法连接服务器 连接被服务器拒绝 vsftpd 331/530 信号处理 信息与通信 远程工作 vasp安装 eureka 查询数据库服务IP地址 SQL Server WSL2 语音识别 AutoDL HCIE 数通 能力提升 面试宝典 技术 IT信息化 VMware安装Ubuntu Ubuntu安装k8s k8s 客户端 mosquitto vue.js SVN Server tortoise svn 数据挖掘 数据可视化 数据分析 计算机 程序员 鸿蒙 鸿蒙系统 其他 华为od django sqlite 服务器主板 AI芯片 交换机 dubbo rabbitmq openssl 密码学 hadoop gateway Clion Nova ResharperC++引擎 Centos7 远程开发 业界资讯 chatgpt DevEco Studio HarmonyOS OpenHarmony 真机调试 kamailio sip VoIP 大数据 大数据平台 代码调试 ipdb 烟花代码 烟花 元旦 asm debian etl dify 硬件架构 运维开发 webstorm ukui 麒麟kylinos openeuler sdkman rust腐蚀 微服务 postgresql .net selenium 测试工具 jar 回显服务器 UDP的API使用 vSphere vCenter db n8n dity make okhttp Java Applet URL操作 服务器建立 Socket编程 网络文件读取 armbian u-boot AI大模型 大模型 大模型入门 Deepseek 大模型教程 升级 CVE-2024-7347 漏洞 相机 需求分析 规格说明书 zabbix 安装教程 GPU环境配置 Ubuntu22 CUDA PyTorch Anaconda安装 微信小程序 elasticsearch web3.py ruoyi springboot 飞书 web3 恒源云 游戏引擎 抗锯齿 僵尸进程 tcp autodl big data mysql离线安装 ubuntu22.04 mysql8.0 opensearch helm 源码 毕业设计 课程设计 hive Hive环境搭建 hive3环境 Hive远程模式 open webui IMM 无人机 echarts onlyoffice 在线office oceanbase 传统数据库升级 银行 前端框架 LLMs 单一职责原则 Python 网络编程 聊天服务器 套接字 TCP Socket rclone AList webdav fnOS 监控k8s集群 集群内prometheus IPMITOOL BMC 硬件管理 oneapi 软件测试 chrome 浏览器下载 chrome 下载安装 mac 谷歌浏览器下载 程序 编程 性能分析 stm32 KylinV10 麒麟操作系统 虚拟机 Vmware 系统架构 计算机外设 gitea asp.net大文件上传 asp.net大文件上传下载 asp.net大文件上传源码 ASP.NET断点续传 asp.net上传文件夹 asp.net上传大文件 .net core断点续传 中间件 iis 移动云 负载均衡 云服务 ddos 可信计算技术 1024程序员节 鲲鹏 GCC aarch64 编译安装 HPC nfs 实时互动 SSL 域名 rsyslog safari pip Mac 系统 ruby 历史版本 下载 安装 网络攻击模型 显卡驱动 魔百盒刷机 移动魔百盒 机顶盒ROM devops Trae IDE AI 原生集成开发环境 Trae AI 工作流 workflow tcpdump WSL win11 无法解析服务器的名称或地址 树莓派 VNC 本地部署 微信小程序域名配置 微信小程序服务器域名 微信小程序合法域名 小程序配置业务域名 微信小程序需要域名吗 微信小程序添加域名 机器人 webrtc EasyConnect yum Cline 黑客技术 Linux24.04 deepin 虚幻 ssrf 失效的访问控制 openwrt ShenTong grafana ux 多线程 fd 文件描述符 open Euler dde 统信UOS LLM Web APP Streamlit deepseek r1 Kali iperf3 带宽测试 网工 springboot远程调试 java项目远程debug docker远程debug java项目远程调试 springboot远程 游戏服务器 TrinityCore 魔兽世界 P2P HDLC 思科 SenseVoice sysctl.conf vm.nr_hugepages 数学建模 adobe elk bug 环境迁移 eclipse HarmonyOS Next wireshark Ubuntu 24.04.1 轻量级服务器 镜像 python3.11 群晖 文件分享 软件工程 Agent Docker Compose docker compose docker-compose odoo 服务器动作 Server action 远程桌面 netty sentinel CPU AI-native Docker Desktop navicat TRAE 高效日志打印 串口通信日志 服务器日志 系统状态监控日志 异常记录日志 minio ROS wps 安卓 less 微信公众平台 ipython MacOS录屏软件 linux安装配置 ui seatunnel DigitalOcean GPU服务器购买 GPU服务器哪里有 GPU服务器 加解密 Yakit yaklang 多个客户端访问 IO多路复用 TCP相关API gitee bootstrap html 温湿度数据上传到服务器 Arduino HTTP 软考 黑客 jupyter 嵌入式 linux驱动开发 arm开发 C++软件实战问题排查经验分享 0xfeeefeee 0xcdcdcdcd 动态库加载失败 程序启动失败 程序运行权限 标准用户权限与管理员权限 智能手机 矩阵 医疗APP开发 app开发 麒麟 bonding 链路聚合 模拟器 教程 压力测试 开机自启动 rag ragflow ragflow 源码启动 tailscale derp derper 中转 express Minecraft 工业4.0 防火墙 NAT转发 NAT Server Unity Dedicated Server Host Client 无头主机 stm32项目 ip wsl frp unix 博客 cursor windows日志 glibc npm redhat ios 小智AI服务端 xiaozhi TTS thingsboard iDRAC R720xd Linux awk awk函数 awk结构 awk内置变量 awk参数 awk脚本 awk详解 Linux的基础指令 freebsd apt 昇腾 npu dell服务器 go css3 中兴光猫 换光猫 网络桥接 自己换光猫 服务器无法访问 ip地址无法访问 无法访问宝塔面板 宝塔面板打不开 XFS xfs文件系统损坏 I_O error es ArkUI 多端开发 智慧分发 应用生态 鸿蒙OS HiCar CarLife+ CarPlay QT RK3588 国内源 iot springsecurity6 oauth2 授权服务器 前后端分离 tensorflow 毕昇JDK xpath定位元素 file server http server web server 个人博客 X11 Xming 主板 电源 网卡 单元测试 功能测试 rdp 实验 我的世界服务器搭建 linux 命令 sed 命令 王者荣耀 Wi-Fi Spring Security 我的世界 我的世界联机 数码 rtsp服务器 rtsp server android rtsp服务 安卓rtsp服务器 移动端rtsp服务 大牛直播SDK .netcore rc.local 开机自启 systemd 缓存 ISO镜像作为本地源 virtualenv 实习 云电竞 云电脑 todesk 交叉编译 职场和发展 wsl2 jetty undertow Linux无人智慧超市 LInux多线程服务器 QT项目 LInux项目 单片机项目 UOS 统信操作系统 NAS Termux Samba selete 高级IO Linux环境 备份SQL Server数据库 数据库备份 傲梅企业备份网络版 llama3 Chatglm 开源大模型 IMX317 MIPI H265 VCU W5500 OLED u8g2 gaussdb 银河麒麟桌面操作系统 Kylin OS xss arm 宝塔面板访问不了 宝塔面板网站访问不了 宝塔面板怎么配置网站能访问 宝塔面板配置ip访问 宝塔面板配置域名访问教程 宝塔面板配置教程 pppoe radius 金仓数据库 2025 征文 数据库平替用金仓 图像处理 3d hugo 5G 3GPP 卫星通信 思科模拟器 Cisco audio vue音乐播放器 vue播放音频文件 Audio音频播放器自定义样式 播放暂停进度条音量调节快进快退 自定义audio覆盖默认样式 kind AI写作 AI作画 聊天室 银河麒麟服务器操作系统 系统激活 小艺 Pura X excel OD机试真题 华为OD机试真题 服务器能耗统计 visual studio code Radius muduo 低代码 弹性计算 虚拟化 KVM 计算虚拟化 弹性裸金属 windwos防火墙 defender防火墙 win防火墙白名单 防火墙白名单效果 防火墙只允许指定应用上网 防火墙允许指定上网其它禁止 智能音箱 智能家居 数据结构 jmeter EMQX 通信协议 Python基础 Python教程 Python技巧 策略模式 单例模式 junit XCC Lenovo SEO 繁忙 解决办法 替代网站 汇总推荐 AI推理 CDN 显示管理器 lightdm gdm dba 文件系统 路径解析 laravel ollama下载加速 vue css wsgiref Web 服务器网关接口 Nuxt.js 基础环境 list 流水线 脚本式流水线 efficientVIT YOLOv8替换主干网络 TOLOv8 skynet RAGFlow ubuntu20.04 开机黑屏 热榜 裸金属服务器 弹性裸金属服务器 cnn DenseNet p2p 命名管道 客户端与服务端通信 CrewAI 7z 离线部署dify Ubuntu共享文件夹 共享目录 Linux共享文件夹 av1 电视盒子 keepalived neo4j 知识图谱 沙盒 sonoma 自动更新 dns word eNSP 企业网络规划 华为eNSP 网络规划 xshell termius iterm2 增强现实 沉浸式体验 应用场景 技术实现 案例分析 AR 多路转接 epoll 数据仓库 数据库开发 数据库架构 database Xinference AD域 阿里云ECS ardunio BLE CORS 跨域 SSH 服务 SSH Server OpenSSH Server 大模型微调 ssh远程登录 ArcTS 登录 ArcUI GridItem arkUI 虚幻引擎 服务网格 istio js DocFlow 图形化界面 gpt CH340 串口驱动 CH341 uart 485 ubuntu24 vivado24 LInux 边缘计算 智能硬件 chrome devtools chromedriver USB网络共享 Playwright 自动化测试 xcode vmware 卡死 EtherNet/IP串口网关 EIP转RS485 EIP转Modbus EtherNet/IP网关协议 EIP转RS485网关 EIP串口服务器 自动化编程 code-server 合成模型 扩散模型 图像生成 word图片自动上传 word一键转存 复制word图片 复制word图文 复制word公式 粘贴word图文 粘贴word公式 序列化反序列化 浏览器开发 AI浏览器 ssh漏洞 ssh9.9p2 CVE-2025-23419 wpf Google pay Apple pay k8s集群资源管理 云原生开发 cd 目录切换 MS Materials RAGFLOW 模拟退火算法 ros2 moveit 机器人运动 YOLOv12 银河麒麟高级服务器 外接硬盘 Kylin Ubuntu 24 常用命令 Ubuntu 24 Ubuntu vi 异常处理 鸿蒙开发 移动开发 flink AI代码编辑器 信息可视化 网页设计 华为机试 ai小智 语音助手 ai小智配网 ai小智教程 esp32语音助手 diy语音助手 数据库系统 语法 lsb_release /etc/issue /proc/version uname -r 查看ubuntu版本 CentOS Stream CentOS AISphereButler remote-ssh 黑苹果 VMware tidb GLIBC 蓝桥杯 sequoiaDB 框架搭建 捆绑 链接 谷歌浏览器 youtube google gmail 视觉检测 图形渲染 VMware创建虚拟机 做raid 装系统 火绒安全 VPS DBeaver kerberos cuda cudnn nvidia 搭建个人相关服务器 内网服务器 内网代理 内网通信 sqlite3 VM搭建win2012 win2012应急响应靶机搭建 攻击者获取服务器权限 上传wakaung病毒 应急响应并溯源 挖矿病毒处置 应急响应综合性靶场 prometheus数据采集 prometheus数据模型 prometheus特点 Windsurf searxng 网络药理学 生物信息学 生信 PPI String Cytoscape CytoHubba RustDesk自建服务器 rustdesk服务器 docker rustdesk 测试用例 Docker引擎已经停止 Docker无法使用 WSL进度一直是0 镜像加速地址 perf 信创 信创终端 中科方德 alias unalias 别名 rime 影刀 #影刀RPA# mcu 雨云 NPS GPU WebRTC uniapp 混合开发 环境安装 JDK TCP协议 regedit 开机启动 camera Arduino 电子信息 composer vscode1.86 1.86版本 ssh远程连接 Redis Desktop wordpress 无法访问wordpess后台 打开网站页面错乱 linux宝塔面板 wordpress更换服务器 产测工具框架 IMX6ULL 管理框架 软件定义数据中心 sddc RTMP 应用层 firewall openstack Xen Logstash 日志采集 webgl 本地化部署 程序员创富 nlp 开发 centos-root /dev/mapper yum clean all df -h / du -sh figma 考研 链表 gradle milvus 京东云 架构与原理 C# MQTTS 双向认证 emqx 基础入门 opcua opcda KEPServer安装 FTP 服务器 私有化 GoogLeNet spark HistoryServer Spark YARN jobhistory 玩机技巧 软件分享 软件图标 matplotlib VSCode 技能大赛 token sas 具身智能 强化学习 dock 加速 prompt 僵尸世界大战 游戏服务器搭建 政务 分布式系统 监控运维 Prometheus Grafana 数据管理 数据治理 数据编织 数据虚拟化 EtherCAT转Modbus ECT转Modbus协议 EtherCAT转485网关 ECT转Modbus串口网关 EtherCAT转485协议 ECT转Modbus网关 proxy模式 充电桩 欧标 OCPP zookeeper 服务器部署ai模型 iBMC UltraISO mamba Anolis nginx安装 linux插件下载 lua 虚拟局域网 vue-i18n 国际化多语言 vue2中英文切换详细教程 如何动态加载i18n语言包 把语言json放到服务器调用 前端调用api获取语言配置文件 MySql 服务器数据恢复 数据恢复 存储数据恢复 raid5数据恢复 磁盘阵列数据恢复 自定义客户端 SAS fast 大模型应用 USB转串口 飞牛NAS 飞牛OS MacBook Pro harmonyOS面试题 域名服务 DHCP 符号链接 配置 GIS 遥感 WebGIS OpenSSH 邮件APP 免费软件 音乐库 飞牛 实用教程 软件需求 docker run 数据卷挂载 交互模式 gpt-3 文心一言 大大通 第三代半导体 碳化硅 Ubuntu Server Ubuntu 22.04.5 ai工具 java-rocketmq v10 软件 ldap 金融 知识库 本地知识库部署 DeepSeek R1 模型 pyqt 网络用户购物行为分析可视化平台 大数据毕业设计 ip命令 新增网卡 新增IP 启动网卡 Kylin-Server 服务器安装 内网环境 Kali Linux 渗透测试 信息收集 h.264 实战案例 IPv4 子网掩码 公网IP 私有IP SSH 密钥生成 SSH 公钥 私钥 生成 etcd 数据安全 RBAC 流式接口 URL 人工智能生成内容 ESXi 项目部署到linux服务器 项目部署过程 压测 ECS 线程 PX4 拓扑图 SSE VLAN 企业网络 状态管理的 UDP 服务器 Arduino RTOS NFS 网卡的名称修改 eth0 ens33 linux环境变量 大文件分片上传断点续传及进度条 如何批量上传超大文件并显示进度 axios大文件切片上传详细教 node服务器合并切片 vue3大文件上传报错提示错误 大文件秒传跨域报错cors 模拟实现 cpp-httplib mariadb xrdp cocoapods threejs 3D make命令 makefile文件 SRS 流媒体 直播 毕设 粘包问题 技术共享 AP配网 AK配网 小程序AP配网和AK配网教程 WIFI设备配网小程序UDP开 Deepseek-R1 私有化部署 推理模型 QT 5.12.12 QT开发环境 Ubuntu18.04 iphone 双系统 GRUB引导 Linux技巧 docker搭建nacos详解 docker部署nacos docker安装nacos 腾讯云搭建nacos centos7搭建nacos Qwen2.5-coder 离线部署 iftop 网络流量监控 uv 崖山数据库 YashanDB 视频编解码 源码剖析 rtsp实现步骤 流媒体开发 RAID RAID技术 磁盘 存储 dash 正则表达式 SysBench 基准测试 项目部署 性能测试 yolov8 雨云服务器 mybatis nac 802.1 portal 远程控制 远程看看 远程协助 软负载 多进程 cpu 实时 使用 midjourney 相差8小时 UTC 时间 AI Agent 字节智能运维 办公自动化 自动化生成 pdf教程 llama.cpp rnn risc-v Dell HPE 联想 浪潮 Attention win服务器架设 windows server trea idea 迁移指南 swoole 三级等保 服务器审计日志备份 Vmamba FTP服务器 ceph Invalid Host allowedHosts g++ g++13 Cookie .net mvc断点续传 visual studio 产品经理 干货分享 黑客工具 密码爆破 MDK 嵌入式开发工具 论文笔记 sublime text arcgis yaml Ultralytics 可视化 Ubuntu DeepSeek DeepSeek Ubuntu DeepSeek 本地部署 DeepSeek 知识库 DeepSeek 私有化知识库 本地部署 DeepSeek DeepSeek 私有化部署 RAG 检索增强生成 文档解析 大模型垂直应用 流量运营 状态模式 深度求索 私域 c 服务器管理 配置教程 网站管理 ue4 着色器 ue5 seleium 宕机切换 服务器宕机 剧本 pyautogui rocketmq bcompare Beyond Compare 执法记录仪 智能安全帽 smarteye triton 模型分析 uni-file-picker 拍摄从相册选择 uni.uploadFile H5上传图片 微信小程序上传图片 线性代数 电商平台 运维监控 服务器时间 transformer VS Code leetcode 推荐算法 Linux find grep jina 匿名管道 DOIT 四博智联 代理 嵌入式系统开发 代理服务器 腾讯云大模型知识引擎 anaconda bot Docker springcloud hexo 系统开发 binder 车载系统 framework 源码环境 常用命令 文本命令 目录命令 kali 共享文件夹 AD 域管理 lio-sam SLAM 网站搭建 serv00 端口测试 嵌入式Linux IPC H3C 微信开放平台 微信公众号配置 EMUI 回退 降级 gnu PVE Unity插件 iventoy VmWare OpenEuler 远程服务 x64 SIGSEGV xmm0 conda配置 conda镜像源 磁盘监控 服务器配置 Node-Red 编程工具 流编程 Typore 灵办AI 大模型部署 上传视频至服务器代码 vue3批量上传多个视频并预览 如何实现将本地视频上传到网页 element plu视频上传 ant design vue vue3本地上传视频及预览移除 bat 端口 查看 ss minecraft TrueLicense firewalld Open WebUI docker部署翻译组件 docker部署deepl docker搭建deepl java对接deepl 翻译组件使用 IO模型 edge浏览器 RoboVLM 通用机器人策略 VLA设计哲学 vlm fot robot 视觉语言动作模型 元服务 应用上架 半虚拟化 硬件虚拟化 Hypervisor 超融合 DNS 换源 Debian ubuntu24.04.1 crosstool-ng micropython esp32 mqtt UDP langchain deep learning rpa 游戏开发 pgpool IPMI 带外管理 硬件 设备 PCI-Express 田俊楠 自动化任务管理 easyui grub 版本升级 扩容 trae ping++ outlook minicom 串口调试工具 多层架构 解耦 磁盘镜像 服务器镜像 服务器实时复制 实时文件备份 Erlang OTP gen_server 热代码交换 事务语义 大模型推理 大模型学习 deekseek ABAP yum源切换 更换国内yum源 音乐服务器 Navidrome 音流 rustdesk chfs ubuntu 16.04 分析解读 SWAT 配置文件 服务管理 网络共享 微信分享 Image wxopensdk dns是什么 如何设置电脑dns dns应该如何设置 vr DeepSeek行业应用 Heroku 网站部署 在线预览 xlsx xls文件 在浏览器直接打开解析xls表格 前端实现vue3打开excel 文件地址url或接口文档流二进 存储维护 NetApp存储 EMC存储 ocr docker命令大全 mm-wiki搭建 linux搭建mm-wiki mm-wiki搭建与使用 mm-wiki使用 mm-wiki详解 风扇控制软件 分布式训练 信号 hosts Linux的权限 怎么卸载MySQL MySQL怎么卸载干净 MySQL卸载重新安装教程 MySQL5.7卸载 Linux卸载MySQL8.0 如何卸载MySQL教程 MySQL卸载与安装 算力 李心怡 Ark-TS语言 MacMini 迷你主机 mini Apple 宠物 免费学习 宠物领养 宠物平台 物联网开发 cmos clickhouse 北亚数据恢复 oracle数据恢复 docker部署Python 社交电子 高效远程协作 TrustViewer体验 跨设备操作便利 智能远程控制 西门子PLC 通讯 HAProxy 软链接 硬链接 直流充电桩 VMware安装mocOS macOS系统安装 WebUI DeepSeek V3 VR手套 数据手套 动捕手套 动捕数据手套 环境配置 ros Claude 大模型面经 阻塞队列 生产者消费者模型 服务器崩坏原因 AnythingLLM AnythingLLM安装