案例分享:Qt用于服务器多图像拼接存在误差的标定工具(像素误差校准)

简介: 案例分享:Qt用于服务器多图像拼接存在误差的标定工具(像素误差校准)

若该文为原创文章,转载请注明原文出处

本文章博客地址:https://blog.csdn.net/qq21497936/article/details/79556106

各位读者,知识无穷而人力有穷,要么改需求,要么找专业人士,要么自己研究

红胖子(红模仿)的博文大全:开发技术集合(包含Qt实用技术、树莓派、三维、OpenCV、OpenGL、ffmpeg、OSG、单片机、软硬结合等等)持续更新中…(点击传送门)

合作案例专栏:案例分享(体验Demo可下载,只定制)

 

需求

       高性能图像拼接服务器由于在存在误差,需要使用标定软件获取计算后的每个位置图片进行像素标定(校准),将人工校准的图像相关信息以指令形式发送给服务器端,服务器的每次拼接将依赖此标定参数。

       标定80幅图,10台服务器,每台服务器8路1080p摄像头。

 

Demo

       下载地址:https://download.csdn.net/download/qq21497936/10286573

               

效果图

 

原理

       打开文件拼接,按照范围1-80的顺序打开图像文件,加载到软件里面,每图片的5个像素作为一个实际坐标像素,记录每个图片相对于坐标第一张图片的左上角x,y的坐标,以及他们的宽度和高度,组成单幅图像的拼接参数。

 

功能

       可加载任意多的图片,对一个图片进行移动,包括上下左右,校准完后点击制作,将会把所有图片指令以指令格式组成协议包发往拼接服务器,拼接服务器各自记录自己部分的拼接参数,以便运行时使用。

 

关键代码

打开文件图片文件代码(打开“按钮”)

void MainWindow::on_pushButton_locaPics_clicked()
{
    QStringList filePaths = QFileDialog::getOpenFileNames(this, tr("添加图片"), "./", tr("Images (*.png *.bmp *.jpg)"));
    if(!filePaths.size())
    {
        QMessageBox::information(this, tr("提示"), tr("您未选择图片"));
        return;
    }
    ui->pushButton_moveXLess->setEnabled(true);
    ui->pushButton_moveXMore->setEnabled(true);
    ui->pushButton_moveUp->setEnabled(true);
    ui->pushButton_moveDown->setEnabled(true);
    ui->pushButton_make->setEnabled(true);
    for(int index = 0; index < _listPixmapButton.size(); index++)
    {
        delete _listPixmapButton.at(index);
    }
    _listPixmapButton.clear();
    for(int index = 0; index < _listLabel.size(); index++)
    {
        delete _listLabel.at(index);
    }
    _listLabel.clear();
    for(int index = 0; index < filePaths.size(); index++)
    {
        PixmapButton * pPixmapButton = new PixmapButton(ui->frame);
        pPixmapButton->setStyleSheet("");
        pPixmapButton->setPosWidth(_picWidth);
        pPixmapButton->setPosHeight(_picHeight);
        pPixmapButton->resize(_picWidth, _picHeight);
        pPixmapButton->setStyleSheet(QString("border-image: url(%1);").arg(filePaths.at(index)));
        connect(pPixmapButton, SIGNAL(updateChecked(PixmapButton*,bool)), this, SLOT(updateChecked(PixmapButton*,bool)));
        _listPixmapButton.push_back(pPixmapButton);
    }
    for(int index = 0; index < _listPixmapButton.size()-1; index++)
    {
        QLabel *pLabel = new QLabel(ui->frame);
        pLabel->setFixedWidth(100);
        pLabel->setText("0 pixel");
        _listLabel.push_back(pLabel);
    }
    _init = false;
    updateFrame();
}

对于每个图像使用QPusuButton,贴上图片作为一个图片单元,QPushButton存在两套x,y,width和height,一套是QPushButton实际的,一套是对应于左边第一张图片的x,y,width,height(方便制作协议)

pixmapbutton.h

#ifndef PIXMAPBUTTON_H
#define PIXMAPBUTTON_H
/**************************************************\
 * Demo: 图像块
 * 描述: 图像大小
 * 作者: 红模仿   QQ: 21497936    博客地址:http://blog.csdn.net/qq21497936/
 * 备注:作者提供各种技术相关服务,如开发、调试、IT集成服务等等
 * 版本信息:    版本号       日期                  描述
 *             v1.0    2018年1月31日           基础版本
\**************************************************/
#include <QWidget>
#include <QPushButton>
#include <QPaintEvent>
#include <QColor>
class PixmapButton : public QPushButton
{
    Q_OBJECT
public:
    explicit PixmapButton(QWidget *parent = nullptr);
public:
    inline void setCheckedOne(bool checkedOne) { _checkedOne = checkedOne; }
    inline void setPosX(int x) { _x = x; }
    inline void setPosY(int y) { _y = y; }
    inline int getPosX() { return _x; }
    inline int getPosY() { return _y; }
    inline void setPosWidth(int width) { _width = width; }
    inline int getPosWidth() { return _width; }
    inline void setPosHeight(int height) { _height = height; }
    inline int getPosHeight() { return _height; }
    inline void setPos(int x, int y) { _x = x; _y = y; }
    inline void setPos(int x, int y, int width, int height) { _x = x; _y = y; _width = width; _height = height; }
    inline void getPos(int &x, int &y, int &width, int &height) { x = _x; y = _y; width = _width; height = _height; }
signals:
    void updateChecked(PixmapButton * pPixmapButton, bool checked);
public slots:
    void slotCheck(bool checked);
protected:
    void paintEvent(QPaintEvent *event) override;
private:
    int _x;
    int _y;
    int _width;
    int _height;
    QColor _color;
    bool _checkedOne;
};
#endif // PIXMAPBUTTON_H

pixmapbutton.cpp

#include "pixmapbutton.h"
#include "mainwindow.h"
#include <QPainter>
#include <QDebug>
PixmapButton::PixmapButton(QWidget *parent)
    : QPushButton(parent),
      _color(QColor(41,19,244)),
      _checkedOne(false)
{
    setCheckable(true);
    resize(400,200);
    _width = 400;
    _height = 200;
    connect(this, SIGNAL(clicked(bool)), this, SLOT(slotCheck(bool)));
}
void PixmapButton::slotCheck(bool checked)
{
    emit updateChecked(this, checked);
    emit updateChecked(this, checked);
    emit updateChecked(this, checked);
}
void PixmapButton::paintEvent(QPaintEvent *event)
{
    QPushButton::paintEvent(event);
}

更新显示图片的代码

void MainWindow::updateFrame()
{
    int x, y, width;
    ui->frame->resize(ui->scrollArea->width(), ui->scrollArea->height());
    if(!_init)
    {
        for(int index = 0; index < _listPixmapButton.size(); index++)
        {
            if(index == 0)
            {
                x = 0;
                width = _listPixmapButton.at(index)->width();
                y = (ui->frame->height() - _listPixmapButton.at(index)->height()) / 2.0f;
            }
            x += SPACER;
            _listPixmapButton.at(index)->move(x,y);
            _listPixmapButton.at(index)->setPosX(x);
            _listPixmapButton.at(index)->setPosY(y);
            _listPixmapButton.at(index)->show();
            _listPixmapButton.at(index)->update();
            x += _listPixmapButton.at(index)->width();
            if(index == _listPixmapButton.size() - 1)
            {
                ui->frame->setFixedWidth(x+SPACER);
            }
            if(_listPixmapButton.at(index)->isChecked())
            {
                int x,y,width,height;
                _listPixmapButton.at(index)->getPos(x, y, width, height);
                ui->label_arrow->move((x+(width-ui->label_arrow->width())/2), y+height+_arrowSpace+_dy);
                ui->label_arrow->setVisible(true);
            }
        }
        _init = true;
    }else{
        for(int index = 0; index < _listPixmapButton.size(); index++)
        {
            PixmapButton *pPixmapButton = _listPixmapButton.at(index);
            pPixmapButton->move(pPixmapButton->getPosX(), pPixmapButton->getPosY()+_dy);
            if(pPixmapButton->isChecked())
            {
                int x,y,width,height;
                pPixmapButton->getPos(x, y, width, height);
                ui->label_arrow->move((x+(width-ui->label_arrow->width())/2), y+height+_arrowSpace+_dy);
                ui->label_arrow->setVisible(true);
            }
        }
    }
    for(int index = 0; index < _listPixmapButton.size(); index++)
    {
        if(index == 0)
            continue;
        int interval = _listPixmapButton.at(index)->getPosX()-(_listPixmapButton.at(index-1)->getPosX()+_listPixmapButton.at(index-1)->getPosWidth());
        _listLabel.at(index-1)->setText(QString("%1 pixel").arg(interval*_scale));
        _listLabel.at(index-1)->move(_listPixmapButton.at(index)->getPosX(), _listPixmapButton.at(index)->getPosY()-_pixelSpace+_dy);
        _listLabel.at(index-1)->setVisible(true);
    }
}

左移代码

void MainWindow::on_pushButton_moveXLess_clicked()
{
    bool isChecked = false;
    for(int index = 0; index < _listPixmapButton.size(); index++)
    {
        PixmapButton *pPixmapButton = _listPixmapButton.at(index);
        if(pPixmapButton->isChecked())
        {
            isChecked = true;
            if(pPixmapButton->getPosX() - _xStep < 0)
            {
                break;
            }else{
                if(index > 0)
                {
                    // 当右边的图片的x与左边的图片x相差小于_minXSapce时,从右边到最后边的所有图片将不会做x移动
                    if(_listPixmapButton.at(index)->getPosX() - _listPixmapButton.at(index-1)->getPosX() < _minXSapce)
                    {
                        break;
                    }
                }
                for(int index2 = index; index2 < _listPixmapButton.size(); index2++)
                {
                    PixmapButton *pPixmapButton = _listPixmapButton.at(index2);
                    pPixmapButton->setPosX(pPixmapButton->getPosX()-_xStep);
                }
                break;
            }
        }
    }
    if(!isChecked)
    {
        int start;
        for(int index = 0; index < _listPixmapButton.size(); index++)
        {
            if(index > 0)
            {
                // 当右边的图片的x与左边的图片x相差小于_minXSapce时,从右边到最后边的所有图片将不会做x移动
                if(_listPixmapButton.at(index)->getPosX() - _listPixmapButton.at(index-1)->getPosX() < _minXSapce)
                {
                    break;
                }
            }
            PixmapButton *pPixmapButton = _listPixmapButton.at(index);
            if(pPixmapButton->getPosX()-_xStep < 0)
            {
                start = index;
                continue;
            }else
            {
                pPixmapButton->setPosX(pPixmapButton->getPosX()-(index-start)*_xStep);
            }
            if(index == _listPixmapButton.size() - 1)
            {
//                ui->frame->setFixedWidth(_listPixmapButton.at(index)->x()+_listPixmapButton.at(index)->width());
            }
        }
    }
    updateFrame();
}

左移代码

void MainWindow::on_pushButton_moveXMore_clicked()
{
    bool isChecked = false;
    for(int index = 0; index < _listPixmapButton.size(); index++)
    {
        PixmapButton *pPixmapButton = _listPixmapButton.at(index);
        if(pPixmapButton->isChecked())
        {
            isChecked = true;
            if(index == 0)
                break;
            for(int index2 = index; index2 < _listPixmapButton.size(); index2++)
            {
                PixmapButton *pPixmapButton = _listPixmapButton.at(index2);
                pPixmapButton->setPosX(pPixmapButton->getPosX()+_xStep);
                if(index2 == _listPixmapButton.size() - 1)
                {
                    if(ui->frame->width() < _listPixmapButton.at(index2)->x()+_listPixmapButton.at(index2)->width())
                        ui->frame->setFixedWidth(_listPixmapButton.at(index2)->x()+_listPixmapButton.at(index2)->width());
                }
            }
            break;
        }
    }
    if(!isChecked)
    {
        for(int index = 0; index < _listPixmapButton.size(); index++)
        {
            if(index == 0)
                continue;
            PixmapButton *pPixmapButton = _listPixmapButton.at(index);
            pPixmapButton->setPosX(pPixmapButton->getPosX()+index*_xStep);
            if(index == _listPixmapButton.size() - 1)
            {
                if(ui->frame->width() < _listPixmapButton.at(index)->x()+_listPixmapButton.at(index)->width())
                    ui->frame->setFixedWidth(_listPixmapButton.at(index)->x()+_listPixmapButton.at(index)->width());
            }
        }
    }
    updateFrame();
}

上移代码

void MainWindow::on_pushButton_moveUp_clicked()
{
    bool isChecked = false;
    for(int index = 0; index < _listPixmapButton.size(); index++)
    {
        PixmapButton *pPixmapButton = _listPixmapButton.at(index);
        if(pPixmapButton->isChecked())
        {
            isChecked = true;
            if(index >= 1)
            {
                if(pPixmapButton->getPosY()+pPixmapButton->height()-_listPixmapButton.at(index-1)->getPosY() <= _minXSapce)
                    break;
            }
            if(index < _listPixmapButton.size()-1)
            {
                if(pPixmapButton->getPosY()+pPixmapButton->height()-_listPixmapButton.at(index+1)->getPosY() <= _minXSapce)
                    break;
            }
            pPixmapButton->setPosY(pPixmapButton->getPosY()-_yStep);
            break;
        }
    }
    if(!isChecked)
    {
        for(int index = 0; index < _listPixmapButton.size(); index++)
        {
            PixmapButton *pPixmapButton = _listPixmapButton.at(index);
            pPixmapButton->setPosY(pPixmapButton->getPosY()-_yStep);
        }
    }
    updateFrame();
}

下移代码

void MainWindow::on_pushButton_moveDown_clicked()
{
    bool isChecked = false;
    for(int index = 0; index < _listPixmapButton.size(); index++)
    {
        PixmapButton *pPixmapButton = _listPixmapButton.at(index);
        if(pPixmapButton->isChecked())
        {
            isChecked = true;
            if(index >= 1)
            {
                if(pPixmapButton->getPosY() - (_listPixmapButton.at(index-1)->getPosY() + _listPixmapButton.at(index-1)->getPosHeight()) >= -_minYSapce)
                    break;
            }
            if(index < _listPixmapButton.size()-1)
            {
                if(pPixmapButton->getPosY() - (_listPixmapButton.at(index+1)->getPosY() + _listPixmapButton.at(index+1)->getPosHeight()) >= -_minYSapce)
                    break;
            }
            pPixmapButton->setPosY(pPixmapButton->getPosY()+_yStep);
            break;
        }
    }
    if(!isChecked)
    {
        for(int index = 0; index < _listPixmapButton.size(); index++)
        {
            PixmapButton *pPixmapButton = _listPixmapButton.at(index);
            pPixmapButton->setPosY(pPixmapButton->getPosY()+_yStep);
        }
    }
    updateFrame();
}

快捷键代码

void MainWindow::keyPressEvent(QKeyEvent *event)
{
    switch (event->key()) {
    case Qt::Key_W:
        if(ui->pushButton_moveUp->isEnabled())
        {
            on_pushButton_moveUp_clicked();
        }
        break;
    case Qt::Key_S:
        if(ui->pushButton_moveDown->isEnabled())
        {
            on_pushButton_moveDown_clicked();
        }
        break;
    case Qt::Key_A:
        if(ui->pushButton_moveXLess->isEnabled())
        {
            on_pushButton_moveXLess_clicked();
        }
        break;
    case Qt::Key_D:
        if(ui->pushButton_moveXMore->isEnabled())
        {
            on_pushButton_moveXMore_clicked();
        }
        break;
    case Qt::Key_O:
        on_pushButton_locaPics_clicked();
        break;
    default:
        break;
    }
}

制作标定协议代码

void MainWindow::on_pushButton_make_clicked()
{
    QList<PicInfo> listPicInfo;
    for(int index = 0;  index < _listPixmapButton.size(); index++)
    {
        PixmapButton *pPixmapButton = _listPixmapButton.at(index);
        PicInfo info;
        info.index = index+1;
        info.x = pPixmapButton->getPosX()*_scale;
        info.y = pPixmapButton->getPosY()*_scale;
        info.width = pPixmapButton->getPosWidth()*_scale;
        info.height = pPixmapButton->getPosHeight()*_scale;
        listPicInfo.push_back(info);
    }
    // show data
    qDebug() << "*******************************************************";
    for(int index = 0; index < listPicInfo.size(); index++)
    {
        qDebug() << "index :" << listPicInfo.at(index).index << "   "
                 << "x :" << listPicInfo.at(index).x << "   "
                 << "y :" << listPicInfo.at(index).y << "   "
                 << "width :" << listPicInfo.at(index).width << "   "
                 << "height :" << listPicInfo.at(index).height;
    }
    qDebug() << "*******************************************************";
}

原博主博客地址:https://blog.csdn.net/qq21497936

原博主博客导航:https://blog.csdn.net/qq21497936/article/details/102478062

本文章博客地址:https://blog.csdn.net/qq21497936/article/details/79556106


相关文章
|
5天前
|
存储 数据挖掘 数据库
服务器数据恢复—OceanStor存储数据恢复案例
华为OceanStor T系列某型号存储中有一组由24块机械硬盘组建的一组RAID5阵列。 运行过程中该存储设备RAID5阵列上多块硬盘出现故障离线,阵列失效,存储中数据无法访问。
|
2天前
|
运维 数据挖掘 Windows
服务器数据恢复—服务器硬盘指示灯亮黄灯的数据恢复案例
服务器硬盘指示灯闪烁黄灯是一种警示,意味着服务器硬盘出现故障即将下线。发现这种情况建议及时更换硬盘。 一旦服务器上有大量数据频繁读写,硬盘指示灯会快速闪烁。服务器上某个硬盘的指示灯只有黄灯亮着,而其他颜色的灯没有亮的话,通常表示这块硬盘出现故障,这时候更换新硬盘同步数据即可。 如果没有及时发现硬盘损坏或者更换硬盘失败导致服务器崩溃,应该如何恢复数据呢?下面通过一个真实案例讲解一下服务器硬盘指示灯亮黄色的数据恢复案例。
|
3天前
|
存储 数据挖掘
服务器数据恢复—zfs文件系统服务器数据恢复案例
一台配有32块硬盘的服务器在运行过程中突然崩溃不可用。经过初步检测,基本上确定服务器硬件不存在物理故障。管理员重启服务器后问题依旧。需要恢复该服务器中的数据。
|
12天前
|
运维 数据挖掘 索引
服务器数据恢复—Lustre分布式文件系统服务器数据恢复案例
5台节点服务器,每台节点服务器上有一组RAID5阵列。每组RAID5阵列上有6块硬盘(其中1块硬盘设置为热备盘,其他5块硬盘为数据盘)。上层系统环境为Lustre分布式文件系统。 机房天花板漏水导致这5台节点服务器进水,每台服务器都有至少2块硬盘出现故障。每台服务器中的RAID5阵列短时间内同时掉线2块或以上数量的硬盘,导致RAID崩溃,服务器中数据无法正常读取。
|
16天前
|
存储 数据挖掘
服务器数据恢复—V7000存储上多块Mdisk成员盘出现故障的数据恢复案例
服务器存储数据恢复环境: 一台V7000存储上共12块SAS机械硬盘(其中1块是热备盘),组建了2组Mdisk,创建了一个pool。挂载在小型机上作为逻辑盘使用,小型机上安装的AIX+Sybase。 服务器存储故障: V7000存储中磁盘出现故障,管理员发现问题后立即更换磁盘。新更换的硬盘在上线同步数据的时候,存储上另一块磁盘也出现问题,导致逻辑盘无法挂接在小型机上,业务暂时中断。V7000存储的管理界面上显示两块硬盘故障脱机。 pool无法加载,其中三个通用卷均无法挂载。
|
25天前
|
安全 数据挖掘
服务器数据恢复—RAID5阵列中两块硬盘离线导致阵列崩溃的数据恢复案例
服务器数据恢复环境: 两组分别由4块SAS接口硬盘组建的raid5阵列,两组raid5阵列划分LUN并由LVM管理,格式化为EXT3文件系统。 服务器故障: RAID5阵列中一块硬盘未知原因离线,热备盘自动激活上线替换离线硬盘。在热备盘上线过程中,raid5阵列中又有一块硬盘离线。热备盘同步失败,该raid阵列崩溃,LVM结构变得不完整,文件系统无法正常使用。
|
1月前
|
存储 监控 调度
云服务器成本优化深度解析与实战案例
本文深入探讨了云服务器成本优化的策略与实践,涵盖基本原则、具体策略及案例分析。基本原则包括以实际需求为导向、动态调整资源、成本控制为核心。具体策略涉及选择合适计费模式、优化资源配置、存储与网络配置、实施资源监控与审计、应用性能优化、利用优惠政策及考虑多云策略。文章还通过电商、制造企业和初创团队的实际案例,展示了云服务器成本优化的有效性,最后展望了未来的发展趋势,包括智能化优化、多云管理和绿色节能。
|
1月前
|
存储 运维 数据挖掘
服务器数据恢复—EVA存储中多块硬盘离线导致存储崩溃的数据恢复案例
一台HP EVA存储中有23块硬盘,挂接到一台windows server操作系统的服务器。 EVA存储上有三个硬盘指示灯亮黄灯,此刻存储还能正常使用。管理员在更换硬盘的过程中,又出现一块硬盘对应的指示灯亮黄灯,存储崩溃,无法使用了。
|
17天前
|
存储 数据挖掘
服务器数据恢复—ZFS文件系统下数据恢复案例
服务器存储数据恢复环境: ZFS Storage 7320存储阵列中有32块硬盘。32块硬盘分为4组,每组8块硬盘,共组建了3组RAIDZ,每组raid都配置了热备盘。 服务器存储故障: 服务器存储运行过程中突然崩溃,排除人为误操作、断电、进水和其他机房不稳定因素。管理员重启服务器存储,系统无法进入,需要恢复服务器存储中的数据。
|
1月前
|
存储 Oracle 关系型数据库
服务器数据恢复—华为S5300存储Oracle数据库恢复案例
服务器存储数据恢复环境: 华为S5300存储中有12块FC硬盘,其中11块硬盘作为数据盘组建了一组RAID5阵列,剩下的1块硬盘作为热备盘使用。基于RAID的LUN分配给linux操作系统使用,存放的数据主要是Oracle数据库。 服务器存储故障: RAID5阵列中1块硬盘出现故障离线,热备盘自动激活开始同步数据,在同步数据的过程中又一块硬盘离线,RAID5阵列瘫痪,上层LUN无法使用。