Opencv 基本操作七 提取两幅图像的重叠区域

news2024/11/17 15:59:42

使用opencv提取连个图像的重叠区域,其本质就是提取两个图像的特征点,然后对两个图像的特征点进行匹配,根据匹配的特征点计算出透视变换矩阵H,然后根据H即可提取出两个图像的重叠区域。这里要注意的是,普通的opencv库没有包含opencv-contrib,无法使用xfeatures2d.hpp里面的SURF、SIFT算法提取图像的特征点。故此,需要自行编译opencv(将opencv-contrib加到动态库中),或者下载别人编译好的opencv。

1、包含opencv-contrib的opencv

win10下vs2019编译的opencv470:https://download.csdn.net/download/a486259/87355761
win10下vs2019编译的opencv453+cuda:
https://download.csdn.net/download/a486259/81328226

2、核心步骤

提取图像重叠区域的关键步骤为,加载图像、提特征点、特征点匹配、计算透视变换矩阵、计算重叠区域、提取重叠区域。

2.1 加载图像

    stringstream fmt1, fmt2;
    fmt1 << "IMG_20221231_160033.jpg";
    Mat left = imread(fmt1.str());//左侧:图片路径
    fmt2 << "IMG_20221231_160037.jpg";
    Mat right = imread(fmt2.str());//右侧:图片路径
    Size msize =  { 512, 512 };//left.size();//
    resize(left, left, msize);
    resize(right, right, msize);

2.2 提取图像特征点

创建surf对象,提取两个图像的特征点和特征描述符

    Ptr<SURF>surf;   //可以容纳800个特征点
    surf = SURF::create(800);//参数 查找的海森矩阵 create 海森矩阵阀值
    vector<KeyPoint>key1, key2;//特征点
    Mat c, d;//特征点描述符
    //提取特征点
    surf->detectAndCompute(left, Mat(), key2, d);
    surf->detectAndCompute(right, Mat(), key1, c);

2.2 提取图像特征点

使用暴力匹配器匹配两个图像中的特征描述符,匹配结果存入 vector<DMatch>中,然后按照比例提取一定匹配的点

    BFMatcher matcher; //暴力匹配器
    vector<DMatch>matches;//DMatch 点和点之间的关系
    //使用暴力匹配器匹配特征点,找到存来
    matcher.match(d, c, matches);

    //3、筛选特征点
    //排序 从小到大
    sort(matches.begin(), matches.end());
    //保留最优的特征点对象
    vector<DMatch>good_matches;//最优
    //设置比例
    int ptrPoint = std::max(50, (int)(matches.size() * 0.15));
    for (int i = 0; i < ptrPoint; i++)
    {
        good_matches.push_back(matches[i]);
    }

    //4、最佳匹配的特征点连成线
    Mat outimg;
    drawMatches(left, key2, right, key1, good_matches, outimg,
        Scalar::all(-1), Scalar::all(-1),
        vector<char>(), DrawMatchesFlags::NOT_DRAW_SINGLE_POINTS);

2.3 计算透视变换矩阵

提取vector<DMatch>中已匹配的特征点存入imagepoint1, imagepoint2中计算透视变换矩阵

//5、计算透视变换矩阵
    //提取已匹配的特征点
    vector<Point2f>imagepoint1, imagepoint2;
    for (int i = 0; i < good_matches.size(); i++)
    {
        imagepoint1.push_back(key1[good_matches[i].trainIdx].pt);
        imagepoint2.push_back(key2[good_matches[i].queryIdx].pt);
    }
    //透视转换
    Mat H = findHomography(imagepoint1, imagepoint2, cv::RANSAC);
    cout << "H:::" << H << endl;

2.4 计算透视角点

根据H矩阵计算出透视角点,并将透视检点绘制为mask

    //定义四个角点坐标。
    Point2f obj_corners[4] = { cv::Point(0,0),cv::Point(left.cols, 0), cv::Point(left.cols, left.rows), cv::Point(0, left.rows) };
    Point scene_corners2[4];  //在srcImage1上画线
    //获取透视变化的角点
    for (int i = 0; i < 4; i++)
    {
        double x = obj_corners[i].x;
        double y = obj_corners[i].y;

        double Z = 1. / (H.at<double>(2, 0) * x + H.at<double>(2, 1) * y + H.at<double>(2, 2));
        double X = (H.at<double>(0, 0) * x + H.at<double>(0, 1) * y + H.at<double>(0, 2)) * Z;
        double Y = (H.at<double>(1, 0) * x + H.at<double>(1, 1) * y + H.at<double>(1, 2)) * Z;
        scene_corners2[i] = cv::Point(cvRound(X), cvRound(Y));
    }

    //使用drawContours将透视检点绘制为mask(超出图像大小的区域不可见)
    Mat roi = Mat::zeros(left.size(), CV_8UC1);
    vector<vector<Point>> contour;
    vector<Point> pts;
    pts.push_back(scene_corners2[0]);
    pts.push_back(scene_corners2[1]);
    pts.push_back(scene_corners2[2]);
    pts.push_back(scene_corners2[3]);
    contour.push_back(pts);
    drawContours(roi, contour, 0, Scalar::all(255), -1);

2.5 截取重叠区域

截取两个图像的重叠区域并绘图展示

//截取第一个图的重复区域
    Mat dstimg1;
    left.copyTo(dstimg1, roi);
    Mat roi_left_by = get_by(roi);
    //截取第二个图的重复区域
    Mat dstimg2 = Mat::zeros(left.size(), CV_8UC3);
    warpPerspective(right, dstimg2, H, left.size());
    //对roi进行逆变换,得到在右图的重叠区域
    Mat roi_right;
    warpPerspective(roi, roi_right, H, left.size(), WARP_INVERSE_MAP);
    Mat roi_right_by = get_by(roi_right);

    imshow("outimg", outimg);
    imshows("ss", {left,dstimg1,roi_left_by+ left ,right,dstimg2,roi_right_by+ right }, 3);
    waitKey();

3、完整示例

3.1 完整代码

其中Mattool.hpp的内容在3.2章

#include <iostream>
#include <opencv2/opencv.hpp>
#include <opencv2/highgui.hpp>
#include <opencv2/xfeatures2d.hpp>
#include <opencv2/calib3d.hpp>
#include <opencv2/imgproc.hpp>
#include "Matutils.hpp"

using namespace std;
using namespace cv;
using namespace cv::xfeatures2d;

Mat sobel_mat(Mat srcImage) {
    Mat grad_x, grad_y;
    Mat abs_grad_x, abs_grad_y;
    //    计算x方向的梯度
    Sobel(srcImage, grad_x, CV_16S, 1, 0, 3, 1, 0, BORDER_DEFAULT);
    convertScaleAbs(grad_x, abs_grad_x);

    //    计算y方向的梯度
    Sobel(srcImage, grad_y, CV_16S, 0, 1, 3, 1, 0, BORDER_DEFAULT);
    convertScaleAbs(grad_y, abs_grad_y);

    //    合并梯度
    Mat dstImage;
    addWeighted(abs_grad_x, 0.5, abs_grad_y, 0.5, 0, dstImage);
    return dstImage;
}
Mat line2red(Mat line) {
    vector<cv::Mat> mv;
    Mat red_line;
    Mat zeros = Mat::zeros(line.size(), CV_8UC1);
    //mv.push_back(zeros);
    //mv.push_back(zeros);
    mv.push_back(line);
    mv.push_back(line);
    mv.push_back(line);
    //将vector内的多个mat合并为一个多通道mat
    cv::merge(mv, red_line);
    return red_line;
}
Mat get_by(Mat roi) {
    Mat roi_left, roi_erode, roi_by;
    Mat element = getStructuringElement(MORPH_RECT, Size(7, 7));
    roi_left = pad_mat(roi);
    morphologyEx(roi_left, roi_erode, MORPH_ERODE, element);
    roi_by = roi_left - roi_erode;
    roi_by = rm_pad(roi_by);
    //cvtColor(roi_by, roi_by, color);
    return line2red(roi_by);
}
void test_img() {
    stringstream fmt1, fmt2;
    fmt1 << "IMG_20221231_160033.jpg";
    Mat left = imread(fmt1.str());//左侧:图片路径
    fmt2 << "IMG_20221231_160037.jpg";
    Mat right = imread(fmt2.str());//右侧:图片路径
    Size msize =  { 512, 512 };//left.size();//
    resize(left, left, msize);
    resize(right, right, msize);

    //1、创建SURF对象
    Ptr<SURF>surf;   //可以容纳800个特征点
    surf = SURF::create(800);//参数 查找的海森矩阵 create 海森矩阵阀值
    vector<KeyPoint>key1, key2;//特征点
    Mat c, d;//特征点描述符
    //提取特征点
    surf->detectAndCompute(left, Mat(), key2, d);
    surf->detectAndCompute(right, Mat(), key1, c);

    //2、进行特征匹配
    BFMatcher matcher; //暴力匹配器
    //特征点对比,保存下来
    vector<DMatch>matches;//DMatch 点和点之间的关系
    //使用暴力匹配器匹配特征点,找到存来
    matcher.match(d, c, matches);

    //3、筛选特征点
    //排序 从小到大
    sort(matches.begin(), matches.end());
    //保留最优的特征点对象
    vector<DMatch>good_matches;//最优
    //设置比例
    int ptrPoint = std::max(50, (int)(matches.size() * 0.15));
    for (int i = 0; i < ptrPoint; i++)
    {
        good_matches.push_back(matches[i]);
    }

    //4、最佳匹配的特征点连成线
    Mat outimg;
    drawMatches(left, key2, right, key1, good_matches, outimg,
        Scalar::all(-1), Scalar::all(-1),
        vector<char>(), DrawMatchesFlags::NOT_DRAW_SINGLE_POINTS);


    //5、进行透视变化
    //特征点配对
    vector<Point2f>imagepoint1, imagepoint2;
    for (int i = 0; i < good_matches.size(); i++)
    {
        imagepoint1.push_back(key1[good_matches[i].trainIdx].pt);
        imagepoint2.push_back(key2[good_matches[i].queryIdx].pt);
    }
    //透视转换
    Mat H = findHomography(imagepoint1, imagepoint2, cv::RANSAC);
    cout << "H:::" << H << endl;

    //根据H矩阵计算出透视角点
    //定义四个角点坐标。
    Point2f obj_corners[4] = { cv::Point(0,0),cv::Point(left.cols, 0), cv::Point(left.cols, left.rows), cv::Point(0, left.rows) };
    Point scene_corners2[4];  //在srcImage1上画线
    //获取透视变化的角点
    for (int i = 0; i < 4; i++)
    {
        double x = obj_corners[i].x;
        double y = obj_corners[i].y;

        double Z = 1. / (H.at<double>(2, 0) * x + H.at<double>(2, 1) * y + H.at<double>(2, 2));
        double X = (H.at<double>(0, 0) * x + H.at<double>(0, 1) * y + H.at<double>(0, 2)) * Z;
        double Y = (H.at<double>(1, 0) * x + H.at<double>(1, 1) * y + H.at<double>(1, 2)) * Z;
        scene_corners2[i] = cv::Point(cvRound(X), cvRound(Y));
    }

    //使用drawContours将透视检点绘制为mask(超出图像大小的区域不可见)
    Mat roi = Mat::zeros(left.size(), CV_8UC1);
    vector<vector<Point>> contour;
    vector<Point> pts;
    pts.push_back(scene_corners2[0]);
    pts.push_back(scene_corners2[1]);
    pts.push_back(scene_corners2[2]);
    pts.push_back(scene_corners2[3]);
    contour.push_back(pts);
    drawContours(roi, contour, 0, Scalar::all(255), -1);

    //截取第一个图的重复区域
    Mat dstimg1;
    left.copyTo(dstimg1, roi);
    Mat roi_left_by = get_by(roi);


    //截取第二个图的重复区域
    Mat dstimg2 = Mat::zeros(left.size(), CV_8UC3);
    warpPerspective(right, dstimg2, H, left.size());
    //对roi进行逆变换,得到在右图的重叠区域
    Mat roi_right;
    warpPerspective(roi, roi_right, H, left.size(), WARP_INVERSE_MAP);
    Mat roi_right_by = get_by(roi_right);

    imshow("outimg", outimg);
    imshows("ss", {left,dstimg1,roi_left_by+ left ,right,dstimg2,roi_right_by+ right }, 3);
    waitKey();
}
int main(int argc, char* argv[])
{
    test_img();
    return 0;
}

3.2 Matutils.hpp

这是进行图像处理中通用的函数库

#ifndef __Matutils__
#define __Matutils__
#pragma once
#include <iostream>
#include <opencv2/opencv.hpp>
#include <opencv2/highgui.hpp>
#include <opencv2/imgproc.hpp>

#include <iostream>  
#include <vector>
#include <io.h>
#include <stdlib.h>
#include <iostream>
#include <string>
using namespace std;
using namespace cv;
//对图像填充黑边
inline  Mat pad_mat(Mat img,int pad=10) {
    Mat imgWindow = Mat::zeros(img.cols + 2*pad , img.rows + 2 * pad, img.type());
    img.copyTo(imgWindow(Rect({ pad ,pad }, img.size())));
    return imgWindow;
}
//移除图像填充的黑边
inline  Mat rm_pad(Mat img, int pad = 10) {
    Size size = { img.cols - 2 * pad , img.rows - 2 * pad };
    return img(Rect({ pad ,pad }, size));
}
//图像拼接
inline  void multipleImage(vector<Mat> imgVector, Mat& dst, int imgCols, int MAX_PIXEL = 300)
{
    //两列图像间的空白区域
    int pad = 10;
    int imgNum = imgVector.size();
    //选择图片最大的一边 将最大的边按比例变为300像素
    Size imgOriSize = imgVector[0].size();
    int imgMaxPixel = max(imgOriSize.height, imgOriSize.width);
    //获取最大像素变为MAX_PIXEL的比例因子
    double prop = imgMaxPixel < MAX_PIXEL ? (double)imgMaxPixel / MAX_PIXEL : MAX_PIXEL / (double)imgMaxPixel;
    Size imgStdSize(imgOriSize.width * prop, imgOriSize.height * prop); //窗口显示的标准图像的Size

    Mat imgStd; //标准图片
    Point2i location(0, 0); //坐标点(从0,0开始)
    //Mat imgWindow(imgStdSize.height * ((imgNum - 1) / imgCols + 1), imgStdSize.width * imgCols+ pad * imgCols-pad, imgVector[0].type());
    int imgRows = (imgNum - 1) / imgCols + 1;
    Mat imgWindow = Mat::zeros(imgStdSize.height * imgRows + pad * imgRows - pad, imgStdSize.width * imgCols + pad * imgCols - pad, imgVector[0].type());
    for (int i = 0; i < imgNum; i++)
    {
        location.x = (i % imgCols) * (imgStdSize.width + pad);
        location.y = (i / imgCols) * imgStdSize.height;
        resize(imgVector[i], imgStd, imgStdSize, prop, prop, INTER_LINEAR); //设置为标准大小

        //将imgStd复制到imgWindow的指定区域中
        imgStd.copyTo(imgWindow(Rect(location, imgStdSize)));
    }
    dst = imgWindow;
}
//多图显示
inline  void imshows(string title, vector<Mat> imgVector,  int imgCols = -1) {
    Mat dst;
    if (imgCols == -1) {
        imgCols = imgVector.size();
    }
    multipleImage(imgVector, dst, imgCols);
    namedWindow(title);
    imshow(title, dst);
    imwrite(title + ".png", dst);
}
//删除小面积的连通域
inline  Mat deleteMinWhiteArea(Mat src, int min_area) {
    Mat labels, stats, centroids, img_color, grayImg;
    //1、连通域信息统计
    int nccomps = connectedComponentsWithStats(
        src, //二值图像
        labels,
        stats,
        centroids
    );

    //2、连通域状态区分
    //为每一个连通域初始化颜色表
    vector<Vec3b> colors(nccomps);
    colors[0] = Vec3b(0, 0, 0); // background pixels remain black.
    for (int i = 1; i < nccomps; i++)
    {
        colors[i] = Vec3b(rand() % 256, rand() % 256, rand() % 256);
        //面积阈值筛选
        if ((stats.at<int>(i, CC_STAT_AREA) < min_area))
        {
            //如果连通域面积不合格则置黑
            colors[i] = Vec3b(0, 0, 0);
        }
    }
    //3、连通域删除
    //按照label值,对不同的连通域进行着色
    img_color = Mat::zeros(src.size(), CV_8UC3);
    for (int y = 0; y < img_color.rows; y++)
    {
        int* labels_p = labels.ptr<int>(y);//使用行指针,加速运算
        Vec3b* img_color_p = img_color.ptr<Vec3b>(y);//使用行指针,加速运算
        for (int x = 0; x < img_color.cols; x++)
        {
            int label = labels_p[x];//取出label值
            CV_Assert(0 <= label && label <= nccomps);
            img_color_p[x] = colors[label];//设置颜色
        }
    }
    //return img_color;
    //如果是需要二值结果则将img_color进行二值化
    cvtColor(img_color, grayImg, COLOR_BGR2GRAY);
    threshold(grayImg, grayImg, 1, 255, THRESH_BINARY);
    return grayImg;
}
//删除图形中小面积的黑色孔洞
inline  Mat deleteMinBlackArea(Mat src, int min_area) {
    Mat inv = 255 - src;//颜色取反
    Mat res = deleteMinWhiteArea(inv, min_area);
    return 255 - res;//颜色取反
}
//找图中topk个连通域
inline  bool mypairsort(pair<int, int> i, pair<int, int> j) { return (i.second > j.second); }
inline  Mat findTopKArea(Mat srcImage, int topk)
{
    Mat temp;
    Mat labels;
    srcImage.copyTo(temp);

    //1. 标记连通域
    int n_comps = connectedComponents(temp, labels, 4, CV_16U);
    vector<pair<int, int>> histogram_of_labels;
    for (int i = 0; i < n_comps; i++)//初始化labels的个数为0
    {
        histogram_of_labels.push_back({ i,0 });
    }

    int rows = labels.rows;
    int cols = labels.cols;
    for (int row = 0; row < rows; row++) //计算每个labels的个数--即连通域的面积
    {
        for (int col = 0; col < cols; col++)
        {
            histogram_of_labels.at(labels.at<unsigned short>(row, col)).second += 1;
        }
    }
    //histogram_of_labels.at(0).second = 0; //将背景的labels个数设置为0

    //2.对连通域进行排序
    std::sort(histogram_of_labels.begin(), histogram_of_labels.end(), mypairsort);
    //3. 取前k个连通域的labels id
    vector<int> select_labels;
    for (int i = 0; i < topk; i++)
    {
        if (histogram_of_labels[i].first == 0) {
            topk += 1;
            //如果碰到背景,则跳过,且topk+1
        }
        else {
            select_labels.push_back(histogram_of_labels[i].first);
        }
    }

    //3. 将label id在select_labels的连通域标记为255,并将其他连通域置0
    for (int row = 0; row < rows; row++)
    {
        for (int col = 0; col < cols; col++)
        {
            int now_label_id = labels.at<unsigned short>(row, col);
            if (std::count(select_labels.begin(), select_labels.end(), now_label_id)) {
                labels.at<unsigned short>(row, col) = 255;
            }
            else {
                labels.at<unsigned short>(row, col) = 0;
            }
        }
    }

    //4. 将图像更改为CV_8U格式
    labels.convertTo(labels, CV_8U);
    return labels;
}
//获取脊线
inline Mat get_ridge_line(Mat dst, int ksize = 3) {
    Mat skeleton, result, open_dst;
    Mat kernel = getStructuringElement(MORPH_CROSS, Size(ksize, ksize));
    skeleton = Mat::zeros(dst.rows, dst.cols, dst.type());
    while (true) {
        if (sum(dst)[0] == 0) {
            break;
        }
        morphologyEx(dst, dst, MORPH_ERODE, kernel);//消除毛刺,删除部分连通域
        morphologyEx(dst, open_dst, MORPH_OPEN, kernel);
        result = dst - open_dst;
        skeleton = skeleton + result;
    }
    return skeleton;
}
#endif

3.3 测试效果

两个图像的特征点匹配关系
在这里插入图片描述
算法提取出的重叠区域(中间列),第一例为两个原图,最后一列为重叠区域在原图中的效果。
在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Kali Linux断网攻击

前言 ARP攻击就是通过伪造IP地址和MAC地址实现ARP欺骗&#xff0c;能够在网络中产生大量的ARP通信量使网络阻塞&#xff0c;攻击者只要持续不断的发出伪造的ARP响应包就能更改目标主机ARP缓存中的IP-MAC条目&#xff0c;造成网络中断或中间人攻击。 一句话解释&#xff1a; 断…

2022年总结 | 从初二学生到算法作者的蜕变之路

目录 一年的创作历程 我和 CSDN 在编程竞赛的合作 About CBC 技术社区的发展 夜跑奇遇 About 博客之星 新年致谢 元旦祝福 一年的创作历程 2022年&#xff0c;这一年对于我来说是十分重要的一年。在这一年里&#xff0c;我作为一名初二在校学生&#xff0c;在CSDN上发布…

2003-2021年高铁站开通时间

2003-2021年高铁站开通时间数据 1、时间&#xff1a;2003-2021年 2、指标&#xff1a;高铁站名称、开通时间、所在省份、所在城市、所属线路名称、以及相关备注 3、指标说明&#xff1a; 高速铁路&#xff0c;简称高铁&#xff0c;是指设计标准等级高、可供列车安全高速行驶…

需要提醒你关于 golang 中 map 使用的几点注意事项

日常的开发工作中&#xff0c;map 这个数据结构相信大家并不陌生&#xff0c;在 golang 里面&#xff0c;当然也有 map 这种类型 关于 map 的使用&#xff0c;还是有蛮多注意事项的&#xff0c;如果不清楚&#xff0c;这些事项&#xff0c;关键时候可能会踩坑&#xff0c;我们…

【攻防世界】Web warmup

知识点讲解 这一题主要是利用了include的特性 如果include的文件名中含有“/”&#xff0c;那么它会识别其为一个带目录的文件&#xff0c;只有最后一个“/”后的字符串对应的文件会被包含&#xff0c;而前面的字符串都只是在指定目录 意思是&#xff0c;如果我们的payload是这…

ArcGIS基础实验操作100例--实验36创建特征线约束TIN

本实验专栏参考自汤国安教授《地理信息系统基础实验操作100例》一书 实验平台&#xff1a;ArcGIS 10.6 实验数据&#xff1a;请访问实验1&#xff08;传送门&#xff09; 高级编辑篇--实验36 创建特征线约束TIN 目录 一、实验背景 二、实验数据 三、实验步骤 &#xff08;1…

搜狗PR权重在线查询,搜狗PR值查询方法

什么是搜狗PR权重&#xff1f; 搜狗权重(SogouRank)是由搜狗搜索引擎官方发布网页评级数据&#xff0c;搜狗权重是搜狗衡量网页重要性的指标&#xff0c;是机器根据搜狗评级算法自动计算出来的。搜狗权重值从1至10不等&#xff0c;网页评级越高&#xff0c;该网页在搜索中越…

异步通信技术AJAX | AJAX乱码问题、异步与同步、手动封装一个jQuery库

目录 一&#xff1a;快速搞定AJAX&#xff08;第三篇&#xff09; 1、AJAX乱码问题 2、AJAX的异步与同步 3、AJAX代码封装 4、手动封装一个jQuery库 一&#xff1a;快速搞定AJAX&#xff08;第三篇&#xff09; 1、AJAX乱码问题 &#xff08;1&#xff09;发送ajax get 或…

2022年我的22个感悟

这些感悟&#xff0c;有些是我自己感悟出来的&#xff1b;更有一些&#xff0c;是别人说的&#xff0c;引起了我的共鸣或深思。这些感悟&#xff0c;可能没有什么实用价值&#xff0c;但能让人对这个世界多一份思考。1. 现代社会的奴隶“小工是酒店或餐馆的奴隶&#xff0c;而他…

7-10 网红点打卡攻略

一个旅游景点&#xff0c;如果被带火了的话&#xff0c;就被称为“网红点”。大家来网红点游玩&#xff0c;俗称“打卡”。在各个网红点打卡的快&#xff08;省&#xff09;乐&#xff08;钱&#xff09;方法称为“攻略”。你的任务就是从一大堆攻略中&#xff0c;找出那个能在…

Python解题 - CSDN周赛第19期 - 醉酒的狱卒

本期题目依然难度不高&#xff0c;可惜状态不佳&#xff0c;未能取得满分&#xff0c;而且解题的思路也没能做到最简&#xff0c;直到赛后才想到还可以有另外的有趣的解法。我想这本身也是比赛的乐趣之一吧&#xff0c;不识庐山真面目&#xff0c;只缘身在此山中。 第一题&…

【阶段一】Python快速入门03篇:数据结构-元组、字典与集合

本篇的思维导图: 数据结构-元组 元组的概念 元组(tuple)虽然与列表类似,但也有不同之处,元组的元素不能修改;元组使用小括号,而列表使用中括号。 新建一个元组 元组的创建比较简单,直接将一组数据元素用小括号括起来即可。

第1章 生物和生物圈

张惠怡*&#xff0c;张钊* (萧县城东初级中学&#xff0c;淮北师范大学计算机科学与技术学院&#xff0c;安徽 淮北&#xff0c;安徽 宿州) *These authors contributed to the work equllly and should be regarded as co-first authors. &#x1f31e;欢迎来到生物的世界 …

校招前端二面高频vue面试题

vue-router中如何保护路由 分析 路由保护在应用开发过程中非常重要&#xff0c;几乎每个应用都要做各种路由权限管理&#xff0c;因此相当考察使用者基本功。 体验 全局守卫&#xff1a; const router createRouter({ ... }) ​ router.beforeEach((to, from) > {// .…

【LeetCode每日一题】——263.丑数

文章目录一【题目类别】二【题目难度】三【题目编号】四【题目描述】五【题目示例】六【解题思路】七【题目提示】八【时间频度】九【代码实现】十【提交结果】一【题目类别】 数学 二【题目难度】 简单 三【题目编号】 263.丑数 四【题目描述】 丑数 就是只包含质因数 …

3GPP R17 RedCap

什么是RedCap RedCap&#xff0c;即Reduced Capability的简称&#xff0c;在早期的3GPP讨论过程中&#xff0c;它也被称为NR Light。RedCap是针对IoT场景&#xff0c;为了降低终端复杂度、成本和功耗而提出。相比NR标准版本&#xff0c;RedCap主要在如下几个方面做了简化&#…

C++语言级别的多线程

1.线程概念 好处&#xff1a;可以跨平台&#xff1a;windows / linux / mac 线程间的互斥&#xff1a;mutex / lock_quard / unique_lock 线程间的通信&#xff1a;condition_variable atomic : 原子类型 基于CAS操作的原子类型 线程安全的 sleep_for :睡眠 C语言层面调用thre…

配置Debian11服务器安装SSH,创建新用户并允许远程SSH远程登录,并禁止root用户远程SSH登录

一、在 Debian 中添加 sudo 用户 1.创建新用户 首先&#xff0c;要创建用户&#xff0c;当前用户必须是 root 用户或者 sudo 用户。 使用下面adduser 命令创建一个用户名为test的sudo用户&#xff0c;按照提示输入密码&#xff0c;使用 adduser 命令&#xff0c;还会创建用户…

(三分钟)速览传统边缘检测算子

边缘检测的传统方法&#xff1a; 图像边缘是图像最基本的特征&#xff0c;所谓边缘(Edge) 是指图像局部特性的不连续性。灰度或结构等信息的突变处称之为边缘。例如&#xff0c;灰度级的突变、颜色的突变,、纹理结构的突变等。边缘是一个区域的结束&#xff0c;也是另一个区域…

如何在星巴克连接家中Windows台式机?(安卓,iOS, Windows, macOS配合frp公网iP实现)...

zhaoolee 最近热衷于和海外热心老哥们交换硬盘中的单机游戏资源(BT下载)&#xff0c;家中有Windows台式机&#xff0c; 适合长时间挂机下载BT资源&#xff0c;zhaoolee希望能随时连接到Windows台式机新增下载任务&#xff0c;安装体积超大的主机游戏。 另外&#xff0c;公司有一…