【QT教程】使用qcustomplot完成对图像的拖动、框选缩放、自动缩放、游标等操作

本文主要是介绍【QT教程】使用qcustomplot完成对图像的拖动、框选缩放、自动缩放、游标等操作,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

  • 1.Qt 配置qcustomplot
  • 2.图像拖拽功能
  • 3.图像框选放大
  • 4.曲线自动缩放
  • 5.图像游标
  • 6.【完整代码】将曲线抽象成一个类,以便复用

1.Qt 配置qcustomplot

首先下载qcustomplot官网(qcustomplot下载地址)下载最近的源码,我用的是2.1.1版本。下载完成后解压,得到如下文件。
在这里插入图片描述
将.h和.cpp文件加入到工程中
在这里插入图片描述
如果使用的是Qt creator的话,在.pro文件中加入printsupport

在这里插入图片描述

打开.ui文件,选中Widget模块拖入到界面当中

在这里插入图片描述
右键这个Widget,选择【提升为】
在这里插入图片描述

输入 QCustomPlot,点击添加、提升为

在这里插入图片描述
之后我们为这个控件改个名字就可以操作它啦。这里我给他重命名为widget_Force。

2.图像拖拽功能

首先在主线程中声明qcustomplot类,并实现一个对象。

    QTimer *plotTimer;//曲线刷新定时器/***************曲线绘制**********************//*选中的曲线画笔*/QVector<QPen> selectedPen;/*选中的图例字体*/QFont YHfontBold;/*力曲线*/QCustomPlot *widget_Force;CurvePlot *curves_Force;

图像拖拽功能

		//禁用选择矩形widget_Force->setSelectionRectMode(QCP::SelectionRectMode::srmNone);//使能拖动widget_Force->setInteraction(QCP::iRangeDrag, true);

演示

在这里插入图片描述

3.图像框选放大

        widget_Force->setInteraction(QCP::iRangeDrag, false);//取消拖动widget_Force->setSelectionRectMode(QCP::SelectionRectMode::srmZoom);

演示
在这里插入图片描述

4.曲线自动缩放

右键曲线实现自动缩放功能需要添加一个右键菜单栏。
首先定义信号槽

    //右键菜单connect(widget_Force, &QCustomPlot::customContextMenuRequested, this, &MainWindow::contextMenuRequestForce);

槽函数如下

void MainWindow::contextMenuRequestForce(QPoint pos)
{QMenu *menu = new QMenu(this);menu->setStyleSheet("font:11pt;font-family:Microsoft YaHei");menu->setAttribute(Qt::WA_DeleteOnClose);menu->popup(widget_Force->mapToGlobal(pos));menu->addAction("调整范围", curves_Force, &CurvePlot::rescaleAxes);menu->addAction("清除选中曲线", curves_Force, &CurvePlot::clearCurve);
}

rescaleAxes 函数如下

void CurvePlot::rescaleAxes()//曲线全部显示
{//给第一个graph设置rescaleAxes(),后续所有graph都设置rescaleAxes(true)即可实现显示所有曲线。//rescaleAxes(true)时如果plot的X或Y轴本来能容纳下本graph的X或Y数据点,//那么plot的X或Y轴的可视范围就无需调整,只有plot容纳不下本graph时,才扩展plot两个轴的显示范围。//见博客https://www.csdn.net/gather_26/MtTaYg2sMzgxNDgtYmxvZwO0O0OO0O0O.htmlmyPlot->graph(0)->rescaleAxes();myPlot->graph(1)->rescaleAxes(true);myPlot->replot();
}

演示
在这里插入图片描述

5.图像游标

使用游标的话需要配合鼠标事件完成
首先声明一下游标

	/*************游标*************/bool tracerEnable;//游标使能QCPItemTracer *tracer0 = nullptr; // 0号曲线游标QCPItemTracer *tracer1 = nullptr; // 1号曲线游标QCPItemText *tracer0Label = nullptr; // 0号曲线X游标标签QCPItemText *tracer1Label = nullptr;// 1号曲线Y轴游标标签void setVisibleTracer(bool trueorfalse);//游标可见/不可见 true/false

然后配置游标的属性

if (arg1){qDebug() << "act_tracer is on!";tracerEnable = true;tracer0 = new QCPItemTracer(myPlot);tracer0->setStyle(QCPItemTracer::tsCrosshair);//游标样式:十字星、圆圈、方框tracer0->setPen(QPen(Qt::green));//设置tracer的颜色绿色tracer0->setPen(QPen(Qt::DashLine));//虚线游标tracer0->setBrush(QBrush(Qt::red));tracer0->setSize(10);tracer0->setInterpolating(true);//false禁用插值tracer1 = new QCPItemTracer(myPlot);tracer1->setStyle(QCPItemTracer::tsCrosshair);//游标样式:十字星、圆圈、方框tracer1->setPen(QPen(Qt::green));//设置tracer的颜色绿色tracer1->setPen(QPen(Qt::DashLine));//虚线游标tracer1->setBrush(QBrush(Qt::red));tracer1->setSize(10);tracer1->setInterpolating(true);//false禁用插值tracer0Label = new QCPItemText(myPlot);tracer0Label->setClipToAxisRect(false);tracer0Label->setLayer("overlay");tracer0Label->setPen(QPen(Qt::green));tracer0Label->setFont(QFont("Microsoft YaHei", 10));tracer0Label->setPadding(QMargins(2, 2, 2, 2));tracer0Label->setPositionAlignment(Qt::AlignLeft | Qt::AlignTop);//下面这个语句很重要,它将游标说明锚固在tracer位置处,实现自动跟随tracer0Label->position->setType(QCPItemPosition::ptAxisRectRatio);//位置类型(当前轴范围的比例为单位/实际坐标为单位)tracer0Label->position->setParentAnchor(tracer0->position);tracer1Label = new QCPItemText(myPlot);tracer1Label->setClipToAxisRect(false);tracer1Label->setLayer("overlay");tracer1Label->setPen(QPen(Qt::red));tracer1Label->setFont(QFont("Microsoft YaHei", 10));tracer1Label->setPadding(QMargins(2, 2, 2, 2));tracer1Label->setPositionAlignment(Qt::AlignLeft | Qt::AlignTop);//下面这个语句很重要,它将游标说明锚固在tracer位置处,实现自动跟随tracer1Label->position->setType(QCPItemPosition::ptAxisRectRatio);//位置类型(当前轴范围的比例为单位/实际坐标为单位)tracer1Label->position->setParentAnchor(tracer1->position);}else{qDebug() << "act_tracer is off !";tracerEnable = false;setVisibleTracer(false);}

然后编写鼠标事件,并进行信号槽绑定。

   //游标鼠标事件信号槽connect(widget_Force, &QCustomPlot::mouseMove, curves_Force, &CurvePlot::myMouseMoveEvent);
void CurvePlot::myMouseMoveEvent(QMouseEvent * event)//鼠标移动事件
{if (tracerEnable)//游标使能判断{double x = myPlot->xAxis->pixelToCoord(event->pos().x());//鼠标点的像素坐标转plot坐标foundRange = true;QrangeX0 = myPlot->graph(0)->getKeyRange(foundRange, QCP::sdBoth);//获取0号曲线X轴坐标范围QrangeX0_lower = QrangeX0.lower;QrangeX0_upper = QrangeX0.upper;QrangeX1 = myPlot->graph(1)->getKeyRange(foundRange, QCP::sdBoth);//获取1号曲线X轴坐标范围QrangeX1_lower = QrangeX1.lower;QrangeX1_upper = QrangeX1.upper;//如果鼠标移动超出0号曲线X轴范围,则0号曲线隐藏游标if (x < QrangeX0_upper && x > QrangeX0_lower){tracer0->setGraph(myPlot->graph(0));//设置游标吸附在traceGraph这条曲线上tracer0->setGraphKey(x);//将游标横坐标(key)设置成刚获得的横坐标数据x (这就是游标随动的关键代码)tracer0->updatePosition(); //使得刚设置游标的横纵坐标位置生效double traceX0 = tracer0->position->key();double traceY0 = tracer0->position->value();tracer0Label->setText(QString::number(traceX0, 'f', 3));//游标文本框,指示游标的X值tracer0Label->setText(QString("x = %1\ny = %2").arg(QString::number(traceX0, 'f', 3)).arg(QString::number(traceY0, 'f', 3)));tracer0->setVisible(true);tracer0Label->setVisible(true);}else{tracer0->setVisible(false);tracer0Label->setVisible(false);}//如果鼠标移动超出1号曲线X轴范围,则1号曲线隐藏游标if (x < QrangeX1_upper && x > QrangeX1_lower){double traceX1 = tracer1->position->key();double traceY1 = tracer1->position->value();tracer1->setGraph(myPlot->graph(1));//设置游标吸附在traceGraph这条曲线上tracer1->setGraphKey(x);tracer1->updatePosition(); //使得刚设置游标的横纵坐标位置生效tracer1Label->setText(QString("x = %1\ny = %2").arg(QString::number(traceX1, 'f', 3)).arg(QString::number(traceY1, 'f', 3)));tracer1->setVisible(true);tracer1Label->setVisible(true);}else{tracer1->setVisible(false);tracer1Label->setVisible(false);}myPlot->replot(QCustomPlot::rpQueuedReplot); //刷新图标,不能省略}
}

效果展示
在这里插入图片描述

6.【完整代码】将曲线抽象成一个类,以便复用

这里我写了一个CurvePlot的类,如果界面上同时有多个绘图窗口,那么就可以方便的进行复用。通过这个类可以设置曲线范围等参数,也将一些鼠标事件写了进去。这个是根据之前的一个项目做的,绘图框中绘制两条曲线,一条是目标曲线,一条是实时采集到的曲线。.大家可以在这个基础上进行修改。

CurvePlot.h如下

#ifndef CURVEPLOT_H
#define CURVEPLOT_H
#include "qcustomplot.h"#pragma execution_character_set("utf-8")
class CurvePlot :public QObject
{Q_OBJECT
public:explicit CurvePlot(QCustomPlot * Plot);virtual ~CurvePlot();void setxAxisName(QString name);void setyAxisName(QString name);void setxAxisRange(double lower, double upper);void setyAxisRange(double lower, double upper);void setGraphName(QString name0, QString name1);void setSelectLegend(bool enabled);void setLegendFont(QFont font);/*************曲线数据*************/int curIndex;//当前曲线号 = 0, = 1, = -1时未选中曲线//0号曲线(目标曲线)QVector<double> x0;//存储x坐标的向量QVector<double> y0;//存储y坐标的向量QList<QList<QVariant>> x_y0;//1号曲线(采集曲线)QVector<double> x1;//存储x坐标的向量QVector<double> y1;//存储y坐标的向量QList<QList<QVariant>> x_y1;public slots:void on_act_tracerToggled(bool arg1);void myMouseMoveEvent(QMouseEvent* event);//鼠标事件void rescaleAxes();//自动调整void clearCurve();//清除曲线void clearAllCurves();//清除所有曲线
private:/*绘图窗口*/QCustomPlot * myPlot = nullptr;/*画笔*/QVector<QPen> pen;/*************游标*************/bool tracerEnable;//游标使能QCPItemTracer *tracer0 = nullptr; // 0号曲线游标QCPItemTracer *tracer1 = nullptr; // 1号曲线游标QCPItemText *tracer0Label = nullptr; // 0号曲线X游标标签QCPItemText *tracer1Label = nullptr;// 1号曲线Y轴游标标签void setVisibleTracer(bool trueorfalse);//游标可见/不可见 true/false/*********曲线X轴范围**********/QCPRange QrangeX0;//0号曲线X轴范围double  QrangeX0_lower;double  QrangeX0_upper;QCPRange QrangeX1;//1号曲线X轴范围double  QrangeX1_lower;double  QrangeX1_upper;bool foundRange;
};#endif

CurvePlot.cpp如下

#include "CurvePlot.h"CurvePlot::CurvePlot(QCustomPlot * Plot)
{myPlot = Plot;tracerEnable = false;/*************绘图模块***************///设置坐标轴字体QFont YHfont("Microsoft YaHei", 10, QFont::Normal);myPlot->setFont(YHfont);myPlot->xAxis->setLabelFont(YHfont);myPlot->xAxis->setTickLabelFont(YHfont);myPlot->yAxis->setLabelFont(YHfont);myPlot->yAxis->setTickLabelFont(YHfont);//设定右上角图例标注的字体myPlot->legend->setFont(YHfont);//设定右上角图例标注可见myPlot->legend->setVisible(true);//添加图形myPlot->addGraph();myPlot->addGraph();//曲线全部可见myPlot->graph(0)->rescaleAxes();myPlot->graph(1)->rescaleAxes(true);//设置画笔QPen pen0(Qt::blue, 1, Qt::SolidLine, Qt::RoundCap, Qt::RoundJoin);QPen pen1(Qt::red, 1, Qt::SolidLine, Qt::RoundCap, Qt::RoundJoin);pen.append(pen0);pen.append(pen1);myPlot->graph(0)->setPen(pen[0]);myPlot->graph(1)->setPen(pen[1]);//设置线型myPlot->graph(0)->setLineStyle(QCPGraph::lsLine);myPlot->graph(1)->setLineStyle(QCPGraph::lsLine);//设置线上点的风格myPlot->graph(0)->setScatterStyle(QCPScatterStyle::ssNone);myPlot->graph(1)->setScatterStyle(QCPScatterStyle::ssNone);//右键菜单自定义myPlot->setContextMenuPolicy(Qt::CustomContextMenu);/**********鼠标操作图像模块************/myPlot->selectionRect()->setPen(QPen(Qt::black, 1, Qt::DashLine));//设置选框的样式:虚线myPlot->selectionRect()->setBrush(QBrush(QColor(0, 0, 100, 50)));//设置选框的样式:半透明浅蓝myPlot->setInteraction(QCP::iRangeDrag, true); //鼠标单击拖动 QCPAxisRect::mousePressEvent() 左键拖动myPlot->setInteraction(QCP::iRangeZoom, true); //滚轮滑动缩放myPlot->setInteraction(QCP::iSelectAxes, true);myPlot->setInteraction(QCP::iSelectLegend, true); //图例可选myPlot->selectionRect()->setPen(QPen(Qt::black, 1, Qt::DashLine));myPlot->selectionRect()->setBrush(QBrush(QColor(0, 0, 100, 50)));
}CurvePlot::~CurvePlot()
{
}
//X轴名称
void CurvePlot::setxAxisName(QString name)
{myPlot->xAxis->setLabel(name);
}
//Y轴名称
void CurvePlot::setyAxisName(QString name)
{myPlot->yAxis->setLabel(name);
}
//X轴范围
void CurvePlot::setxAxisRange(double lower, double upper)
{myPlot->xAxis->setRange(lower, upper);
}
//Y轴范围
void CurvePlot::setyAxisRange(double lower, double upper)
{myPlot->yAxis->setRange(lower, upper);
}
//曲线图例名称
void CurvePlot::setGraphName(QString name0, QString name1)
{//设置右上角图形标注名称myPlot->graph(0)->setName(name0);myPlot->graph(1)->setName(name1);
}
void CurvePlot::setSelectLegend(bool enabled)
{if (enabled){myPlot->setInteraction(QCP::iSelectLegend, true); //图例可选}else{myPlot->setInteraction(QCP::iSelectLegend, false); //图例可选}
}
void CurvePlot::setLegendFont(QFont font)
{myPlot->legend->setFont(font);
}
void CurvePlot::on_act_tracerToggled(bool arg1)
{if (arg1){qDebug() << "act_tracer is on!";tracerEnable = true;tracer0 = new QCPItemTracer(myPlot);tracer0->setStyle(QCPItemTracer::tsCrosshair);//游标样式:十字星、圆圈、方框tracer0->setPen(QPen(Qt::green));//设置tracer的颜色绿色tracer0->setPen(QPen(Qt::DashLine));//虚线游标tracer0->setBrush(QBrush(Qt::red));tracer0->setSize(10);tracer0->setInterpolating(true);//false禁用插值tracer1 = new QCPItemTracer(myPlot);tracer1->setStyle(QCPItemTracer::tsCrosshair);//游标样式:十字星、圆圈、方框tracer1->setPen(QPen(Qt::green));//设置tracer的颜色绿色tracer1->setPen(QPen(Qt::DashLine));//虚线游标tracer1->setBrush(QBrush(Qt::red));tracer1->setSize(10);tracer1->setInterpolating(true);//false禁用插值tracer0Label = new QCPItemText(myPlot);tracer0Label->setClipToAxisRect(false);tracer0Label->setLayer("overlay");tracer0Label->setPen(QPen(Qt::green));tracer0Label->setFont(QFont("Microsoft YaHei", 10));tracer0Label->setPadding(QMargins(2, 2, 2, 2));tracer0Label->setPositionAlignment(Qt::AlignLeft | Qt::AlignTop);//下面这个语句很重要,它将游标说明锚固在tracer位置处,实现自动跟随tracer0Label->position->setType(QCPItemPosition::ptAxisRectRatio);//位置类型(当前轴范围的比例为单位/实际坐标为单位)tracer0Label->position->setParentAnchor(tracer0->position);tracer1Label = new QCPItemText(myPlot);tracer1Label->setClipToAxisRect(false);tracer1Label->setLayer("overlay");tracer1Label->setPen(QPen(Qt::red));tracer1Label->setFont(QFont("Microsoft YaHei", 10));tracer1Label->setPadding(QMargins(2, 2, 2, 2));tracer1Label->setPositionAlignment(Qt::AlignLeft | Qt::AlignTop);//下面这个语句很重要,它将游标说明锚固在tracer位置处,实现自动跟随tracer1Label->position->setType(QCPItemPosition::ptAxisRectRatio);//位置类型(当前轴范围的比例为单位/实际坐标为单位)tracer1Label->position->setParentAnchor(tracer1->position);}else{qDebug() << "act_tracer is off !";tracerEnable = false;setVisibleTracer(false);}
}
void CurvePlot::myMouseMoveEvent(QMouseEvent * event)//鼠标移动事件
{if (tracerEnable)//游标使能判断{double x = myPlot->xAxis->pixelToCoord(event->pos().x());//鼠标点的像素坐标转plot坐标foundRange = true;QrangeX0 = myPlot->graph(0)->getKeyRange(foundRange, QCP::sdBoth);//获取0号曲线X轴坐标范围QrangeX0_lower = QrangeX0.lower;QrangeX0_upper = QrangeX0.upper;QrangeX1 = myPlot->graph(1)->getKeyRange(foundRange, QCP::sdBoth);//获取1号曲线X轴坐标范围QrangeX1_lower = QrangeX1.lower;QrangeX1_upper = QrangeX1.upper;//如果鼠标移动超出0号曲线X轴范围,则0号曲线隐藏游标if (x < QrangeX0_upper && x > QrangeX0_lower){tracer0->setGraph(myPlot->graph(0));//设置游标吸附在traceGraph这条曲线上tracer0->setGraphKey(x);//将游标横坐标(key)设置成刚获得的横坐标数据x (这就是游标随动的关键代码)tracer0->updatePosition(); //使得刚设置游标的横纵坐标位置生效double traceX0 = tracer0->position->key();double traceY0 = tracer0->position->value();tracer0Label->setText(QString::number(traceX0, 'f', 3));//游标文本框,指示游标的X值tracer0Label->setText(QString("x = %1\ny = %2").arg(QString::number(traceX0, 'f', 3)).arg(QString::number(traceY0, 'f', 3)));tracer0->setVisible(true);tracer0Label->setVisible(true);}else{tracer0->setVisible(false);tracer0Label->setVisible(false);}//如果鼠标移动超出1号曲线X轴范围,则1号曲线隐藏游标if (x < QrangeX1_upper && x > QrangeX1_lower){double traceX1 = tracer1->position->key();double traceY1 = tracer1->position->value();tracer1->setGraph(myPlot->graph(1));//设置游标吸附在traceGraph这条曲线上tracer1->setGraphKey(x);tracer1->updatePosition(); //使得刚设置游标的横纵坐标位置生效tracer1Label->setText(QString("x = %1\ny = %2").arg(QString::number(traceX1, 'f', 3)).arg(QString::number(traceY1, 'f', 3)));tracer1->setVisible(true);tracer1Label->setVisible(true);}else{tracer1->setVisible(false);tracer1Label->setVisible(false);}myPlot->replot(QCustomPlot::rpQueuedReplot); //刷新图标,不能省略}
}void CurvePlot::rescaleAxes()//曲线全部显示
{//给第一个graph设置rescaleAxes(),后续所有graph都设置rescaleAxes(true)即可实现显示所有曲线。//rescaleAxes(true)时如果plot的X或Y轴本来能容纳下本graph的X或Y数据点,//那么plot的X或Y轴的可视范围就无需调整,只有plot容纳不下本graph时,才扩展plot两个轴的显示范围。//见博客https://www.csdn.net/gather_26/MtTaYg2sMzgxNDgtYmxvZwO0O0OO0O0O.htmlmyPlot->graph(0)->rescaleAxes();myPlot->graph(1)->rescaleAxes(true);myPlot->replot();
}void CurvePlot::clearCurve()
{switch (this->curIndex){case 0:this->x0.clear();this->y0.clear();this->x_y0.clear();myPlot->graph(0)->setData(this->x0, this->y0);break;case 1:this->x1.clear();this->y1.clear();this->x_y1.clear();myPlot->graph(1)->setData(this->x1, this->y1);break;case -1:qDebug() << "未选中曲线!";break;}myPlot->replot(QCustomPlot::rpQueuedReplot);
}void CurvePlot::clearAllCurves()
{//this->x0.clear();//this->y0.clear();//this->x_y0.clear();//myPlot->graph(0)->setData(this->x0, this->y0);this->x1.clear();this->y1.clear();this->x_y1.clear();myPlot->graph(1)->setData(this->x1, this->y1);myPlot->replot(QCustomPlot::rpQueuedReplot);
}void CurvePlot::setVisibleTracer(bool trueorfalse)
{tracer0->setVisible(trueorfalse);tracer1->setVisible(trueorfalse);tracer0Label->setVisible(trueorfalse);tracer1Label->setVisible(trueorfalse);myPlot->replot(QCustomPlot::rpQueuedReplot); //刷新图标,不能省略
}

这篇关于【QT教程】使用qcustomplot完成对图像的拖动、框选缩放、自动缩放、游标等操作的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1074497

相关文章

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

中文分词jieba库的使用与实景应用(一)

知识星球:https://articles.zsxq.com/id_fxvgc803qmr2.html 目录 一.定义: 精确模式(默认模式): 全模式: 搜索引擎模式: paddle 模式(基于深度学习的分词模式): 二 自定义词典 三.文本解析   调整词出现的频率 四. 关键词提取 A. 基于TF-IDF算法的关键词提取 B. 基于TextRank算法的关键词提取

基于人工智能的图像分类系统

目录 引言项目背景环境准备 硬件要求软件安装与配置系统设计 系统架构关键技术代码示例 数据预处理模型训练模型预测应用场景结论 1. 引言 图像分类是计算机视觉中的一个重要任务,目标是自动识别图像中的对象类别。通过卷积神经网络(CNN)等深度学习技术,我们可以构建高效的图像分类系统,广泛应用于自动驾驶、医疗影像诊断、监控分析等领域。本文将介绍如何构建一个基于人工智能的图像分类系统,包括环境

使用SecondaryNameNode恢复NameNode的数据

1)需求: NameNode进程挂了并且存储的数据也丢失了,如何恢复NameNode 此种方式恢复的数据可能存在小部分数据的丢失。 2)故障模拟 (1)kill -9 NameNode进程 [lytfly@hadoop102 current]$ kill -9 19886 (2)删除NameNode存储的数据(/opt/module/hadoop-3.1.4/data/tmp/dfs/na

Hadoop数据压缩使用介绍

一、压缩原则 (1)运算密集型的Job,少用压缩 (2)IO密集型的Job,多用压缩 二、压缩算法比较 三、压缩位置选择 四、压缩参数配置 1)为了支持多种压缩/解压缩算法,Hadoop引入了编码/解码器 2)要在Hadoop中启用压缩,可以配置如下参数

Makefile简明使用教程

文章目录 规则makefile文件的基本语法:加在命令前的特殊符号:.PHONY伪目标: Makefilev1 直观写法v2 加上中间过程v3 伪目标v4 变量 make 选项-f-n-C Make 是一种流行的构建工具,常用于将源代码转换成可执行文件或者其他形式的输出文件(如库文件、文档等)。Make 可以自动化地执行编译、链接等一系列操作。 规则 makefile文件

使用opencv优化图片(画面变清晰)

文章目录 需求影响照片清晰度的因素 实现降噪测试代码 锐化空间锐化Unsharp Masking频率域锐化对比测试 对比度增强常用算法对比测试 需求 对图像进行优化,使其看起来更清晰,同时保持尺寸不变,通常涉及到图像处理技术如锐化、降噪、对比度增强等 影响照片清晰度的因素 影响照片清晰度的因素有很多,主要可以从以下几个方面来分析 1. 拍摄设备 相机传感器:相机传

嵌入式QT开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设

pdfmake生成pdf的使用

实际项目中有时会有根据填写的表单数据或者其他格式的数据,将数据自动填充到pdf文件中根据固定模板生成pdf文件的需求 文章目录 利用pdfmake生成pdf文件1.下载安装pdfmake第三方包2.封装生成pdf文件的共用配置3.生成pdf文件的文件模板内容4.调用方法生成pdf 利用pdfmake生成pdf文件 1.下载安装pdfmake第三方包 npm i pdfma

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]