Opencv的mat转换成qimage或者qpixmap

7 篇文章 0 订阅

目录

一、方法一

二、方法二

三、比较

方法一优缺点

方法二优缺点

一、方法一

本方法基本思路是把图片编码成某种格式图片的Buffer,然后QT或者OpenCV框架用该Buffer来构造出图片。

cv::Mat转成QImage和QPixmap

QImage MatToQImage(const Mat& cvImage) {
    vector<uchar> imgBuf;
    imencode(".bmp", cvImage, imgBuf);

    QByteArray baImg((char*)imgBuf.data(), static_cast<int>(imgBuf.size()));
    QImage image;
    image.loadFromData(baImg, "BMP");
    return image;
}

QPixmap MatToPixmap(const Mat& cvImage)
{
    return QPixmap::fromImage(matToImage(cvImage));
}

QImage和QPixmap转换cv::Mat 

Mat QImageToMat(const QImage image) {
    QByteArray ba;
    QBuffer buffer(&ba);
    buffer.open(QIODevice::WriteOnly);
    image.save(&buffer, "BMP");
    vector<uchar> imgBuf;
    return imencode(std::vector<uchar>(ba.begin(), ba.end()), IMREAD_COLOR);
}

Mat PixmapToMat(const QPixmap& image) {
    return imageToMat(image.toImage());
}

我把图转换成BMP格式而没有转换成JPG、PNG格式,主要考虑的是性能因素。编码成JPG和PNG格式,编码器需要更多的CPU计算才可以完成,但是消耗少量的内存;与之相反,转换成BMP消耗更多的内存,而节省大量的CPU计算。请根据你自己的约束选择正确的策略。

二、方法二

cv::Mat转成QImage

QImage MatToQImage(const cv::Mat& mat)
{
	if (mat.type() == CV_8UC1)
	{
		QImage image(mat.cols, mat.rows, QImage::Format_Indexed8);
		// Set the color table (used to translate colour indexes to qRgb values)
		image.setColorCount(256);
		for (int i = 0; i < 256; i++)
		{
			image.setColor(i, qRgb(i, i, i));
		}
		// Copy input Mat
		uchar* pSrc = mat.data;
		for (int row = 0; row < mat.rows; row++)
		{
			uchar* pDest = image.scanLine(row);
			memcpy(pDest, pSrc, mat.cols);
			pSrc += mat.step;
		}
		return image;
	}
	// 8-bits unsigned, NO. OF CHANNELS = 3
	else if (mat.type() == CV_8UC3)
	{
		// Copy input Mat
		const uchar* pSrc = (const uchar*)mat.data;
		// Create QImage with same dimensions as input Mat
		QImage image(pSrc, mat.cols, mat.rows, mat.step, QImage::Format_RGB888);
		return image.rgbSwapped();
	}
	else if (mat.type() == CV_8UC4)
	{
		// Copy input Mat
		const uchar* pSrc = (const uchar*)mat.data;
		// Create QImage with same dimensions as input Mat
		QImage image(pSrc, mat.cols, mat.rows, mat.step, QImage::Format_ARGB32);
		return image.copy();
	}
	else
	{
		//MessageInfo("ERROR: Mat could not be converted to QImage.", 1);
		//emit sig_RunInfo("ERROR: Mat could not be converted to QImage.", 1);
		//if (!globalPara.IsInlineRun) Runstateinfo("ERROR: Mat could not be converted to QImage.", 1);
		return QImage();
	}
}

QImage转成cv::Mat

cv::Mat QImageToMat(QImage image)
{
	cv::Mat mat;
	switch (image.format())
	{
	case QImage::Format_ARGB32:
	case QImage::Format_RGB32:
	case QImage::Format_ARGB32_Premultiplied:
		mat = cv::Mat(image.height(), image.width(), CV_8UC4, (void*)image.constBits(), image.bytesPerLine());
		break;
	case QImage::Format_RGB888:
		mat = cv::Mat(image.height(), image.width(), CV_8UC3, (void*)image.constBits(), image.bytesPerLine());
		cv::cvtColor(mat, mat, CV_BGR2RGB);
		break;
	case QImage::Format_Indexed8:
		mat = cv::Mat(image.height(), image.width(), CV_8UC1, (void*)image.constBits(), image.bytesPerLine());
		break;
	}
	return mat;
}

三、比较

方法一优缺点

优点

  1. 该方案不依赖任何第三方的代码。
  2. 该方案适应性比较好,能处理各种格式的图片。

缺点

  1. 该方案转换过程中需要编解码过程,性能会受到影响。
  2. 该方案需要Buffer来保存临时生成的图片,需要消耗更多的内存。

方法二优缺点

优点

  1. 生成QImage和QPixmap不需要重新申请内存,直接使用Mat的内存,效率比较高,并且节省内存。
  2. 所有代码只在一个头文件中,比较容易集成。

缺点

  1. 支持部分图片格式,适应性不好。
  • 4
    点赞
  • 33
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

TryRestart

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值