当前位置: 首页 > article >正文

C++ OpenCV实现简单的自瞄脚本(OpenCV实战)

练枪的时候发现打的靶子特征很醒目,而且操控的逻辑也不是说特别难,刚好会一点点C++和OpenCV,为什么不试试写一个小程序来帮助我们瞄准呢?

实现效果

FabConvert.com_8c0d30f1f7092a48e2fa05163023e292

我们主要是通过这款游戏测试自瞄

image-20240912163608365

简单的调参之后本周世界排名也是打到了第一名,下面是实现的简单思路和逻辑,可以跟着我一起实战一下

dfd81ce633476a03a9abb4c7394a03c

主要思路讲解

我们实际上实现自动瞄准的逻辑也很简单

识别目标,确定目标坐标,移动鼠标,开枪

实现这四步就可以很简单的实现自瞄,转化为代码,我们则需要做到下面两个步骤

  • 目标识别检测
  • 鼠标位置控制

下面我们分别来讲解实现这两个功能


目标识别

这里用的是OpenCV来实现检测,因为这一次的目标比较简单,大概都是这样的蓝色小圆球

image-20240912171122227

其实这里面思路有很多,可以进行进行边缘检测,阈值监测什么的,但是毕竟是自瞄,我们要选择一个计算任务小的,这样子响应速度更快

如果是识别圆形轮廓,就分为了利用算子进行边缘检测,识别圆形轮廓,这样子不仅计算量大,在手枪出现挡住靶子的情况也没有办法及时识别(上图所示),于是我们果断抛弃这种思路。

ps:为了方便看效果,我简单的写一个小程序来显示图片,捕获屏幕和移动鼠标后面会讲到,这里我放一下测试的小程序

int main()
{
 Mat img = cv::imread("D:\desktop\\test.png");
 while (true)
 {
     cv::imshow("Original Image", img);
     cv::waitKey(100);
 }
}

image-20241017091712047

还有一种思路,我们直接寻找蓝色,这样子也分为以下几个步骤

  • 创建一个掩膜,只保留蓝色区域

     Scalar lower_blue(82, 199, 118);
     Scalar upper_blue(97, 255, 255);
    
  • 使用掩膜提取蓝色区域

     Mat mask;
     inRange(hsv, lower_blue, upper_blue, mask);
    

    image-20241017093230428

  • 查找轮廓

    Canny(mask, edges, 80, 255);
    

    image-20241017093246285

接下来就是确定鼠标点击的点,在这里也是测试了多种情况,最好的办法是直接识别质心,这样子可以使得鼠标的容错会更大,如何去除误识别区域呢,这个通过优化算法可以达到很好的效果,但是会极大地增大延时,所以在前面蓝色区域识别以及很精确地调试过了,这里只需要规定目标轮廓体积达到阈值即可

我们写一下代码,首先是质心的寻找

// 存储找到的轮廓
    vector<vector<Point>> contours;
    // 在经过边缘检测的图像 edges 中查找轮廓
    // RETR_EXTERNAL 表示只提取最外层的轮廓
    // CHAIN_APPROX_SIMPLE 表示对轮廓进行简化,只保留轮廓的端点
    findContours(edges, contours, RETR_EXTERNAL, CHAIN_APPROX_SIMPLE);

    // 初始化最近轮廓的中心为一个无效的点(-1,-1)
    Point closestContourCenter(-1, -1);

那接下来就是找到并画出圆心

for (const auto& contour : contours)
    {
        Moments contourMoments = moments(contour);
        if (contourMoments.m00!= 0.0)
        {
            // 根据矩计算轮廓的中心位置(质心)
            Point contourCenter(contourMoments.m10 / contourMoments.m00, contourMoments.m01 / contourMoments.m00);

            // 在图像上标记质心
            circle(img, contourCenter, 5, Scalar(0, 0, 255), -1);
			cout << "质心坐标: (" << contourCenter.x << ", " << contourCenter.y << ")" << endl;
        }
    }

image-20241017101901658

这样子我们就成功找到了需要射击的点


屏幕捕获

我们需要引用一个系统库

#include <windows.h>

以及这个函数

Mat captureScreen()
{
    // 获取显示器的大小
    int screenWidth = GetSystemMetrics(SM_CXSCREEN);
    int screenHeight = GetSystemMetrics(SM_CYSCREEN);

    // 创建设备描述表
    HDC hSrcDC = GetDC(0); // 获取整个桌面的设备上下文
    HDC hMemDC = CreateCompatibleDC(hSrcDC); // 创建与桌面兼容的设备上下文

    // 创建位图对象,并准备将其存储在内存中
    HBITMAP hBitmap = CreateCompatibleBitmap(hSrcDC, screenWidth, screenHeight);
    HBITMAP hOldBitmap = (HBITMAP)SelectObject(hMemDC, hBitmap);

    // 将屏幕复制到位图中
    BitBlt(hMemDC, 0, 0, screenWidth, screenHeight, hSrcDC, 0, 0, SRCCOPY);

    // 从位图中获取像素数据
    Mat screen(screenHeight, screenWidth, CV_8UC4);
    BYTE* data = (BYTE*)screen.data;
    GetBitmapBits(hBitmap, screenWidth * screenHeight * 4, data);

    // 清理
    SelectObject(hMemDC, hOldBitmap);
    DeleteObject(hBitmap);
    DeleteDC(hMemDC);
    ReleaseDC(0, hSrcDC);

    return screen;
}

然后两行代码就可以实现了

int main()
{
    namedWindow("screen capture", WINDOW_NORMAL);

    while (true)
    {
        Mat screen = captureScreen();

        // 显示捕获的屏幕图像
        imshow("screen capture", screen);

        waitKey(100);
    }

    return 0;
}

image-20241017102422698


打靶子!

其实打靶子就是移动鼠标然后点击

我们需要一些函数来移动鼠标和点击,这里同样还是windows.h库

大家直接用就好

POINT GetMouseCurPoint()
{
    POINT mypoint;
    GetCursorPos(&mypoint);//获取鼠标当前所在位置
    return mypoint;

}

void MouseLeftDown()//鼠标左键按下 
{
    INPUT  Input = { 0 };
    Input.type = INPUT_MOUSE;
    Input.mi.dwFlags = MOUSEEVENTF_LEFTDOWN;
    SendInput(1, &Input, sizeof(INPUT));
}

void MouseLeftUp()//鼠标左键放开 
{
    INPUT  Input = { 0 };
    Input.type = INPUT_MOUSE;
    Input.mi.dwFlags = MOUSEEVENTF_LEFTUP;
    SendInput(1, &Input, sizeof(INPUT));
}

void MouseMove(int x, int y)//鼠标移动到指定位置 
{
    double fScreenWidth = ::GetSystemMetrics(SM_CXSCREEN) - 1;//获取屏幕分辨率宽度 
    double fScreenHeight = ::GetSystemMetrics(SM_CYSCREEN) - 1;//获取屏幕分辨率高度 
    double fx = x * (65535.0f / fScreenWidth);
    double fy = y * (65535.0f / fScreenHeight);
    //printf("fScreenWidth %lf , fScreenHeight %lf, fx %lf, fy %lf \n", fScreenWidth, fScreenHeight, fx, fy);

    INPUT  Input = { 0 };
    Input.type = INPUT_MOUSE;
    Input.mi.dwFlags = MOUSEEVENTF_MOVE | MOUSEEVENTF_ABSOLUTE;
    Input.mi.dx = fx;
    Input.mi.dy = fy;
    SendInput(1, &Input, sizeof(INPUT));
}

有了这些库就能很轻松的操控鼠标点击

但是常玩fps的大家都知道,准星移动是滑动的,起始点都是屏幕中心,目标点是靶子的位置

所以我也是写了一个滑动鼠标的函数,这里我们需要用到PID来控制鼠标轨迹

从网上随便找一份PID类

class PIDController {
public:
    PIDController(double Kp, double Ki, double Kd)
        : Kp(Kp), Ki(Ki), Kd(Kd), lastError(0), integral(0) {}

    // Update the position based on error between current and target
    void updatePosition(POINT& current, const POINT& target, double dt) {
        int errorX = target.x - current.x;
        int errorY = target.y - current.y;

        // Proportional term
        double pTermX = Kp * errorX;
        double pTermY = Kp * errorY;

        // Integral term (not used in this simple example)
        integral += errorX + errorY; // This is a simplification
        double iTermX = Ki * integral * dt;
        double iTermY = Ki * integral * dt;

        // Derivative term (not used in this simple example)
        double dTermX = Kd * (errorX - lastError);
        double dTermY = Kd * (errorY - lastError);

        // Update position
        current.x += pTermX; // + iTermX + dTermX;
        current.y += pTermY; // + iTermY + dTermY;

        // Save error for next derivative calculation
        lastError = errorX + errorY;
    }

private:
    double Kp, Ki, Kd;
    double lastError;
    double integral;
};

然后就是无聊的调参了,这里已经帮大家调教好了

void moveFromTo(POINT start, POINT target, double dt) {
    PIDController pid(0.1, 0, 0); // Kp, Ki, Kd values
    POINT current = start;

    while (abs(current.x - target.x) > 10 || abs(current.y - target.y) > 10) 
    {
        pid.updatePosition(current, target, dt);
        std::cout << "Current Position: (" << current.x << ", " << current.y << ")" << std::endl;
        MouseMove(current.x, current.y);
        printf("1\n");
    }
}

这里有个小细节,因为是PID控制的鼠标,有些时候可能会震荡时间过长,我的解决方法就是扩大目标范围,在范围内直接点击就好

那么综上所述,我们就只剩最后一件事情了

设计打靶顺序可以很好的提高效率,我随便写了一份最简单的,大家要是有好想法可以在评论区分享

我的想法是找到离屏幕中心点距离最近的点优先考虑,这样子的坏处是计算量比较大,可能耗时很多,最终也是只能打到十七万分

    // 获取图像的宽度
    int width = screen.cols;
    // 获取图像的高度
    int height = screen.rows;
    // 初始化最小距离为最大的双精度浮点数
    double mindistance = DBL_MAX;
    // 初始化最近轮廓的中心为一个无效的点(-1,-1)
    Point closestContourCenter(-1, -1);

    // 遍历找到的所有轮廓
    for (const auto& contour : contours)
    {
        // 计算当前轮廓的矩
        Moments contourMoments = moments(contour);
        // 检查矩中的零阶矩是否为非零值,如果为零则说明轮廓不存在或无效
        if (contourMoments.m00!= 0.0)
        {
            // 根据矩计算轮廓的中心位置
            // m10/m00 为 x 坐标,m01/m00 为 y 坐标
            Point contourCenter(contourMoments.m10 / contourMoments.m00, contourMoments.m01 / contourMoments.m00);
            // 计算当前轮廓中心与图像中心(假设图像中心为(960,540))的水平距离差
            double dx = 960 - contourCenter.x;
            // 计算当前轮廓中心与图像中心(假设图像中心为(960,540))的垂直距离差
            double dy = 540 - contourCenter.y;
            // 计算当前轮廓中心与图像中心的欧氏距离
            double distance = sqrt(dx * dx + dy * dy);
            // 如果当前轮廓中心与图像中心的距离小于之前找到的最小距离
            if (distance < mindistance)
            {
                // 更新最小距离
                mindistance = distance;
                // 更新最近轮廓中心
                closestContourCenter = contourCenter;
            }
        }
    }

最终整合

到此为止以及完成了所有的准备工作,我们可以识别到目标,可以移动可以点击

但是还是有一个问题就是程序退出的问题,如果程序一直运行将一直拉扯你的鼠标让你无法点击只能重启(别问我是怎么知道的)

这里有一个好办法就是使用时间库来控制程序执行时间,打靶一局一分钟,可以把程序设置成一分五秒自动结束

// 记录开始时间
auto starttime = std::chrono::steady_clock::now();
// 设置结束时间为开始时间加上 4 分钟
auto endtime = starttime + std::chrono::minutes(1) + std::chrono::seconds(5);
// 获取当前时间
auto currenttime = std::chrono::steady_clock::now();
// 如果当前时间大于等于结束时间
if (currenttime >= endtime)
{
    // 跳出循环
    break;
}

记得引用

#include <chrono>

好的那么接下来就是开始实战

我们需要先打开游戏,打开游戏的窗口模式,监测窗口小图标的位置,在程序一开始的时候先点击游戏最小化的图标,然后点击继续游戏,然后开始打靶子

那屏幕上的位置需要大家用提到的函数来自行测试

// 打印最近的轮廓中心
        if (closestContourCenter.x!= -1 && closestContourCenter.y!= -1)
        {
            std::cout << "closest contour center: (" << closestContourCenter.x << ", " << closestContourCenter.y << ")" << std::endl;
            start = GetMouseCurPoint();
            POINT aid = { closestContourCenter.x, closestContourCenter.y };
            // 从一个点移动到另一个点
            moveFromTo(target4, aid, 0.001);
            MouseLeftDown();
            MouseLeftUp();
        }

这一步是打靶子的逻辑,大家可以试试看,0.001是调好的dt,不用再管他


完整代码

main.cpp(前面需要小小改动)

#define _CRT_SECURE_NO_WARNINGS 1
#include "sa.h"
using namespace cv;


int main()
{
    // 获取鼠标当前位置并输出
    POINT mousePos = GetMouseCurPoint();
    std::cout << "mouse position: " << mousePos.x << "," << mousePos.y << std::endl;

    // 记录起始位置
    POINT start = GetMouseCurPoint();
    POINT target = { 0, 0 };
    // 模拟鼠标左键按下和抬起
    MouseLeftDown();
    MouseLeftUp();
    POINT target1 = { 178, 345 };
    POINT target2 = { 1780, 50 };
    POINT target4 = { 960, 531 };

    std::cout << "起始位置: (" << start.x << ", " << start.y << ")" << std::endl;

    // 从起始位置移动到目标位置
    // 这两次点击一次是打开游戏,一次是点击继续游戏(根据自己的屏幕来修改)
    moveFromTo(start, target2, 0.05);
    MouseLeftDown();
    MouseLeftUp();

    start = GetMouseCurPoint();
    moveFromTo(start, target1, 0.05);
    MouseLeftDown();
    MouseLeftUp();

    Sleep(1000);
    MouseLeftDown();
    MouseLeftUp();

    // 初始化一个窗口,窗口大小可调整
    namedWindow("screen capture", WINDOW_NORMAL);

    // 记录开始时间
    auto starttime = steady_clock::now();
    // 设置结束时间为开始时间加上 4 分钟
    auto endtime = starttime + minutes(4);

    while (true)
    {
        // 捕获屏幕
        Mat screen = captureScreen();

        // 转换为 BGR 格式以便后续处理
        cvtColor(screen, screen, COLOR_BGRA2BGR);

        // 将 BGR 颜色空间转换为 HSV 颜色空间
        Mat hsv;
        cvtColor(screen, hsv, COLOR_BGR2HSV);

        // 定义蓝色在 HSV 空间的下界和上界
        Scalar lower_blue(82, 199, 118);
        Scalar upper_blue(97, 255, 255);

        // 创建一个掩膜,只保留蓝色区域
        Mat mask;
        inRange(hsv, lower_blue, upper_blue, mask);

        // 使用掩膜提取蓝色区域,对掩膜进行边缘检测
        Mat edges;
        Canny(mask, edges, 80, 255);

        // 显示边缘检测结果
        imshow("edges", edges);

        // 查找轮廓
        vector<vector<Point>> contours;
        findContours(edges, contours, RETR_EXTERNAL, CHAIN_APPROX_SIMPLE);

        int width = screen.cols;
        int height = screen.rows;
        double mindistance = DBL_MAX;
        Point closestContourCenter(-1, -1);

        for (const auto& contour : contours)
        {
            // 计算轮廓的矩
            Moments contourMoments = moments(contour);
            if (contourMoments.m00 != 0.0)
            {
                // 根据矩计算轮廓中心
                Point contourCenter(contourMoments.m10 / contourMoments.m00, contourMoments.m01 / contourMoments.m00);
                // 手动计算两点之间的距离
                double dx = 960 - contourCenter.x;
                double dy = 540 - contourCenter.y;
                double distance = sqrt(dx * dx + dy * dy);
                if (distance < mindistance)
                {
                    // 更新最小距离和最近轮廓中心
                    mindistance = distance;
                    closestContourCenter = contourCenter;
                }
            }
        }

        // 创建一个图像用于绘制轮廓和最近的轮廓中心
        Mat drawing = Mat::zeros(screen.size(), screen.type());
        // 在图像上绘制轮廓
        drawContours(drawing, contours, -1, Scalar(0, 255, 0), 2, LINE_8, vector<Vec4i>(), 0, Point());

        if (closestContourCenter.x != -1 && closestContourCenter.y != -1)
        {
            // 在图像上绘制最近的轮廓中心和图像中心
            circle(drawing, closestContourCenter, 5, Scalar(0, 0, 255), -1);
            circle(drawing, Point(960, 540), 5, Scalar(255, 0, 0), -1);
        }

        // 显示捕获到的屏幕图像
        //imshow("screen capture", drawing);

        // 打印最近的轮廓中心
        if (closestContourCenter.x != -1 && closestContourCenter.y != -1)
        {
            std::cout << "closest contour center: (" << closestContourCenter.x << ", " << closestContourCenter.y << ")" << std::endl;
            start = GetMouseCurPoint();
            POINT aid = { closestContourCenter.x, closestContourCenter.y };
            // 从一个点移动到另一个点
            moveFromTo(target4, aid, 0.001);
            MouseLeftDown();
            MouseLeftUp();
        }

        // 检查是否有按键按下
        if (waitKey(1) >= 0) break;

        MouseLeftDown();
        MouseLeftUp();
        Sleep(100);

        // 获取当前时间
        auto currenttime = steady_clock::now();
        if (currenttime >= endtime)
        {
            break;
        }
    }

    // 释放资源
    destroyAllWindows();

    return 0;
}

函数文件demo.cpp

#define _CRT_SECURE_NO_WARNINGS 1
#include "sa.h"

POINT GetMouseCurPoint()
{
    POINT mypoint;
    GetCursorPos(&mypoint);//获取鼠标当前所在位置
    return mypoint;

}

void MouseMove(int x, int y)//鼠标移动到指定位置 
{
    double fScreenWidth = ::GetSystemMetrics(SM_CXSCREEN) - 1;//获取屏幕分辨率宽度 
    double fScreenHeight = ::GetSystemMetrics(SM_CYSCREEN) - 1;//获取屏幕分辨率高度 
    double fx = x * (65535.0f / fScreenWidth);
    double fy = y * (65535.0f / fScreenHeight);
    //printf("fScreenWidth %lf , fScreenHeight %lf, fx %lf, fy %lf \n", fScreenWidth, fScreenHeight, fx, fy);

    INPUT  Input = { 0 };
    Input.type = INPUT_MOUSE;
    Input.mi.dwFlags = MOUSEEVENTF_MOVE | MOUSEEVENTF_ABSOLUTE;
    Input.mi.dx = fx;
    Input.mi.dy = fy;
    SendInput(1, &Input, sizeof(INPUT));
}

void moveFromTo(POINT start, POINT target, double dt) {
    PIDController pid(0.1, 0, 0); // Kp, Ki, Kd values
    POINT current = start;

    while (abs(current.x - target.x) > 10 || abs(current.y - target.y) > 10) 
    {
        pid.updatePosition(current, target, dt);
        std::cout << "Current Position: (" << current.x << ", " << current.y << ")" << std::endl;
        MouseMove(current.x, current.y);
        printf("1\n");
    }
}

void MouseLeftDown()//鼠标左键按下 
{
    INPUT  Input = { 0 };
    Input.type = INPUT_MOUSE;
    Input.mi.dwFlags = MOUSEEVENTF_LEFTDOWN;
    SendInput(1, &Input, sizeof(INPUT));
}

void MouseLeftUp()//鼠标左键放开 
{
    INPUT  Input = { 0 };
    Input.type = INPUT_MOUSE;
    Input.mi.dwFlags = MOUSEEVENTF_LEFTUP;
    SendInput(1, &Input, sizeof(INPUT));
}

void MouseRightDown()//鼠标右键按下 
{
    INPUT  Input = { 0 };
    Input.type = INPUT_MOUSE;
    Input.mi.dwFlags = MOUSEEVENTF_RIGHTDOWN;
    SendInput(1, &Input, sizeof(INPUT));
}

void MouseRightUp()//鼠标右键放开 
{
    INPUT  Input = { 0 };
    Input.type = INPUT_MOUSE;
    Input.mi.dwFlags = MOUSEEVENTF_RIGHTUP;
    SendInput(1, &Input, sizeof(INPUT));
}

Mat captureScreen()
{
    // 获取显示器的大小
    int screenWidth = GetSystemMetrics(SM_CXSCREEN);
    int screenHeight = GetSystemMetrics(SM_CYSCREEN);

    // 创建设备描述表
    HDC hSrcDC = GetDC(0); // 获取整个桌面的设备上下文
    HDC hMemDC = CreateCompatibleDC(hSrcDC); // 创建与桌面兼容的设备上下文

    // 创建位图对象,并准备将其存储在内存中
    HBITMAP hBitmap = CreateCompatibleBitmap(hSrcDC, screenWidth, screenHeight);
    HBITMAP hOldBitmap = (HBITMAP)SelectObject(hMemDC, hBitmap);

    // 将屏幕复制到位图中
    BitBlt(hMemDC, 0, 0, screenWidth, screenHeight, hSrcDC, 0, 0, SRCCOPY);

    // 从位图中获取像素数据
    Mat screen(screenHeight, screenWidth, CV_8UC4);
    BYTE* data = (BYTE*)screen.data;
    GetBitmapBits(hBitmap, screenWidth * screenHeight * 4, data);

    // 清理
    SelectObject(hMemDC, hOldBitmap);
    DeleteObject(hBitmap);
    DeleteDC(hMemDC);
    ReleaseDC(0, hSrcDC);

    return screen;
}

库文件sa.h

#pragma once

#include <iostream>
#include <windows.h>
#include <cmath>
#include <opencv2/opencv.hpp>
#include <chrono>

using namespace std;
using namespace cv;
using namespace std::chrono; // 使用chrono命名空间

POINT GetMouseCurPoint();
void MouseMove(int x, int y);//鼠标移动到指定位置 
void moveFromTo(POINT start, POINT target, double dt);
void MouseLeftDown();
void MouseLeftUp();//鼠标左键放开 
void MouseRightDown();//鼠标右键按下
void MouseRightUp();//鼠标右键放开
Mat captureScreen();

class PIDController {
public:
    PIDController(double Kp, double Ki, double Kd)
        : Kp(Kp), Ki(Ki), Kd(Kd), lastError(0), integral(0) {}

    // Update the position based on error between current and target
    void updatePosition(POINT& current, const POINT& target, double dt) {
        int errorX = target.x - current.x;
        int errorY = target.y - current.y;

        // Proportional term
        double pTermX = Kp * errorX;
        double pTermY = Kp * errorY;

        // Integral term (not used in this simple example)
        integral += errorX + errorY; // This is a simplification
        double iTermX = Ki * integral * dt;
        double iTermY = Ki * integral * dt;

        // Derivative term (not used in this simple example)
        double dTermX = Kd * (errorX - lastError);
        double dTermY = Kd * (errorY - lastError);

        // Update position
        current.x += pTermX; // + iTermX + dTermX;
        current.y += pTermY; // + iTermY + dTermY;

        // Save error for next derivative calculation
        lastError = errorX + errorY;
    }

private:
    double Kp, Ki, Kd;
    double lastError;
    double integral;
};

程序是突发奇想的产物,还有很多的不足,大家如果有更好的建议可以在评论区指出

谢谢大家的观看!!!


http://www.kler.cn/a/354645.html

相关文章:

  • 常见的开源网络操作系统
  • 日语IT用语笔记
  • 无网络时自动切换备用网络环境
  • 数据结构(1~10)
  • maven的简单介绍
  • 【MySQL】深度学习数据库开发技术:使用CC++语言访问数据库
  • 永恒之蓝漏洞
  • 毕业设计之—基于ManTra-Net的图像篡改检测方法研究与应用实现
  • uni-app关闭底部系统导航栏的控制按钮BUG
  • SSTI模板注入+CTF实例
  • TS学习——type与interface的异同点
  • c++基础知识复习(1)
  • pdf删除几个页面怎么操作?PDF页面删除的快捷方法
  • Unity UGUI SuperScrollView介绍
  • ETL技术在金蝶云星空与旺店通WMS集成中的应用
  • 期货交易程序化,哪些API可供选择及如何使用?
  • taro小程序如何全局监听路由变化?
  • 基于SpringBoot+Vue+uniapp微信小程序的乡村政务服务系统的详细设计和实现(源码+lw+部署文档+讲解等)
  • AI学习指南深度学习篇-生成对抗网络的变体及扩展
  • 02 go语言(golang) - 包和模块
  • 什么是 SQL 命令?SQL 如何工作?
  • Maven基于构建阶段分析多余的依赖
  • Linux 下find常用命令整理(更新ing)
  • HCIP-HarmonyOS Application Developer 习题(十二)
  • 阿加犀构建开发者生态的全链路赋能之旅
  • QGIS--DEMTO3D