关于cv::imread读取图片类型的初探[通俗易懂]

关于cv::imread读取图片类型的初探[通俗易懂]关于cv::imread读取图片类型的初探问题来源环境首先生成单通道和三通道的png图片cv::imread函数及其参数不同参数读取rgb图像不同参数读取单通道图片问题来源在处理深度图的时候,在用cv::imread读取深度图像时,本以为得到的是单通道图,但实际是三通道图。所以仔细看了一下cv::imread函数。环境Ubuntu16Opencv4.0.0首先生成单通道和三通…

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE稳定放心使用

问题来源

在处理深度图的时候,在用 cv::imread 读取深度图像时,本以为得到的是单通道图,但实际是三通道图。所以仔细看了一下 cv::imread 函数。

环境

Ubuntu16
Opencv 4.0.0

首先生成单通道和三通道的png图片

#include <iostream>
#include <opencv2/core.hpp>
#include <opencv2/highgui.hpp>
using namespace std;
int main(){
	cv::Mat mat1(480,480,CV_8UC3,cv::Scalar(255,128,0));
	cv::imshow("mat1",mat1);
	cv::Mat mat2(480,480,CV_8UC1,cv::Scalar(128));
	cv::imshow("mat2",mat2);
	cv::imwrite("mat1.png",mat1);
	cv::imwrite("mat2.png",mat2);
	cv::waitKey(0);
	return 1;
}

两张图片,一张三通道图片有颜色,一张单通道图片,无颜色。保存成PNG格式。文件大小分别为2.3KB和1.3KB。
在这里插入图片描述

cv::imread函数及其参数

Mat cv::imread (
	const String & filename, 
	int flags = IMREAD_COLOR 
)

enum cv::ImreadModes{
	IMREAD_UNCHANGED,			//-1   使图像保持原样输出  
	IMREAD_GRAYSCALE,			//0   把图像转成单通道的灰度图输出
	IMREAD_COLOR ,				//1	//把图像转成三通道的rgb图输出
	IMREAD_ANYDEPTH, 			//2   //If set, return 16-bit/32-bit image when the input has the corresponding depth, otherwise convert it to 8-bit.
	IMREAD_ANYCOLOR	,			//4   //以任何可能的颜色格式读取图像
	IMREAD_LOAD_GDAL, 			//8 	//use the gdal driver for loading the image
	IMREAD_REDUCED_GRAYSCALE_2,	//16	//输出单通道灰度图,并且将图像缩小为原来的1/2
	IMREAD_REDUCED_COLOR_2 ,	//17  //输出三通道的rgb图,并且缩小图像到原来的1/2
	IMREAD_REDUCED_GRAYSCALE_4, //32  //单通道  1/4
	IMREAD_REDUCED_COLOR_4 ,	//33	//三通道  1/4
	IMREAD_REDUCED_GRAYSCALE_8, //64	//单通道  1/8
	IMREAD_REDUCED_COLOR_8 ,	//65	//三通道  1/8
	IMREAD_IGNORE_ORIENTATION 	//128	//do not rotate the image according to EXIF's orientation flag. 
}

对我们有意义的有参数-1,0,1。
而参数IMREAD_COLOR 默认值为1。
下面是一些具体例子而已。

不同参数读取rgb图像

#include <iostream>
#include <opencv2/core.hpp>
#include <opencv2/highgui.hpp>
#include <opencv2/opencv.hpp>
using namespace std;
int main(){
	cv::Mat mat1 = cv::imread("mat1.png");
	cout<<mat1.type()<<endl;
	cout<<mat1(cv::Range(0,3),cv::Range(0,3))<<endl;

	cv::Mat mat1_0 = cv::imread("mat1.png",0);
	cout<<mat1_0.type()<<endl;
	cout<<mat1_0(cv::Range(0,3),cv::Range(0,3))<<endl;

	cv::Mat mat1_1;
	cv::cvtColor(mat1,mat1_1,cv::COLOR_RGB2GRAY);
	cout<<mat1_1.type()<<endl;
	cout<<mat1_1(cv::Range(0,3),cv::Range(0,3))<<endl;

	return 1;
}

输出

16
[255, 128,   0, 255, 128,   0, 255, 128,   0;
 255, 128,   0, 255, 128,   0, 255, 128,   0;
 255, 128,   0, 255, 128,   0, 255, 128,   0]
0
[104, 104, 104;
 104, 104, 104;
 104, 104, 104]
0
[151, 151, 151;
 151, 151, 151;
 151, 151, 151]

单通道读取不等于直接把图像转为灰度图。

附 cv::Mat.type()

		C1 	C2 	C3 	C4 
CV_8U   0   8   16	24
CV_8S   1   9	17	25
CV_16U  2   10	18	26
CV_16S  3   11	19	27
CV_32S  4   12	20	28
CV_32F  5   13	21	29
CV_64F  6   14	22	30 

不同参数读取单通道图片

#include <iostream>
#include <opencv2/core.hpp>
#include <opencv2/highgui.hpp>
#include <opencv2/opencv.hpp>
using namespace std;
int main(){

	cv::Mat mat2 = cv::imread("mat2.png");
	cout<<mat2.type()<<endl;
	cout<<mat2(cv::Range(0,3),cv::Range(0,3))<<endl;

	cv::Mat mat2_00 = cv::imread("mat2.png",-1);
	cout<<mat2_00.type()<<endl;
	cout<<mat2_00(cv::Range(0,3),cv::Range(0,3))<<endl;

	cv::Mat mat2_0 = cv::imread("mat2.png",0);
	cout<<mat2_0.type()<<endl;
	cout<<mat2_0(cv::Range(0,3),cv::Range(0,3))<<endl;

	cv::Mat mat2_1 = cv::imread("mat2.png",1);
	cout<<mat2_1.type()<<endl;
	cout<<mat2_1(cv::Range(0,3),cv::Range(0,3))<<endl;


	return 1;
}

输出:

16
[128, 128, 128, 128, 128, 128, 128, 128, 128;
 128, 128, 128, 128, 128, 128, 128, 128, 128;
 128, 128, 128, 128, 128, 128, 128, 128, 128]
0
[128, 128, 128;
 128, 128, 128;
 128, 128, 128]
0
[128, 128, 128;
 128, 128, 128;
 128, 128, 128]
16
[128, 128, 128, 128, 128, 128, 128, 128, 128;
 128, 128, 128, 128, 128, 128, 128, 128, 128;
 128, 128, 128, 128, 128, 128, 128, 128, 128]

参数-1和0是期望的输出。
参数1也按照预期进行了复制。
无参数时,参数并非默认-1 。

当我打出默认参数的时候,才注意到我只需要找到默认参数cv::IMREAD_COLOR(见“cv::imread函数及其参数”),而打印出来是1。

遇到的一些情况

以上笔者读取的是自己制作的图片,但在项目中碰到的一张深度图,如图
在这里插入图片描述
(在此显示不太清楚)
在-1参数下读取的type为2,也即是16UC1。
总结起来,在读取图像后,需要确认读取格式和自己预期是否相同。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/182233.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 原地算法矩阵置0_矩阵归一化处理步骤

    原地算法矩阵置0_矩阵归一化处理步骤给定一个 m x n 的矩阵,如果一个元素为 0 ,则将其所在行和列的所有元素都设为 0 。请使用 原地 算法。进阶:一个直观的解决方案是使用 O(mn) 的额外空间,但这并不是一个好的解决方案。一个简单的改进方案是使用 O(m + n) 的额外空间,但这仍然不是最好的解决方案。你能想出一个仅使用常量空间的解决方案吗?示例 1:输入:matrix = [[1,1,1],[1,0,1],[1,1,1]]输出:[[1,0,1],[0,0,0],[1,0,1]]示例 2:输入:matrix

    2022年8月8日
    6
  • navicat mac激活码-激活码分享

    (navicat mac激活码)这是一篇idea技术相关文章,由全栈君为大家提供,主要知识点是关于2021JetBrains全家桶永久激活码的内容https://javaforall.net/100143.htmlIntelliJ2021最新激活注册码,破解教程可免费永久激活,亲测有效,上面是详细链接哦~23LNPMIJZT-eyJsaWNlb…

    2022年3月29日
    97
  • Predicate接口

    Predicate接口Predicate 接口的默认方法 可以实现条件中的与 amp amp 或 非 判断 通过默认方法实现 importjava util function Predicate Predicate 接口 条件判断 对某种数据类型进行判断 从而得到一个 boolean 值结果 publiccl

    2025年6月19日
    3
  • Ubuntu下内核编程之第一个模块helloworld[通俗易懂]

    Ubuntu下内核编程之第一个模块helloworld[通俗易懂]模块是驱动开发的必经之路,这也是我们在前边要构建内核源码树的目的所在。因为模块属于kernel编程了,和用户空间的c语言变成不同。他要依附于内核源码树的存在而存在,下面就开始我们的第一个模块的编写吧。     Ubuntu下内核编程之第一个模块——我的叫jun_module。随你怎么起名。以下是建立的全过程:     1、首先肯定是编写模块的源代码,和Makefile。我们建立一个专用

    2022年10月8日
    3
  • Ubuntu中搭建ICE服务器(Coturn)

    Ubuntu中搭建ICE服务器(Coturn)1.WebRTC的P2P穿透WebRTC的P2P穿透部分是由libjingle实现的.步骤顺序大概是这样的:尝试直连.通过STUN服务器进行穿透无法穿透则通过TURN服务器中转STUN服务器比较简单.网上也有很多公开的STUN服务器可以用于测试,例如:stun.ideasip.com在WebRTC的P2P应用中,使用公开的STUN服务器时,有时响应比较慢,这就需要自己搭一个…

    2022年6月14日
    34
  • java中的scanner是什么_java中的Scanner类是什么?如何使用?「建议收藏」

    java中的scanner是什么_java中的Scanner类是什么?如何使用?「建议收藏」java中的Scanner类是什么?如何使用?发布时间:2020-05-2016:36:48来源:亿速云阅读:204作者:鸽子Scanner类介绍java.util.Scanner是Java5的新特征,可以通过Scanner类来获取用户的输入。创建Scanner对象的基本语法:Scanners=newScanner(System.in);实例:接下来我们演示一个最简单的数据…

    2022年7月8日
    22

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号