opencv笔记2:图像ROI
time:2015年 10月 03日 星期六 12:03:45 CST
opencv笔记2:图像ROI
ROI
ROI意思是Region Of Interests,感兴趣区域,是一个图中的一个子区域。
OpenCV中定义的ROI是矩形的。
ROI的用处包括而不限于:提取出ROI区域做进一步处理(比如人脸识别、车牌识别);将另一张图片贴放到ROI区域。
这里以第二种用处为例,将一个logo图像添加到一张大图上指定的ROI区域。
图像贴放
粗略想想,包括这四个步骤
- 定义大图和小图
- 在大图上定义ROI区域
- 小图贴放到ROI区域
- 显示新的大图
这里在fedora22下使用dnf(yum)安装了opencv3.0,如下代码可以运行良好
#include <iostream>
#include <opencv2/opencv.hpp>
using namespace std;
using namespace cv;
int main() {
//【1】定义大图和小图
//【2】在大图上定义ROI区域
//【3】小图贴放到ROI区域
//【4】显示新的大图
//【1】定义大图和小图
Mat srcImage = imread("/home/chris/workspace/clion/dota_pa.jpg");
Mat logoImage = imread("/home/chris/workspace/clion/dota_ss.png");
if(!srcImage.data){
cerr << "读取srcImage错误!" << endl;
exit;
}
if(!logoImage.data){
cerr << "读取logoImage错误!" << endl;
exit;
}
//【2】在大图上定义ROI区域
Mat imageROI = srcImage(Rect(20, 25, logoImage.cols, logoImage.rows));
//【3】小图贴放到ROI区域
logoImage.copyTo(imageROI);
//【4】显示新的大图
imshow("利用ROI实现图像叠加示例窗口", srcImage);
waitKey(0);
destroyAllWindows();
return 0;
}
这里copyTo()函数中只用到一个参数,浅墨的教程上有第二个参数:图像掩模(mask),我尝试了下,发现不用mask也没关系,mask的定义也不一定要是灰度图(即imread第二个参数设定为0)。对此,我表示不理解,反正能用就好了。
图像线性混合
直接把logo图替换掉ROI区域的做法通常不是很好,如果能把两者各按一定比例进行混合就显得更好些,这需要用到图像混合技术。
图像线性混合,其实就是不同图片(这里以两张图为例),各自按照一定比例系数进行相加,没什么神奇的地方:
result = alpha*src1 + beta*src2 + gamma
使用函数addWeighted(src1, alpha, src2, beta, gamma, dstImage, dtype=-1)
这里dtype参数,是输出阵列的可选深度,有默认值-1。;当两个输入数组具有相同的深度时,这个参数设置为-1(默认值),即等同于src1.depth()。
OK,粗略想下线性混合的步骤:
- 定义两幅图像
- 定义权重和偏置量等参数
- 按参数做图像混合
- 显示结果
上代码:
#include <iostream>
#include <opencv2/opencv.hpp>
using namespace std;
using namespace cv;
int main() {
//【1】定义两幅图像
//【2】定义权重和偏置量等参数
//【3】按参数做图像混合
//【4】显示结果
//【1】定义两幅图像
Mat srcImage1 = imread("/home/chris/workspace/clion/mogu.jpg");
Mat srcImage2 = imread("/home/chris/workspace/clion/rain.jpg");
//【2】定义权重和偏置量等参数
double alpha = 0.5; //第一幅图的权重
double beta = 1- alpha; //第二幅图的权重
double gamma = 0.0; //偏置量
//【3】按参数做图像混合
Mat dstImage; //目标图像,存放结果
addWeighted(srcImage1, alpha, srcImage2, beta, gamma, dstImage);
//【4】显示结果
imshow("线性混合结果", dstImage);
waitKey(0);
destroyAllWindows();
return 0;
}
好吧,其实这东西很简单。应用到ROI区域,对应的代码为:
#include <iostream>
#include <opencv2/opencv.hpp>
using namespace std;
using namespace cv;
int main() {
//【1】定义两幅图像
//【2】定义ROI区域
//【3】定义权重和偏置量等参数
//【4】按参数做图像混合
//【5】显示结果
//【1】定义两幅图像
Mat srcImage1 = imread("/home/chris/workspace/clion/dota_pa.jpg");
Mat srcImage2 = imread("/home/chris/workspace/clion/dota_ss.png");
//【2】定义ROI区域
Mat imageROI = srcImage1(Rect(20, 25, srcImage2.cols, srcImage2.rows));
//【3】定义权重和偏置量等参数
double alpha = 0.5; //第一幅图的权重
double beta = 1- alpha; //第二幅图的权重
double gamma = 0.0; //偏置量
//【4】按参数做图像混合
addWeighted(imageROI, alpha, srcImage2, beta, gamma, imageROI);
//【5】显示结果
imshow("线性混合结果", srcImage1);
waitKey(0);
destroyAllWindows();
return 0;
}
多通道的分离和混合
既然前面把图像混合讲到了,那么按照通道进行混合也就可以做了!比如要把看起来是白色的logo图,和指定背景图上ROI区域混合,并且希望logo图显示为红色。这就需要ROI图本身为红色通道,logo图读取为灰度图像(保证都是单通道),然后混合,这之后由于ROI区域就是原图中的一个通道上的一个区域,原图像的红色通道变化了,现在把三个通道重新合并回去,就能够得到想要的效果图了。
粗略想下,步骤为:
- 定义义两幅图像
- 分离通道
- 在通道图上定义ROI区域
- 定义权重和偏置量等参数
- 按参数做图像混合
- 合并通道
- 显示结果
对应的代码为:
#include <iostream>
#include <opencv2/opencv.hpp>
using namespace std;
using namespace cv;
int main() {
//义两幅图像
//【2】分离通道
//【3】在通道图上定义ROI区域
//【4】定义权重和偏置量等参数
//【5】按参数做图像混合
//【6】合并通道
//【7】显示结果
//【1】定义两幅图像
Mat srcImage = imread("/home/chris/workspace/clion/dota_pa.jpg");
Mat logoImage = imread("/home/chris/workspace/clion/dota_logo.jpg", 0);
//【2】分离通道
vector<Mat> channels;
split(srcImage, channels);
//【3】在通道图上定义ROI区域
Mat redChannelImage = channels.at(2);
Mat imageROI = redChannelImage(Rect(50, 25, logoImage.cols, logoImage.rows));
//【4】定义权重和偏置量等参数
double alpha = 1.0;
double beta = 0.5;
double gamma = 0.0;
//【5】按参数做图像混合
addWeighted(imageROI, alpha, logoImage, beta, gamma, imageROI);
//【6】合并通道
merge(channels, srcImage);
//【7】显示结果
imshow("0通道", srcImage);
waitKey(0);
destroyAllWindows();
return 0;
}
opencv笔记2:图像ROI的更多相关文章
- Python下opencv使用笔记(图像频域滤波与傅里叶变换)
Python下opencv使用笔记(图像频域滤波与傅里叶变换) 转载一只程序喵 最后发布于2018-04-06 19:07:26 阅读数 1654 收藏 展开 本文转载自 https://blog ...
- OpenCV学习笔记(3)——图像的基本操作
获取图像的像素值并修改 获取图像的属性(信息) 图像的ROI() 图像通道的拆分及合并 1.获取并修改像素值 先读入图像装入一个图像实体,然后该实体相当于一个多维list,可以直接用数组操作提取像素信 ...
- opencv笔记6:角点检测
time:2015年10月09日 星期五 23时11分58秒 # opencv笔记6:角点检测 update:从角点检测,学习图像的特征,这是后续图像跟踪.图像匹配的基础. 角点检测是什么鬼?前面一篇 ...
- OpenCV笔记大集锦(转载)
整理了我所了解的有关OpenCV的学习笔记.原理分析.使用例程等相关的博文.排序不分先后,随机整理的.如果有好的资源,也欢迎介绍和分享. 1:OpenCV学习笔记 作者:CSDN数量:55篇博文网址: ...
- opencv笔记5:频域和空域的一点理解
time:2015年10月06日 星期二 12时14分51秒 # opencv笔记5:频域和空域的一点理解 空间域和频率域 傅立叶变换是f(t)乘以正弦项的展开,正弦项的频率由u(其实是miu)的值决 ...
- opencv笔记4:模板运算和常见滤波操作
time:2015年10月04日 星期日 00时00分27秒 # opencv笔记4:模板运算和常见滤波操作 这一篇主要是学习模板运算,了解各种模板运算的运算过程和分类,理论方面主要参考<图像工 ...
- opencv笔记3:trackbar简单使用
time:2015年 10月 03日 星期六 13:54:17 CST # opencv笔记3:trackbar简单使用 当需要测试某变量的一系列取值取值会产生什么结果时,适合用trackbar.看起 ...
- opencv笔记1:opencv的基本模块,以及环境搭建
opencv笔记1:opencv的基本模块,以及环境搭建 安装系统 使用fedora22-workstation-x86_64 安装opencv sudo dnf install opencv-dev ...
- opencv-学习笔记(6)图像梯度Sobel以及canny边缘检测
opencv-学习笔记(6)图像梯度Sobel以及canny边缘检测 这章讲了 sobel算子 scharr算子 Laplacion拉普拉斯算子 图像深度问题 Canny检测 图像梯度 sobel算子 ...
随机推荐
- 二分匹配之最大权值匹配算法---KM模板
百科:http://baike.baidu.com/link?url=vbM3H4XmfrsWfP-epdlR2sVKSNzOq4hXnWDqm5uo8fd7VWsF2SmhDV35XyVUDvVjv ...
- J2EE中getParameter与getAttribute以及EL表达式${requestScope}和${param[]}
getParameter ① 得到的都是String类型的.如http://name.jsp?name=xy中的xy ② 获取POST/GET传递的参数值 ③ 用于客户端重定向,如点击链接或提交按扭时 ...
- AC日记——热浪 codevs 1557 (最短路模板题)
1557 热浪 时间限制: 1 s 空间限制: 256000 KB 题目等级 : 钻石 Diamond 题解 查看运行结果 题目描述 Description 德克萨斯纯朴的民眾们这个夏 ...
- 使用List的addAll()方法请判空指针
在写代码的时候经常会用到List,Set的addAll()方法,但是要注意addAll()方法不能传入空指针. package link.mengya.utils; import link.mengy ...
- linux下清除Squid缓存的方法记录
在日常运维工作中,只要用到squid缓存服务,就会常常被要求清理squid缓存.比如公司领导要求删一篇新闻,新闻是生成的静态.运维人员把服务器上静态的新闻页面删除了后,不料代理服务器上缓存还有.缓存服 ...
- 05SpringMvc_映射器SimpleUrlHanderMapping
这篇文章讲的还是映射器,映射器类有两种,前一篇文章讲的是BeanNameUrlHanderMapping映射器类.今天讲的是SimpleUrlHanderMapping映射器类. 这两个映射器类有什么 ...
- ES配置文件参考与参数详解
cluster.name: data-cluster node.name: "data-es-05" #node.data: false # Indexing & Cach ...
- Console的使用——Google Chrome代码调试
Google Chrome控制台为开发者提供了网页和应用程序调试的几种方法,本文通过基本操作.控制台API.命令行API来介绍控制台的使用. 基本操作 1.开启控制台 可以通过下列三种方式开启 ...
- tkinter 改变按钮状态
import tkinter as tk def btn1_change_btn1(event): '''方式一:通过事件控制自己''' if event.widget['state'] == 'no ...
- PC网站应用接入微信登录
参考文档: https://open.weixin.qq.com/cgi-bin/showdocument?action=dir_list&t=resource/res_list&ve ...