MFC CRectTracker 类用法详解

news2025/1/11 12:37:53

      CRectTracker 类并非 Microsoft Foundation Class (MFC) 库中应用很广泛的一个类,一般教科书中很少有提到。在编程中如果需编写选择框绘制以及选择框大小调整、移动等程序时,用CRectTracker 类就会做到事半而功倍。下面详细介绍MFC CRectTracker 类。

MFC CRectTracker 类详解

        CRectTracker 类的构造函数

        CRectTracker 类的构造函数的原型如下:

CRectTracker();

CRectTracker(

LPCRECT lpSrcRect,

UINT nStyle);

参数:

       lpSrcRect 矩形对象的指针

       nStyle CRectTracker 对象的样式, 有一下几种样式:

CRectTracker::solidLine 对矩形边框使用实线。

 CRectTracker::dottedLine 对矩形边框使用虚线。

 CRectTracker::hatchedBorder 对矩形边框使用带阴影的图案。

 CRectTracker::resizeInside 调整位于矩形内的图柄的大小。

 CRectTracker::resizeOutside 调整位于矩形外的图柄的大小。

 CRectTracker::hatchInside 带阴影的图案覆盖整个矩形。

默认构造函数使用来自 lpSrcRect CRectTracker 的值初始化 对象,并将其他大小初始化为

系统默认值。 如果创建对象时没有参数,那么  m_rect 和  m_nStyle 数据成员就不会被初

始化。

        CRectTracker 类的公共方法(公有成员函数)

       CRectTracker 类的公共方法如下:

下面逐一介绍CRectTracker 类的公共方法。

        CRectTracker::AdjustRect

        CRectTracker::AdjustRect是一个虚函数,使用调整大小图柄调整跟踪矩形的大小时由框架调用。其原型如下:

virtual void AdjustRect(

int nHandle,

LPRECT lpRect);

参数:

 nHandl 所用图柄的索引

 lpRect 指向矩形当前大小的指针。(矩形的大小由其高度和宽度决定。)

CRectTracker::Draw

调用此函数以绘制矩形的外部线和内部区域。其原型如下:

void Draw(CDC* pDC) const;

参数: pDC 指向要进行绘制的设备上下文的指针。

       CRectTracker::DrawTrackerRect

      CRectTracker::DrawTrackerRect是一个虚函数,每当跟踪器的位Track 或TrackRubberBand 成员函数内部发生变化时由框架调用。其原型如下:

virtual void DrawTrackerRect(

LPCRECT lpRect,

CWnd* pWndClipTo,

CDC* pDC,

CWnd* pWnd);

       参数:

              lpRect 指向包含要绘制的矩形RECT的指针。

              pWndClipTo 指向用于剪切矩形的窗口的指针。

              pDC 指向要进行绘制的设备上下文的指针。

              pWnd 指向要在其中进行绘制的窗口的指针。

注解:默认实现调用  CDC::DrawFocusRect ,这将绘制一个虚线矩形。重写此函数以在跟踪操作期间提供不同的反馈。

        CRectTracker::GetHandleMask

       CRectTracker::GetHandleMask是一个虚函数,框架调用此成员函数来检索矩形调整大小图柄的掩码。其原型如下:

virtual UINT GetHandleMask() const;

返回值:CRectTracker 项的调整大小图柄的掩码。

一个矩形有 8 个调整大小图柄,编号为 0-7。 每个调整大小图柄由掩码中的一个位表

示;该位的值为 2^ n,其中 n 是调整大小图柄的编号。

重写此成员函数以隐藏或显示指示的调整大小图柄。

        CRectTracker::GetTrueRect

        调用此函数以检索矩形的坐标。其原型如下:

void GetTrueRect(LPRECT lpTrueRect) const;

参数:lpTrueRect 指向将包含CRectTracker 对象的设备坐标的RECT结构的指针。

CRectTracker::HitTest

调用此函数以了解用户是否已抓取一个调整大小图柄。其原型如下:

int HitTest(CPoint point) const;

参数:point 要测试的点,以设备坐标表示。

返回值:返回的值基于枚举类型  CRectTracker::TrackerHit ,可以具有以下值之一:

       CRectTracker::hitNothing -1

       CRectTracker::hitTopLeft 0

       CRectTracker::hitBottomRight 2

       CRectTracker::hitBottomLeft 3

       CRectTracker::hitTop 4

       CRectTracker::hitRight 5

       CRectTracker::hitBottom 6

       CRectTracker::hitLeft 7

       CRectTracker::hitMiddle 8

        CRectTracker::NormalizeHit

        调用此函数以转换可能反转的图柄。其原型如下:

int NormalizeHit(int nHandle) const;

参数:nHandle 用户选择的图柄。

返回值:规范化图柄的索引。

       CRectTracker::OnChangedRect

CRectTracker::OnChangedRect是一个虚函数,每当在调用Track期间跟踪器矩形发生变化时由框架调用。其原型如下:

virtual void OnChangedRect(const CRect& rectOld);

参数: rectOld 包含  CRectTracker 对象的旧设备坐标。

        CRectTracker::SetCursor

        当光标在CRectTracker 对象的区域上时,调用此函数来改变光标的形状。从处理 WM_SETCURSOR 消息(通常为  OnSetCursor )的窗口的函数内部调用此函数。其原型如下:

BOOL SetCursor(CWnd* pWnd,UINT nHitTest) const;

参数:

pWnd 指向当前包含光标的窗口。

       nHitTest 上一命中测试的结果,来自WM_SETCURSOR消息。

返回值:如果上一命中是在跟踪器矩形上,则为非零值;否则为 0。

        CRectTracker::Track

        调用此函数以显示矩形大小的用户界面。其原型如下:

BOOL Track(

CWnd* pWnd,

CPoint point,

BOOL bAllowInvert = FALSE,

CWnd* pWndClipTo = NULL);

参数:

pWnd 包含矩形的窗口对象。

Point 相对于工作区的当前鼠标位置的设备坐标。

bAllowInvert 如果为 TRUE,则可以沿 x 轴或 y 轴反转矩形;否则为 FALSE。

pWndClipTo 绘制操作将剪切到的窗口。 如果为 NULL,pWnd 将用作剪切矩形。

返回值:如果按下 ESC 键,跟踪过程会停止,跟踪器中存储的矩形不会改变,并且返回 0。 如果更改已提交,通过移动鼠标并释放鼠标左键后,会在跟踪器的矩形中记录新的位置和/或大小,并返回非零值。

        CRectTracker::TrackRubberBand

        调用此函数以执行橡皮筋选择。该函数的原型如下:

BOOL TrackRubberBand(

CWnd* pWnd,

CPoint point,

BOOL bAllowInvert = TRUE);

参数:

       pWnd 包含矩形的窗口对象。

       point  相对于工作区的当前鼠标位置的设备坐标。

       bAllowInvert 如果为 TRUE,则可以沿 x 轴或 y 轴反转矩形;否则为 FALSE。

       返回值:如果鼠标已移动且矩形不为空,则为非零值;否则为 0。

        CRectTracker 类的公共数据成员(成员变量)

        CRectTracker 类的公共数据成员(成员变量)如下:

        上面已经详细介绍了CRectTracker类,下面用一个对话框程序来演示CRectTracker类的用法。

CRectTracker类应用示例

        新建一个对话框Project来演示,CRectTracker类的用法。对话框界面如下:

为简化图像处理程序,这里会用到OpenCV,OpenCV的配置如下(这里使用的4.90,用其他低一些的版本也快):

       由于用OpenCV显示图像,显示窗口难以嵌入MFC对话框。 为便于图像显示许为对话框程序添加Mat对象转Cimage对象程序及图像显示程序。

Mat对象转Cimage对象程序的代码如下:

void CRectTrackerTestDlg::MatToCImage(Mat& src, CImage& dst)
{
	if (src.empty() || (src.type() != CV_8UC3 && src.type() != CV_8UC1)) {
		return;
	}
	// 如果CImage对象有附加图像就分离并销毁图像
	if (!dst.IsNull())
		dst.Destroy();

	//创建CImage对象附加图像,需与源图像大小类型一致
	dst.Create(src.cols, src.rows, 8 * src.channels());

	if (src.channels() == 1)
	{
		//将源位图转成八位灰度图时,CImage对象需用到颜色表,需定义一个RGBQUAD数组,并填充该数组
		RGBQUAD* colorTable = new RGBQUAD[256];
		for (int i = 0; i < 256; i++)
		{
			colorTable[i].rgbRed = i;
			colorTable[i].rgbGreen = i;
			colorTable[i].rgbBlue = i;
		}

		//设置颜色表RGB分量值
		dst.SetColorTable(0, 255, colorTable);
	}


	int rows = src.rows;
	int cols = src.cols;
	uchar channels = src.channels();
	//内存中的数据传送,注意这里是逐行传送。
	for (int i = 0; i < rows; i++)
	{
		memcpy(dst.GetPixelAddress(0, i), src.ptr<uchar>(i), cols * channels);
	}
}

图像显示程序的代码如下:

if (mImage.IsNull())
	MessageBox(L"No Image to Display!", L"系统提示", MB_ICONWARNING | MB_OK);
else
{
	CClientDC dc(this);
	mImage.BitBlt(dc.GetSafeHdc(), 0, 0, SRCCOPY);
}

         下面再为对话框按钮添加事件处理程序代码,”打开图像“按钮的事件处理程序的代码如下:

void CRectTrackerTestDlg::OnBnClickedOpen()
{
	mString = "正在进行打开图像操作...";
	mInformation.SetWindowTextW(mString);
	CFileDialog fdlg(TRUE, NULL, NULL, OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT, _T("All files(*.*)|*.*||"));
	if (fdlg.DoModal() == IDOK)
	{
		m_Path = fdlg.GetPathName();
		m_strEx = fdlg.GetFileExt();
		m_strName = fdlg.GetFileName();

		m_Path.ReleaseBuffer();
		m_strEx.ReleaseBuffer();
		m_strName.ReleaseBuffer();


		m_str = CT2A(m_Path);
		src = imread(m_str);
		dst = src;
		if (src.empty())
		{
			mString.Format(L"打开图像文件%s失败,文件格式不正确或文件已损坏!", m_strName);
			mInformation.SetWindowTextW(mString);
		}
		else
		{

			MatToCImage(dst, mImage);	//send Mat object data to CImage objiect
			//DispalyImage(mImage);		//dispaly image
			Invalidate();
			mString.Format(L"已打开:%s ", m_Path);
			mInformation.SetWindowTextW(mString);
		}
	}
	else
	{

		mString = "已取消打开图像!";
		mInformation.SetWindowTextW(mString);
	}
}

由于打开图像的代码中DisplayImage(mImage)已被屏蔽掉,需在Opaint中添加图像显示代码,如下:

        “缩放图像”按钮的事件处理程序的代码如下:

void CRectTrackerTestDlg::OnBnClickedScaleImage()
{
	UpdateData(1);
	int nWidth, nHeight;
	nWidth = (int)(fscale * dst.cols) / 4 * 4; //fscale为按钮旁边编辑框绑定变量用以设定缩放系数
	nHeight = fscale * dst.rows;
	resize(dst, dst, Size(nWidth, nHeight));
	
	MatToCImage(dst, mImage);	//send Mat object data to CImage objiect
	Invalidate();

	mString.Format(L"缩放操作已完成,现在图像大小是缩放前的 %f 倍", fscale);
	mInformation.SetWindowTextW(mString);
}

        为“选择ROI区域”按钮添加事件处理程序,“选择ROI区域”按钮的事件处理程序代码如下:

void CRectTrackerTestDlg::OnBnClickedSelectRoiArea()
{
	if (bPikFrameEanble)
	{
		bPikFrameEanble = false;
		pickRect = CRect(0, 0, 0, 0);
		pickFrame.m_rect = pickRect;
		Invalidate();
	}
	bDrawPickFrame = true;
}

仅有上面这个事件处理程序,是没法实现选择ROI区域的,首先需声明CRectTracker变量,变量还需初始化,然后在鼠标消息处理程序中添加:选择框绘制、大小调整、移动等相关代码。这里声明的CRectTracker变量及其相关变量如下:

CRectTracker变量初始化的代码如下:

        在OnLButtonDown(UINT nFlags, CPoint point)函数中加入如下代码:

void CRectTrackerTestDlg::OnLButtonDown(UINT nFlags, CPoint point)
{
	CDC* pDC = GetDC();
	if (bDrawPickFrame)
	{
		bStartDraw = true;
		bDrawPickFrame = false;
	}
	else
	{
		if (bPikFrameEanble)
		{
			pickFrame.Track(this, point);
			//pickFrame.Draw(pDC);
			pickFrame.GetTrueRect(&pickRect);
			Invalidate();
		}
	}
	ReleaseDC(pDC);
	CDialogEx::OnLButtonDown(nFlags, point);
}

      再在OnLButtonUp(UINT nFlags, CPoint point)中加入代码:代码如下:

void CRectTrackerTestDlg::OnLButtonUp(UINT nFlags, CPoint point)
{
	CDC* pDC = GetDC();
	if (bStartDraw)
	{
		bStartDraw = false;
		bPikFrameEanble = true;
		pickFrame.TrackRubberBand(this, point);
		pickFrame.Draw(pDC);
		pickFrame.GetTrueRect(&pickRect);
	}
	ReleaseDC(pDC);
	CDialogEx::OnLButtonUp(nFlags, point);
}

到此可以实现选择ROI区域了,试运行程序,结果如下:

点击打开图像按钮,选择图像文件,如下:

点击打开,结果如下:

点击“选取ROI区域”按钮,将鼠标指针移动到图像适当位置,点击鼠标左键(只能一次),然后拖动鼠标,可以看到随着鼠标指针移动有一个变化的矩形,在适当位置再次点击鼠标左键,即绘制出了一个选择框,如下:

将鼠标指针移动到选择矿内,按下鼠标左键(不松),拖动鼠标,即可移动选择框。将鼠标指针移动到选择框的图柄上,按下鼠标左键(不松),拖动鼠标,即可改变选择框的大小。但是目前鼠标的指针不会随着不同的操作变化。

        现在来改善这一不足,添加OnSetCursor(CWnd* pWnd, UINT nHitTest, UINT message)函数,在OnSetCursor(CWnd* pWnd, UINT nHitTest, UINT message)函数中加入如下代码:

BOOL CRectTrackerTestDlg::OnSetCursor(CWnd* pWnd, UINT nHitTest, UINT message)
{
	if (pWnd == this && pickFrame.SetCursor(this, nHitTest))
	{
		// 如果SetCursor成功设置了光标形状,则返回TRUE  
		return TRUE;
	}
	return CDialogEx::OnSetCursor(pWnd, nHitTest, message);
}

再次试运行,点击“选取ROI区域”按钮,绘制选取框,这时可以看到当鼠标指针移动到选取框内时,鼠标指针变成了4箭头(移动)图标,指针移动到不同的图柄上时,指针图标也会发生改变。

   添加"截取ROI区域"按钮的事件处理程序代码,代码如下:

void CRectTrackerTestDlg::OnBnClickedKeepRoiArea()
{
	dst = dst(Rect(pickRect.left, pickRect.top, pickRect.Width(), pickRect.Height()));
	MatToCImage(dst, mImage);
	bPikFrameEanble = false;
	pickRect = CRect(0, 0, 0, 0);
	pickFrame.m_rect = pickRect;
	mString = "截取ROI区域操作已完成";
	mInformation.SetWindowTextW(mString);
	Invalidate();
}

  添加"保存图像"按钮的事件处理程序代码,代码如下:

void CRectTrackerTestDlg::OnBnClickedSaveImage()
{
	mString = "正在进行图像存储操作...";
	mInformation.SetWindowTextW(mString);
	CString mfilter = _T("图片文件(*.bmp *.png *.jpg *.webp *.tif)|*.bmp;*.png;*jpg,*.webp,*.tif|All Files (*.*)|*.*||");
	CFileDialog fdlg(FALSE, NULL, 0, OFN_OVERWRITEPROMPT, mfilter, NULL);
	if (fdlg.DoModal() == IDOK)
	{
		m_Path = fdlg.GetPathName();
		m_strEx = fdlg.GetFileExt();
		m_strName = fdlg.GetFileName();

		m_Path.ReleaseBuffer();
		m_strEx.ReleaseBuffer();
		m_strName.ReleaseBuffer();
	}
	else
	{
		mString = "图像存储操作已被取消";
		mInformation.SetWindowTextW(mString);
		return;
	}
	if (m_strEx == "BMP" || m_strEx == "bmp" || m_strEx == "dib" || m_strEx == "TIF" || m_strEx == "tif" || m_strEx == "tiff" || m_strEx == "PNG" || m_strEx == "png"
		|| m_strEx == "jpg" || m_strEx == "JPG" || m_strEx == "jpe" || m_strEx == "jpeg" || m_strEx == "jp2" || m_strEx == "webp" || m_strEx == "avif" || m_strEx == "pbm"
		|| m_strEx == "pgm" || m_strEx == "ppm" || m_strEx == "pxm" || m_strEx == "pnm" || m_strEx == "pfm" || m_strEx == "sr" || m_strEx == "ras" || m_strEx == "exr"
		|| m_strEx == "hdr" || m_strEx == "pic")
	{
		m_str = CT2A(m_Path);
		imwrite(m_str, dst);
		mString.Format(L"图像文件已存储到:%s ", m_Path);
		mInformation.SetWindowTextW(mString);
	}
	else if (m_strEx == "")
	{
		m_Path += ".bmp";
		m_str = CT2A(m_Path);
		imwrite(m_str, dst);
		mString.Format(L"图像文件已存储到:%s ", m_Path);
		mInformation.SetWindowTextW(mString);
	}
}

到此,示例程序的代码已完成。试运行,结果如下:

点击“打开图像”按钮,选择打开图像文件:

点击“打开”按钮,打开图像,如下:

在“缩放图像”按钮旁的输入框中输入2.5,然后点击“缩放图像”按钮,结果如下:

点击“选取ROI区域”按钮,将鼠标指针移动到图片适当位置,点击鼠标左键,然后拖动鼠标到适当位置,点击鼠标左键,选取图像中自己感兴趣的区域,如下:

点击“截取ROI区域”按钮,结果如下:

点击“保存图像”按钮,选定路径,输入存储图像名:

点击保存按钮,结果如下:

再点击“打开图像”按钮,选择刚才存储的图片:

点击“打开”按钮,结果如下:

说明存储的图片可以正常打开。本示例程序测试到此结束。本示例程序的源代码已上传到CSDN,如果需要查看细节,可以去下载。下载链接为:https://download.csdn.net/download/billliu66/89541184

如果需要重新编译,因OpenCV的路径及版本不一定一致,一般需要重新配置 OpenCV。本示例程序是基于OpenCv4.90及VS2022编写。

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

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

相关文章

【学习笔记】无人机(UAV)在3GPP系统中的增强支持(十二)-无人机群在物流中的应用

引言 本文是3GPP TR 22.829 V17.1.0技术报告&#xff0c;专注于无人机&#xff08;UAV&#xff09;在3GPP系统中的增强支持。文章提出了多个无人机应用场景&#xff0c;分析了相应的能力要求&#xff0c;并建议了新的服务级别要求和关键性能指标&#xff08;KPIs&#xff09;。…

Flutter应用开发:掌握StatefulWidget的实用技巧

前言 随着移动应用的日益复杂&#xff0c;状态管理成为了 Flutter 应用开发中的一项重要挑战。 状态&#xff0c;即应用中的可变数据&#xff0c;它驱动着用户界面的渲染和交互。 在 Flutter 这样的声明式 UI 框架中&#xff0c;如何高效、可维护地管理状态&#xff0c;对于…

【Java--数据结构】队列

欢迎关注个人主页&#xff1a;逸狼 创造不易&#xff0c;可以点点赞吗~ 如有错误&#xff0c;欢迎指出~ 目录 队列 队列的方法 队列方法使用举例 模拟实现队列 使用链表实现队列 使用数组实现队列 设计循环队列 双端队列 用队列实现栈 队列 只允许在一端进行插入数据操作&…

昇思25天学习打卡营第15天|基于MindNLP+MusicGen生成自己的个性化音乐

MusicGen是来自Meta AI的Jade Copet等人提出的基于单个语言模型&#xff08;LM&#xff09;的音乐生成模型&#xff0c;能够根据文本描述或音频提示生成高质量的音乐样本&#xff0c;相关研究成果参考论文《Simple and Controllable Music Generation》。 MusicGen模型基于Tra…

notepad++中文出现异体汉字,怎么改正

notepad显示异体字&#xff0c;如何恢复&#xff1f; 比如 “门” 和 “直接” 的"直"字&#xff0c;显示成了 方法 修改字体&#xff0c; 菜单栏选择 Settings(设置&#xff09;&#xff0c;Style Configurator…&#xff08;语言格式设置…&#xff09;&#xf…

年轻人「躺平」、「摆烂」现象的根源是什么?

年轻人「躺平」、「摆烂」现象的根源是什么? 穷人没有资格躺平 我可以躺平吗?当然可以了! 对于有些人来说是躺平在房车里,直接开到命运的终点;而你是躺在马路中间,被命运的车轮反复碾压。 中国一线城市的00后,他们的父母多是没有哥哥、姐姐、弟弟、妹妹的独生子女,…

【Linux】多线程_7

文章目录 九、多线程8. POSIX信号量根据信号量环形队列的生产者消费者模型代码结果演示 未完待续 九、多线程 8. POSIX信号量 POSIX信号量和SystemV信号量作用相同&#xff0c;都是用于同步操作&#xff0c;达到无冲突的访问共享资源目的。 但POSIX可以用于线程间同步。 创建…

怎样在 PostgreSQL 中优化对复合索引的选择性?

&#x1f345;关注博主&#x1f397;️ 带你畅游技术世界&#xff0c;不错过每一次成长机会&#xff01;&#x1f4da;领书&#xff1a;PostgreSQL 入门到精通.pdf 文章目录 怎样在 PostgreSQL 中优化对复合索引的选择性一、理解复合索引的概念二、选择性的重要性三、优化复合索…

独立开发者系列(25)——大白话进程

很多小型的规模场景限制下&#xff0c;复杂概念弊端大于利端。不同模式的实现&#xff0c;是根据具体需求来判定&#xff0c;但是理解底层最基础的原理有助于理解很多工具背后的诞生。比如php的swoole workerman 要解决的问题。 首先理解&#xff0c;进程概念&#xff0c;进程…

线程安全(二)synchronized 的底层实现原理、锁升级、对象的内存结构

目录 一、基础使用1.1 不加锁的代码实现1.2 加锁的代码实现二、实现原理2.1 synchronized 简介2.2 对象监控器(Monitor)2.3 加锁过程第一步:判断 Owner 指向第二步:进入 EntryList 阻塞第三步:主动进入 WaitSet 等待三、锁升级3.1 对象的内存结构3.2 Mark Word 对象头3.3 …

实用教程:用 Go 的 net/textproto 包优化文本协议处理

实用教程&#xff1a;用 Go 的 net/textproto 包优化文本协议处理 介绍准备工作环境设置Go 基础回顾 基础使用创建连接发送请求接收响应 高级特性处理 MIME 头多行响应的管理错误处理与调试 实战案例实现一个简单的邮件客户端实现一个基于 net/textproto 的命令行工具 最佳实践…

【Wamp】局域网设备访问WampServer | 使用域名访问Wamp | Wamp配置HTTPS

局域网设备访问WampServer 参考&#xff1a;https://www.jianshu.com/p/d431a845e5cb 修改Apache的httpd.conf文件 D:\Academic\Wamp\program\bin\apache\apache2.4.54.2\conf\httpd.conf 搜索 Require local 和Require all denied&#xff0c;改为Require all granted <…

从头开始学习扩散模型 Stable Diffusion

今天我们来揭开 Stable Diffusion 技术的神秘面纱。 1.稳定扩散原理 Stable Diffusion 在2022年发表&#xff0c;一种基于Latent Diffusion Models的新兴机器学习技术。它基于扩散过程&#xff0c;利用数学模型将机器学习中的高维度数据降低到低维度空间&#xff0c;并在该空间…

【笔记】dbeaver导出数据库结构+数据 再导入其他数据库

导出&#xff1a; 导入 然后将语句粘贴进去 会有报错 选全部跳过 然后就全部添加成功了 虽然我不知道为什么报错 但是能加进去数据结构和数据都在就无所谓了 第二个版本 DBeaver导出sql脚本&#xff0c;执行sql脚本-CSDN博客 通过工具 DBeaver操作 MySQL导入备份的 sql 报错…

写好计算机类博文的技巧

在信息时代&#xff0c;计算机类博文成为了分享知识和经验的重要渠道。无论你是技术专家&#xff0c;还是爱好者&#xff0c;一篇优秀的计算机类博文不仅能展示你的专业能力&#xff0c;还能帮助他人解决问题。以下是写好计算机类博文的一些技巧&#xff0c;帮助你提升写作质量…

Netgear WN604 downloadFile.php 信息泄露漏洞复现(CVE-2024-6646)

0x01 产品简介 NETGEAR WN604是一款由NETGEAR(网件)公司生产的无线接入器(或无线路由器)提供Wi-Fi保护协议(WPA2-PSK, WPA-PSK),以及有线等效加密(WEP)64位、128位和152位支持,保障网络安全。同时支持MAC地址认证、802.1x RADIUS以及EAP TLS、TTLS、PEAP等安全机制,…

Descriptions 描述列表 label-class-name

需求&#xff1a; 在el-descriptions-item加上label-class-name‘fwText’ :deep(.fwText) { font-weight: bold !important; background-color: #f5f7fa !important; }

【理解C++中的树】

目录 一、树&#xff08;Tree&#xff09;的概念1.1、树的基本定义1.2、基本术语1.2、树的性质 二、二叉树2.1、二叉树的定义2.2、特殊二叉树2.2.1、满二叉树2.2.2、完全二叉树2.2.3、二叉排序树2.2.4、平衡二叉树 .3、二叉树的性质2.4、二叉树存储的实现2.4.1、顺序存储2.4.2、…

【香橙派 AIpro测评:探索高效图片分类项目实战】

前言 最近入手了一块香橙派 AIpro开发板&#xff0c;在使用中被它的强大深深震撼&#xff0c;有感而发写下这篇文章。 本文旨在深入探讨OrangePi AIpro的各项性能&#xff0c;从硬件配置、软件兼容性到实际应用案例&#xff0c;全方位解析这款设备如何在开源社区中脱颖而出&am…

C#与PLC通信——如何设置电脑IP地址

前言&#xff1a; 我们与PLC通过以太网通信时&#xff0c;首先要做的就是先设置好电脑的IP&#xff0c;这样才能实现上位机电脑与PLC之间的通信&#xff0c;并且电脑的ip地址和PLC的Ip地址要同处于一个网段&#xff0c;比如电脑的Ip地址为192.168.1.1&#xff0c;那么PLC的Ip地…