OpenCV5-图像几何变换

OpenCV5-图像几何变换

    • 1.图像连接
    • 2.图像尺寸变换
    • 3.图像翻转变换
    • 4.图像仿射变换
    • 5.图像透视变换
    • 6.极坐标变换


1.图像连接

图像连接是指将相同高度或者宽度的图像连接在一起。

vconcat()函数用于实现图像或矩阵的上下连接:上下连接的矩阵应该具有相同的列数,并且具有相同的数据类型和通道数。

void vconcat(const Mat* src,  // Mat矩阵类型数组size_t nsrc,     // 数组数目OutputArray dst);// 连接后的Mat类矩阵void vconcat(InputArray src1,  // 第一个需要连接的Mat类矩阵InputArray src2,  // 第二个需要连接的Mat类矩阵OutputArray dst); // 连接后的Mat类矩阵

hconcat()函数用于实现图像或者矩阵左右连接:左右连接的矩阵应该具有相同的行数,并且具有相同的数据类型和通道数。参数意义同vconcat函数。

void hconcat(const Mat* src, size_t nsrc, OutputArray dst);void hconcat(InputArray src1, InputArray src2, OutputArray dst);

下面是示例代码:

#include <opencv2/core/utils/logger.hpp>
#include <opencv2/opencv.hpp>using namespace cv;
using namespace std;int main()
{cout << "OpenCV Version: " << CV_VERSION << endl;utils::logging::setLogLevel(utils::logging::LOG_LEVEL_SILENT);//矩阵数组的横竖连接Mat matArray[] = { Mat(1, 2, CV_32FC1, cv::Scalar(1)),Mat(1, 2, CV_32FC1, cv::Scalar(2)) };Mat vout, hout;vconcat(matArray, 2, vout);cout << "图像数组竖向连接:" << endl << vout << endl;hconcat(matArray, 2, hout);cout << "图像数组横向连接:" << endl << hout << endl;//矩阵的横竖拼接Mat A = (cv::Mat_<float>(2, 2) << 1, 7, 2, 8);Mat B = (cv::Mat_<float>(2, 2) << 4, 10, 5, 11);Mat vC, hC;vconcat(A, B, vC);cout << "多个图像竖向连接:" << endl << vC << endl;hconcat(A, B, hC);cout << "多个图像横向连接:" << endl << hC << endl;//读取4个子图像,00表示左上角、01表示右上角、10表示左下角、11表示右下角Mat img00 = imread("lena00.png");Mat img01 = imread("lena01.png");Mat img10 = imread("lena10.png");Mat img11 = imread("lena11.png");if (img00.empty() || img01.empty() || img10.empty() || img11.empty()){cout << "请确认图像文件名称是否正确" << endl;return -1;}//显示4个子图像imshow("img00", img00);imshow("img01", img01);imshow("img10", img10);imshow("img11", img11);//图像连接Mat img, img0, img1;//图像横向连接hconcat(img00, img01, img0);hconcat(img10, img11, img1);//横向连接结果再进行竖向连接vconcat(img0, img1, img);//显示连接图像的结果imshow("img0", img0);imshow("img1", img1);imshow("img", img);int k = waitKey(0); // Wait for a keystroke in the windowreturn 0;
}

2.图像尺寸变换

图像的尺寸变换实际上就是改变图像的长和宽,实现图像的缩放。resize()函数用于将图像修改成指定尺寸。

void resize(InputArray src,  // 输入图像OutputArray dst, // 输出图像Size dsize,      // 输出图像尺寸double fx = 0,   double fy = 0,int interpolation = INTER_LINEAR );

fx和fy分别为水平轴和垂直轴的比例因子,如果需要将水平轴或者垂直轴变为原来的两倍,则参数需赋值2。

dsize用来调整输出图像大小,fx和fy一起也可以调整输出图像,因此两类参数在实际使用时只需要使用一类,当根据两个参数计算出来的输出图像尺寸不一致时,以dsize设置的图像尺寸为准。这两类参数的关系式:

dsize = Size(round(fx * src.cols), round(fy * src.rows))

interpolation参数是选择图像插值方法的标志。如果要缩小图像,那么通常使用IMTER_AREA标志会有更好的效果。而在放大图像时,采用INTER_CUBIC和INTER_LINEAR标志通常会有比较好的效果,这两个标志,前者计算速度较慢但效果较好,后者计算速度较快效果也比较理想。

标志参数简记作用
INTER_NEAREST0最邻近插值法
INTER_LINEAR1双线性插值法
INTER_CUBIC2双三次插值
INTER_AREA3使用像素区域关系重新采样,首选用于图像缩小,图像放大时效果与INTER_NEAREST相似
INTER_LANCZOS44Lanczos插值法
INTER_LINEAR_EXACT5位精确双线性插值法
INTER_MAX6用掩码进行插值

下面示例代码以灰度图像形式读入一幅图像,利用INTER_AREA将图像缩小,分别用INTER_CUBIC、INTER_NEAREST、INTER_LINEAR三种方法将图像放大到相同的尺寸:

#include <opencv2/core/utils/logger.hpp>
#include <opencv2/opencv.hpp>using namespace cv;
using namespace std;int main()
{cout << "OpenCV Version: " << CV_VERSION << endl;utils::logging::setLogLevel(utils::logging::LOG_LEVEL_SILENT);Mat gray = imread("lena.png", IMREAD_GRAYSCALE);if (gray.empty()){cout << "请确认图像文件名称是否正确" << endl;return -1;}Mat smallImg, bigImg0, bigImg1, bigImg2;resize(gray, smallImg, Size(15, 15), 0, 0, INTER_AREA);  //先将图像缩小resize(smallImg, bigImg0, Size(30, 30), 0, 0, INTER_NEAREST);  //最近邻差值resize(smallImg, bigImg1, Size(30, 30), 0, 0, INTER_LINEAR);  //双线性差值resize(smallImg, bigImg2, Size(30, 30), 0, 0, INTER_CUBIC);  //双三次差值namedWindow("smallImg", WINDOW_NORMAL);  //图像尺寸太小,一定要设置可以调节窗口大小标志imshow("smallImg", smallImg);namedWindow("bigImg0", WINDOW_NORMAL);imshow("bigImg0", bigImg0);namedWindow("bigImg1", WINDOW_NORMAL);imshow("bigImg1", bigImg1);namedWindow("bigImg2", WINDOW_NORMAL);imshow("bigImg2", bigImg2);int k = waitKey(0); // Wait for a keystroke in the windowreturn 0;
}

3.图像翻转变换

OpenCV中提供了flip函数用于将图像翻转:

void flip(InputArray src,  // 输入图像OutputArray dst, // 输出图像int flipCode);   // 翻转方式标志

flipCode数值大于0表示绕y轴翻转,等于0表示绕x翻转,小于0表示绕两个轴翻转。

#include <opencv2/core/utils/logger.hpp>
#include <opencv2/opencv.hpp>using namespace cv;
using namespace std;int main()
{cout << "OpenCV Version: " << CV_VERSION << endl;utils::logging::setLogLevel(utils::logging::LOG_LEVEL_SILENT);Mat img = imread("lena.png");if (img.empty()){cout << "请确认图像文件名称是否正确" << endl;return -1;}Mat img_x, img_y, img_xy;flip(img, img_x, 0);  //沿x轴对称flip(img, img_y, 1);  //沿y轴对称flip(img, img_xy, -1);  //先x轴对称,再y轴对称imshow("img", img);imshow("img_x", img_x);imshow("img_y", img_y);imshow("img_xy", img_xy);int k = waitKey(0); // Wait for a keystroke in the windowreturn 0;
}

4.图像仿射变换

实现图像的旋转需要确定旋转角度和旋转中心(确定了旋转矩阵),然后进行仿射变换实现旋转。OpenCV提供了getRotationMatrix2D函数用于计算旋转矩阵,提供了warpAffine函数用于实现图像的仿射变换。

inline
Mat getRotationMatrix2D(Point2f center, // 旋转中心double angle,   // 旋转角度,度为单位,正值为逆时针旋转double scale);  // 两个轴的比例因子,实现旋转过程中的图像缩放,不缩放则为1

该函数返回图像旋转矩阵,返回值为Mat类是一个2x3的矩阵。
该函数生成的旋转矩阵与旋转角度和旋转中心的关系:
R o t a t i o n = [ α β ( 1 − α ) ∗ c e n t e r . x − β ∗ c e n t e r . y − β α β ∗ c e n t e r . x + ( 1 − α ) ∗ c e n t e r . y ] Rotation = \begin{bmatrix} \alpha & \beta & (1-\alpha)*center.x - \beta*center.y \\ -\beta & \alpha & \beta*center.x+(1-\alpha)*center.y \end{bmatrix} Rotation=[αββα(1α)center.xβcenter.yβcenter.x+(1α)center.y]
其中
α = s c a l e ∗ c o s ( a n g l e ) β = s c a l e ∗ s i n ( a n g l e ) \alpha = scale * cos(angle) \\ \beta = scale * sin(angle) α=scalecos(angle)β=scalesin(angle)

// 根据旋转矩阵,进行仿射变换
void warpAffine(InputArray src,  // 输入图像OutputArray dst, // 仿射变换后的输出图像InputArray M,    // 2x3的变换矩阵Size dsize,      // 输出图像尺寸int flags = INTER_LINEAR,  // 插值方法标志int borderMode = BORDER_CONSTANT,      // 像素边界外推方法标志const Scalar& borderValue = Scalar()); // 填充边界使用的值,默认为0/*
M:前面求取的旋转矩阵
dsize:输出图像尺寸
flags:插值方法标志,同尺寸变换resize时的插值方法标志,并且多了两个可以和前面几个标志一起使用
WARP_FILL_OUTLIERS:填充所有输出图像的像素,如果部分像素落在输入图像的边界外,则他们的值设定为fillval
WARP_INVERSE_MAP:设置为M输出图像到输入图像的反变换。
*/
enum InterpolationFlags{/** nearest neighbor interpolation */INTER_NEAREST        = 0,/** bilinear interpolation */INTER_LINEAR         = 1,// .../** flag, fills all of the destination image pixels. If some of them correspond to outliers in thesource image, they are set to zero */WARP_FILL_OUTLIERS   = 8,/** flag, inverse transformationFor example, #linearPolar or #logPolar transforms:- flag is __not__ set: \f$dst( \rho , \phi ) = src(x,y)\f$- flag is set: \f$dst(x,y) = src( \rho , \phi )\f$*/WARP_INVERSE_MAP     = 16
};/*
第六个参数borderMode为像素边界外推标志:
BORDER_CONSTANT:用特定值填充,如用i填充:iiiiii|abcdefgh|iiiiiii
BORDER_REPLICATE:两端复制填充,如两端用a和h填充:aaaaaa|abcdefgh|hhhhhhh
BORDER_REFLECT:倒序填充
BORDER_WRAP:正序填充
BORDER_REFLECT_101:不包含边界值的倒序填充
BORDER_TRANSPARENT:随即填充
BORDER_REFLECT101、BORDER_DEFAULT:同BORDER_REFLECT_101
BORDER_ISOLATED:不关心感兴趣区域之外的部分
*/
enum BorderTypes {BORDER_CONSTANT    = 0, //!< `iiiiii|abcdefgh|iiiiiii`  with some specified `i`BORDER_REPLICATE   = 1, //!< `aaaaaa|abcdefgh|hhhhhhh`BORDER_REFLECT     = 2, //!< `fedcba|abcdefgh|hgfedcb`BORDER_WRAP        = 3, //!< `cdefgh|abcdefgh|abcdefg`BORDER_REFLECT_101 = 4, //!< `gfedcb|abcdefgh|gfedcba`BORDER_TRANSPARENT = 5, //!< `uvwxyz|abcdefgh|ijklmno`BORDER_REFLECT101  = BORDER_REFLECT_101, //!< same as BORDER_REFLECT_101BORDER_DEFAULT     = BORDER_REFLECT_101, //!< same as BORDER_REFLECT_101BORDER_ISOLATED    = 16 //!< do not look outside of ROI
};

仿射变换就是图像的旋转、平移和缩放的统称,可以表示为线性变换和平移变换的叠加。仿射变换的数学表示是先乘以一个线性变换矩阵再加上一个平移向量,其中线性变换矩阵为2x2,平移向量为2x1。
M = [ A B ] = [ a 00 a 01 b 00 a 10 a 11 b 10 ] M = \begin{bmatrix} A & B \end{bmatrix} = \begin{bmatrix} a_{00} & a_{01} & b_{00} \\ a_{10} & a_{11} & b_{10} \end{bmatrix} M=[AB]=[a00a10a01a11b00b10]
根据线性变换矩阵A和平移矩阵B,以及图像像素值 [ x y ] \begin{bmatrix}x \\ y\end{bmatrix} [xy],放射变换的数学原理: T = A [ x y ] + B T = A\begin{bmatrix}x \\ y\end{bmatrix} + B T=A[xy]+B,其中T为变换后的像素值。

仿射变换又称为三点变换。如果知道变换前后两幅图像中3个像素点坐标的对应关系,就可以求得仿射变换中的变换矩阵M。OpenCV中也提供了利用3个对应像素点来确定变换矩阵M的函数getAffineTransform():

Mat getAffineTransform(const Point2f src[], // 源图像中的3个像素点坐标const Point2f dst[]  // 目标图像中的3个像素点坐标
);
/*
函数返回值是2x3变换矩阵M。
*/
#include <opencv2/core/utils/logger.hpp>
#include <opencv2/opencv.hpp>using namespace cv;
using namespace std;int main()
{cout << "OpenCV Version: " << CV_VERSION << endl;utils::logging::setLogLevel(utils::logging::LOG_LEVEL_SILENT);Mat img = imread("lena.png");if (img.empty()){cout << "请确认图像文件名称是否正确" << endl;return -1;}Mat rotation0, rotation1, img_warp0, img_warp1;double angle = 30;  //设置图像旋转的角度Size dst_size(img.rows, img.cols);  //设置输出图像的尺寸Point2f center(img.rows / 2.0, img.cols / 2.0);  //设置图像的旋转中心rotation0 = getRotationMatrix2D(center, angle, 1);  //计算放射变换矩阵warpAffine(img, img_warp0, rotation0, dst_size);  //进行仿射变换imshow("img_warp0", img_warp0);//根据定义的三个点进行仿射变换Point2f src_points[3];Point2f dst_points[3];src_points[0] = Point2f(0, 0);  //原始图像中的三个点src_points[1] = Point2f(0, (float)(img.cols - 1));src_points[2] = Point2f((float)(img.rows - 1), (float)(img.cols - 1));dst_points[0] = Point2f((float)(img.rows) * 0.11, (float)(img.cols) * 0.20);  //放射变换后图像中的三个点dst_points[1] = Point2f((float)(img.rows) * 0.15, (float)(img.cols) * 0.70);dst_points[2] = Point2f((float)(img.rows) * 0.81, (float)(img.cols) * 0.85);rotation1 = getAffineTransform(src_points, dst_points);  //根据对应点求取仿射变换矩阵warpAffine(img, img_warp1, rotation1, dst_size);  //进行仿射变换imshow("img_warp1", img_warp1);int k = waitKey(0); // Wait for a keystroke in the windowreturn 0;
}

5.图像透视变换

透视变换是按照物体成像投影规律进行变换,即将物体重新投影到新的成像平面。通常通过透视变换实现对物体图像的校正。在透视变换中,透视前的图像和透视后的图像之间的变换关系可以用3x3的变换矩阵表示,该矩阵可以通过两幅图像中4个对应点的坐标求取,OpenCV中提供了根据4个对应点求取变换矩阵的getPerspectiveTransform函数和进行透视变换的warpPerspective函数:

Mat getPerspectiveTransform(const Point2f src[], // 原图像中的4个像素坐标const Point2f dst[], // 目标图像中的4个像素坐标int solveMethod = DECOMP_LU // 根据对应点坐标计算透视变换矩阵的选择标志
);/*
DECOMP_LU:最佳主轴元素的高斯消元法
DECOMP_SVD:奇异值分解方法
DECOMP_EIG:特征值分解
DECOMP_CHOLESKY:Cholesky分解法
DECOMP_NORMAL:使用正规方程公式,可以和其他标志一起使用。
*/enum DecompTypes {/** Gaussian elimination with the optimal pivot element chosen. */DECOMP_LU       = 0,/** singular value decomposition (SVD) method; the system can be over-defined and/or the matrixsrc1 can be singular */DECOMP_SVD      = 1,/** eigenvalue decomposition; the matrix src1 must be symmetrical */DECOMP_EIG      = 2,/** Cholesky \f$LL^T\f$ factorization; the matrix src1 must be symmetrical and positivelydefined */DECOMP_CHOLESKY:Cholesky分解法= 3,/** QR factorization; the system can be over-defined and/or the matrix src1 can be singular */DECOMP_QR       = 4,/** while all the previous flags are mutually exclusive, this flag can be used together withany of the previous; it means that the normal equations\f$\texttt{src1}^T\cdot\texttt{src1}\cdot\texttt{dst}=\texttt{src1}^T\texttt{src2}\f$ aresolved instead of the original system\f$\texttt{src1}\cdot\texttt{dst}=\texttt{src2}\f$ */DECOMP_NORMAL   = 16
};
void warpPerspective(InputArray src,  // 输入图像OutputArray dst, // 透视变换后图像InputArray M,    // 透视变换矩阵3x3Size dsize,      // 输出图像的尺寸int flags = INTER_LINEAR, // 插值方法标志int borderMode = BORDER_CONSTANT, // 像素边界外推方法标志const Scalar& borderValue = Scalar()); // 填充边界使用的数值,默认情况下为0

warpPerspective函数所有参数含义同仿射变换函数warpAffine()。

下面例子给出了将相机视线不垂直于二维码平面拍摄的图像经过透视变换变成相机视线垂直于二维码平面拍摄的图像。本程序中实现通过ImageWatch插件查看了拍摄图像二维码4个角点的坐标,并希望透视变换后二维码可以填满全部的图像,因此在程序中手动输入4个对应点的像素坐标。但在实际的工程中,二维码的角点坐标可以通过角点检测的方式获取。

#include <opencv2/core/utils/logger.hpp>
#include <opencv2/opencv.hpp>using namespace cv;
using namespace std;int main()
{cout << "OpenCV Version: " << CV_VERSION << endl;utils::logging::setLogLevel(utils::logging::LOG_LEVEL_SILENT);Mat img = imread("noobcvqr.png");if (img.empty()){cout << "请确认图像文件名称是否正确" << endl;return -1;}Point2f src_points[4];Point2f dst_points[4];//通过Image Watch查看的二维码四个角点坐标src_points[0] = Point2f(94.0, 374.0);src_points[1] = Point2f(507.0, 380.0);src_points[2] = Point2f(1.0, 623.0);src_points[3] = Point2f(627.0, 627.0);//期望透视变换后二维码四个角点的坐标dst_points[0] = Point2f(0.0, 0.0);dst_points[1] = Point2f(627.0, 0.0);dst_points[2] = Point2f(0.0, 627.0);dst_points[3] = Point2f(627.0, 627.0);Mat rotation, img_warp;rotation = getPerspectiveTransform(src_points, dst_points);  //计算透视变换矩阵warpPerspective(img, img_warp, rotation, img.size());  //透视变换投影imshow("img", img);imshow("img_warp", img_warp);int k = waitKey(0); // Wait for a keystroke in the windowreturn 0;
}

6.极坐标变换

极坐标变换就是将图像在直角坐标系与极坐标系中相互变换。

可以将一个圆形图像变换成一个矩形图像,圆形图案边缘上的文字经过极坐标变换后可以垂直地排列在新图像的边缘。

OpenCV中提供了warpPolar函数用于实现图像的极坐标变换:

void warpPolar(InputArray src,   // 源图像,可以是灰度图或彩色图OutputArray dst,  // 极坐标变换后输出图像Size dsize,       // 目标图像大小Point2f center,   // 极坐标变换时,极坐标的原点坐标double maxRadius, // 变换时的边界园半径,它也决定了逆变换时的比例参数int flags); // 插值方法与极坐标映射方法标志 |连接/*
该函数实现了图像极坐标和半对数极坐标变换。
center:极坐标变换时极坐标远点在原始图像中的位置,同样适用于逆变换中。
flags:插值方法如INTER_NEAREST,极坐标映射方法如下
WARP_POLAR_LINEAR:极坐标变换
WARP_POLAR_LOG:半对数极坐标变换
WARP_INVERSE_MAP:逆变换
*/
enum WarpPolarMode
{WARP_POLAR_LINEAR = 0, ///< Remaps an image to/from polar space.WARP_POLAR_LOG = 256   ///< Remaps an image to/from semilog-polar space.
};

下面的例子给出了对仪表盘图像进行极坐标正变换和逆变换的示例程序:程序中选择表盘的中心作为极坐标原点

#include <opencv2/core/utils/logger.hpp>
#include <opencv2/opencv.hpp>using namespace cv;
using namespace std;int main()
{cout << "OpenCV Version: " << CV_VERSION << endl;utils::logging::setLogLevel(utils::logging::LOG_LEVEL_SILENT);Mat img = imread("dial.png");if (!img.data){cout << "请检查图像文件名称是否输入正确" << endl;return -1;}Mat img1, img2;Point2f center = Point2f(img.cols / 2, img.rows / 2);  //极坐标在图像中的原点//正极坐标变换warpPolar(img, img1, Size(300, 600), center, center.x, INTER_LINEAR + WARP_POLAR_LINEAR);//逆极坐标变换warpPolar(img1, img2, Size(img.rows, img.cols), center, center.x, INTER_LINEAR + WARP_POLAR_LINEAR + WARP_INVERSE_MAP);imshow("原表盘图", img);imshow("表盘极坐标变换结果", img1);imshow("逆变换结果", img2);int k = waitKey(0); // Wait for a keystroke in the windowreturn 0;
}

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

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

相关文章

智安网络|边缘计算与分布式存储:数字化时代的新趋势

随着数字化时代的到来&#xff0c;数据的产生和存储需求呈现爆炸式增长&#xff0c;传统的集中式存储架构已经无法满足大规模数据存储和处理的需求。分布式存储系统应运而生&#xff0c;成为应对数据存储和处理挑战的解决方案。然而&#xff0c;技术的发展不会止步于此&#xf…

点云直通滤波(附python open3d 代码)

定义了一个名为pass_through的函数,用于根据指定的过滤值名称(filter_value_name)对点云数据进行过滤。函数接受四个参数:点云数据(pcd),最小限制(limit_min)和最大限制(limit_max)以及过滤值名称(filter_value_name)。 在函数内部,首先将点云数据转换为NumPy数…

基于SSM的医用物理学实验考核系统设计与实现

末尾获取源码 开发语言&#xff1a;Java Java开发工具&#xff1a;JDK1.8 后端框架&#xff1a;SSM 前端&#xff1a;采用JSP技术开发 数据库&#xff1a;MySQL5.7和Navicat管理工具结合 服务器&#xff1a;Tomcat8.5 开发软件&#xff1a;IDEA / Eclipse 是否Maven项目&#x…

算法题:柠檬水找零(典型的贪心算法问题)

这道题就是纯贪心算法题&#xff0c;遍历每个顾客&#xff0c;先把钱收了&#xff0c;如果是10块钱就判断手里头有没有5元用于找零&#xff1b;如果是20块钱&#xff0c;先判断是不是有10元5元&#xff0c;如果没有就再判断是否有3个5元。没有的话就直接返回 False。(完整题目附…

CSP模拟51联测13 B.狗

CSP模拟51联测13 B.狗 文章目录 CSP模拟51联测13 B.狗题目大意题目描述输入格式输出格式样例样例 1inputoutput 思路 题目大意 题目描述 小G养了很多狗。 小G一共有 n n n\times n nn 条狗&#xff0c;在一个矩阵上。小G想让狗狗交朋友&#xff0c;一条狗狗最多只能交一个…

vue2项目中使用element ui组件库的table,制作表格,改表格的背景颜色为透明的

el-table背景颜色变成透明_el-table背景透明_讲礼貌的博客-CSDN博客 之前是白色的&#xff0c;现在变透明了&#xff0c;背景颜色是蓝色

加密市场波动:地缘政治与美股走弱引发不确定性!

伴随着国庆假期的结束&#xff0c;多日波动率维持低位的加密市场也似乎开始苏醒。近期多次突破28000美元未果的比特币&#xff0c;于9日15:00开始从27800美元附近下跌&#xff0c;最低跌至27260美元&#xff0c;同期以太坊也至1550美元左右&#xff0c;创近半个月来新低。 Coin…

vue接入高德地图获取经纬度

&#x1f90d;step1:高德地图开放平台&#xff0c;根据指引注册成为高德开放平台开发者&#xff0c;并申请 web 平台&#xff08;JS API&#xff09;的 key 和安全密钥; &#x1f90d;step2:在html引入安全密钥&#xff08;获取经纬度用&#xff0c;不然会报错&#xff09; <…

基于SpringBoot的大型商场应急预案管理系统

目录 前言 一、技术栈 二、系统功能介绍 员工信息管理 预案信息管理 预案类型统计 事件类型管理 三、核心代码 1、登录模块 2、文件上传模块 3、代码封装 前言 随着信息技术在管理上越来越深入而广泛的应用&#xff0c;管理信息系统的实施在技术上已逐步成熟。本文介绍…

手撕各种排序

> 作者简介&#xff1a;დ旧言~&#xff0c;目前大一&#xff0c;现在学习Java&#xff0c;c&#xff0c;c&#xff0c;Python等 > 座右铭&#xff1a;松树千年终是朽&#xff0c;槿花一日自为荣。 > 目标&#xff1a;掌握每种排序的方法&#xff0c;理解每种排序利弊…

springboot-配置文件优先级

官方文档 https://docs.spring.io/spring-boot/docs/2.7.16/reference/htmlsingle/#features.external-config Spring Boot允许外部化配置&#xff0c;这样就可以在不同的环境中使用相同的应用程序代码。您可以使用各种外部配置源&#xff0c;包括Java属性文件、YAML文件、环境…

(js)封装年月日获取方法,页面根据type判断显示当前年,年月,日期

(js)封装年月日获取方法&#xff0c;页面根据type判断显示当前年&#xff0c;年月&#xff0c;日期 项目src——>utils——>index.js // 获取当前年&#xff0c;年月&#xff0c;日期&#xff0c;type, export function getYearMonth(type) {var date new Date()var ye…

ROS 工作空间及功能包

ROS工作空间&#xff08;workspace&#xff09;是一个存放工程开发相关文件的文件夹。 1. 什么是ROS的工作空间 使用ROS实现机器人开发的主要手段是写代码&#xff0c;这些代码文件存放的空间就是工作空间。 工作空间&#xff08;workspace&#xff09;是一个用于存放工程开发…

Spring Data Redis使用方式

1.导入Spring Data Redis的maven坐标 pom.xml <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-redis</artifactId> </dependency> 2. 配置Redis数据源 2.1application.yml文件…

【工具】SSH端口转发管理器,专门管理SSH Port Forwarding

转载请注明出处&#xff1a;小锋学长生活大爆炸[xfxuezhang.cn] 开源代码看这里&#xff1a;http://xfxuezhang.cn/index.php/archives/1151/ 背景介绍 有时候需要用到ssh的端口转发功能。目前来说&#xff0c;要么是cmd里手敲指令&#xff0c;但每次敲也太麻烦了&#xff1b;或…

JVM命令行监控工具

JVM命令行监控工具 概述 性能诊断是软件工程师在日常工作中需要经常面对和解决的问题&#xff0c;在用户体验至上的今天&#xff0c;解决好应用的性能问题能带来非常大的收益。 Java作为最流行的编程语言之一&#xff0c;其应用性能诊断一直受到业界广泛关注&#xff0c;可能…

游戏缺少dll文件用什么修复?教你多种dll文件修复方法搞定!

在玩游戏的时候&#xff0c;有时候会遇到一些dll文件缺失的问题&#xff0c;导致游戏无法正常运行。这对于广大游戏爱好者来说无疑是一种巨大的打击。但是不要担心&#xff0c;我们总会有方法来解决这个问题。本文将详细介绍几种解决方法&#xff0c;帮助你轻松修复游戏缺少dll…

Vue3的props需要注意的地方(简写与监视属性)

在工作中踩了props的坑&#xff0c;总结一下&#xff1a; 1.props是可以在模板语法中简写的。就好比&#xff0c;toRefs了一下state。我们没必要在模板语法中加上props.xxx&#xff1b; 2.有时我们需要监视props的内容&#xff0c;可以用到监视属性watch。我们可以先复习一下…

技术干货:解密最受欢迎的开源 Serverless 框架弹性技术实现

作者&#xff1a;元毅 Knative 是一款基于 Kubernetes 的开源 Serverless 应用编排框架&#xff0c;其目标是制定云原生、跨平台的 Serverless 应用编排标准。Knative 主要功能包括基于请求的自动弹性、缩容到 0、多版本管理、基于流量的灰度发布以及事件驱动等。 弹性是 Ser…

强化学习问题(一)--- 输入conda activate base无法激活虚拟环境

起因&#xff1a;在Pycharm中&#xff0c;基于python新建了环境&#xff0c;输入conda activate base后突然无法激活虚拟环境了 解决&#xff1a; 1.找到Anaconda Prompt右击进入文件所在位置 2. 右击进入属性 3. 复制cmd.exe开始到最后的路径 4. 粘贴到pycharm-settings-…