关于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)
上一篇 2022年10月14日 上午6:36
下一篇 2022年10月14日 上午6:46


相关推荐

  • Python实现TXT、CSV、XLS等格式转换 and 图像显示(超详细教程)

    Python实现TXT、CSV、XLS等格式转换 and 图像显示(超详细教程)

    2020年11月8日
    272
  • deepseek如何本地化部署?新手必学的DeepSeek本地部署教程

    deepseek如何本地化部署?新手必学的DeepSeek本地部署教程

    2026年3月16日
    6
  • LGPL中文版

    LGPL中文版两水平线内为原出处 txt 文件完整版 该篇翻译文章原出处为 http gnu freehostingg com lgpl GNU 较宽松公共许可证 简体中文翻译版 nbsp 声明 这是一份 GNU 较宽松公共许可证非正式的中文翻译 它不是自由软体基金会所发布 并且不能适用于使用 GNULGPL 的软体 只有 GNULGPL 英文原文的版本才行 然而 我们希望这份翻译能帮助中文的使用者

    2026年3月16日
    1
  • 解决方案:VS2017 无法打开源文件 stdio.h main.h 等头文件[通俗易懂]

    解决方案:VS2017 无法打开源文件 stdio.h main.h 等头文件[通俗易懂]问题描述:在VS2017中运行解决方案是有错误:“E1696 无法打开源文件“stdio.h” ”…原因:这种问题一般发生在该项目代码是在网上下载而来的情况,或者电脑重装新的系统等情况,导致电脑系统与该项目生成时所采用的windowsSDK不同,从而在默认的位置(已发生变化)找不到许多源文件。解决方案:1.在C++项目处(示例为“Fibonacci”),鼠标右击,弹出的菜…

    2022年6月16日
    82
  • 人工势场法matlab讲解_【机器人路径规划】人工势场法

    人工势场法matlab讲解_【机器人路径规划】人工势场法阅读本文需要的基础知识为:理解机器人的构型空间。建议阅读:机器人运动规划中的Cspace怎样理解?为什么不直接在笛卡尔坐标系下运算呢?本文的实现程序与使用说明见我的学习工具箱:小明工坊:【个人开源】机器人运动规划学习工具箱使用说明基本原理1.概述我们打两个比方来说明人工势场法的作用机理。首先,我们把构型空间比作一个电势场平面,机器人(的当前构型)比作空间中一点。如果让机器人的起点和障碍物带正…

    2022年6月16日
    28
  • 用python高精度求自然常数e

    用python高精度求自然常数e用 python 高精度求自然常数 e 泰勒展开与 e 的求法 python 和神奇的 decimal 计算比较完整代码泰勒展开与 e 的求法大家伙儿知道计算机里的 eee 是怎么求出来的吗 这还要从神奇的泰勒展开讲起 简单的说 就是 eee 可以表示成 e 10 11 12 1n e frac 1 0 frac 1 1 frac 1 2 cdots frac 1 n cdotse 0 1 1 1 2 1 n 1 很显然 当 nnn 足够小的时候

    2026年3月19日
    2

发表回复

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

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