关于我们

质量为本、客户为根、勇于拼搏、务实创新

< 返回新闻公共列表

vs2019+Qt 使用 Qlabel 在界面上显示图像及显示失真问题

发布时间:2023-06-26 23:00:51

        在使用 Qt 设计界面时,通常会涉及到在界面上显示图片的问题,而要在界面上显示图片需要使用控件 Qlabel 和 函数 QImage ,下面对控件和函数逐一做出介绍!!!


一、Qlabel 常见成员方法


1、setText(const QString &text)  --------------  设置显示文本


2、void setAlignment(Qt::Alignment)  --------------  设置文本显示位置


3、void setFont(const QFont &)  --------------  设置字体


4、void setPixmap(const QPixmap &)  --------------  设置图片


5、void setMovie(QMovie *movie)  --------------  设置动图


二、QImage 函数介绍


1、QImage 函数基本定义:


QImage(uchar * data, int width, int height, int bytesPerLine, Format format, QImageCleanupFunction cleanupFunction = 0, void * cleanupInfo = 0)

   


2、QImage 函数形参介绍 


(1)data 直接使用 Mat 类型的 data 即可,如:image.data;


(2)width 表示图像的宽,即图像的列;


(3)height表示高,即图像的宽;


(4)bytesPerline 表示每行拥有的字节数(如果没有该参数则默认按照 4 字节对齐的方式显示,见另个构造函数);


(5)format 表示图片格式,彩色图一般采用Format_RGB888,灰度图则使用Format_indexed8;


注意:bytesPerline 参数如果设置不当可能会导致图片显示失真,见下图;

  图 1 原图 

图 2 显示失真


 三、代码和结果


1、代码实现


cv::Mat image = cv::imread("D:\\image.png");//要显示的影像  cvtColor(image, image, CV_BGR2RGB); //转换色彩空间,把RGB转为BGR  //把 Mat 转换成 QImage QImage img_1 = QImage((const unsigned char*)(image.data), image.cols, image.rows, image.cols * image.channels(), QImage::Format_RGB888);  ui.label->setPixmap(QPixmap::fromImage(img_1));  //设定 Label 尺寸 ui.label->resize(QSize(img_1.width(), img_1.height()));

   

说明:cv::Mat 存储图像通道顺序为:RGB,而 QImage 存储图像通道顺序为:BGR,所以在显示前需要进行通道上的转换!


2、显示结果


说明:QImage 函数中,如果 bytesPerline 参数不设置,则会默认按照 4 字节对齐的方式显示图像(如果不满足四字节对齐要求则不会使用零填充),对于不满足四字节对齐的影像在显示的时候就会出现上述失真问题,若将 bytesPerline 参数设置为 image.cols*image.channels() (如上述代码所示)则可解决图像显示失真问题(如上图显示结果)!!!


四、四字节对齐原理


 如果图像没有做字节对齐,则在对图像进行逐像素遍历的时候会出现差错,也即图像的失真。而 Opencv 中 Mat 矩阵的创建通常是默认没有字节对齐的,所以,如果使用 QImage 对 Mat矩阵进行显示需要进行字节对齐处理!


       每个像素所占字节数等于图像通道数,也即一个像素占 3 个字节(常规图像为三通道);本次实验案例使用的图像尺寸为:5528*3857 ,3857 为行、5528 为列,所以一行图像数据所占字节数为:5528*3 字节。


       所谓四字节对齐即看 5528*3 是否是 4 的整数倍,如果不是则需要使用零对其补齐,方能正常显示图像,而 QImage 函数中,如果不对 bytesPerline 参数设置,则默认以四字节对齐的方式显示图像,且对不满足四字节对齐的不使用零补充;如果把 bytesPerline 参数设置为 image.cols * image.channels() 即设置不满足四字节对齐时使用零补充,然后在显示不满足四字节对齐的图像时,方能正常显示。


/template/Home/leiyu/PC/Static