C++下OpenCV学习笔记 ----图像叠加与混合

C++下OpenCV学习笔记

----图像叠加与混合

一.感兴趣区域:ROI
  1. 定义方法
    (1)Rect定义法
Mat imageRoi;
imageRoi = image(Rect(500, 250, image.cols, image.rows));

        (2)Range定义法

Mat imageRoi;
imageRoi = image(Range(250, 250 + image.rows), Range(200, 200 + image.cols));
  1. 掩膜(mask)与感兴趣区域(ROI)
    (1)定义
    用预先制作的感兴趣区域掩膜与待处理图像相乘,得到感兴趣区域图像,感兴趣区域内图像值保持不变,而区外图像值都为0。
    (2)代码实现
#include<opencv2/opencv.hpp>
using namespace cv;
int main()
{
	Mat srcImage1 = imread("C://Users//441//Desktop//ZL//夏目//1.jpg");
	Mat srcImage2 = imread("C://Users//441//Desktop//ZL//夏目//th(2).jpg");
	Mat imageRoi = srcImage1(Rect(0, 277, srcImage2.cols, srcImage2.rows));
	Mat mask = imread("C://Users//441//Desktop//ZL//夏目//th(2).jpg", 0);
	srcImage2.copyTo(imageRoi, mask);
	namedWindow("ROI图像叠加");
	imshow("ROI图像叠加", srcImage1);
	waitKey(0);
}
srcImage1的尺寸为820x461;
srcImage2的尺寸为252x184.

        (3)运行结果

二.线性混合操作
  1. 原理
    G(x) = (1 - a)F(x) + aQ(x)

>>a的取值范围为0到1之间
>>F(x)和Q(x)为参与混合的两幅图像,G(x)表示输出图像
>>通过对两幅图像的每个像素值做线性加权得到最终的输出图像
>>两幅图像的大小和类型必须完全一致

  1. 实现
    主要运用addWeighted函数实现。
三.计算数组的加权和:addWeighted函数

---->初识addWeighted函数

  1. 原理
    void addWeighted(InputArray src1,double alpha,InputArray src2,double beta,double gamma,OutputArray dst,int dytpe=-1);
    addWeighted函数计算两个数组(src1和src2)的加权和:
    dst = src1[I] * alpha + src[2] * beta + gamma;

>>I为多维数组的索引值
>>当输出数组深度为CV_32S时,函数不适用,内存溢出或结果错误。

  1. 当两幅图片尺寸不同时的叠加
    (1)代码实现
#include <opencv2/opencv.hpp>
using namespace cv;

int main()
{
    Mat src1 = imread("C:\\Users\\441\\Desktop\\ZL\\夏目\\1.jpg");
    Mat src2 = imread("C:\\Users\\441\\Desktop\\ZL\\夏目\\th(2).jpg");
    Mat dst;

    double alpha = 0.3;
    double beta;
    beta = (1.0 - alpha);

    Mat imageRoi = src1(Rect(0, 277, src2.cols, src2.rows));
    addWeighted(src2, alpha, imageRoi, beta, 0, imageRoi);

    imshow("图2", src2);
    imshow("图2`", imageRoi);
    imshow("mixed", src1);
    waitKey(0);
    return 0;
}

        (2)运行结果

四.颜色通道的分离与合并
  1. 通道分离:split( )函数
    (1)原型
    void split(const Mat& src,Mat *mvBegin);

第一个参数:表示要分离颜色通道的矩阵
第二个参数:表示Mat数组的首地址

       void split(InputArray m, OutputArrayOfArrays mv);

第一个参数:表示要分离颜色通道的矩阵
第二个参数:表示一个vector<Mat>对象

        (2)代码实现

#include<opencv2/opencv.hpp>
using namespace cv;
int main()
{
	Mat src = imread("C:\\Users\\441\\Desktop\\ZL\\夏目\\色阶图.png");
	imshow("原图", src);
	Mat channels[3];
	split(src, channels);
	imshow("B", channels[0]);
	imshow("G", channels[1]);
	imshow("R", channels[2]);
	waitKey(0);
}
#include<opencv2/opencv.hpp>
using namespace cv;
int main()
{
	Mat src = imread("C:\\Users\\441\\Desktop\\ZL\\夏目\\色阶图.png");
	imshow("原图", src);
	std::vector<Mat> channels;
	split(src, channels);
	imshow("B", channels.at(0));
	imshow("G", channels.at(1));
	imshow("R", channels.at(2));
	waitKey(0);
}

        (3)运行结果

  1. 通道合并:merge( )函数
    (1)原型
    void merge(const Mat* mv, size_tcount, OutputArray dst);

第一个参数:表示要被合并的矩阵
第二个参数:表示输入矩阵的个数(当输入矩阵为空白的数组时,必须大于1)
第三个参数:表示输出的矩阵(与原矩阵拥有相同尺寸和深度,通道数量是矩阵阵列中的通道总数)

       void merge(InputArrayOfArrays mv,OutputArray dst);

第一个参数:表示vector容器的阵列
第二个参数:表示输出的矩阵(与原矩阵拥有相同尺寸和深度,通道数量是矩阵阵列中的通道总数)

        (2)代码实现

#include<opencv2/opencv.hpp>
using namespace cv;
int main()
{
	Mat src = imread("C:\\Users\\441\\Desktop\\ZL\\夏目\\色阶图.png");
	imshow("原图", src);
	Mat channels[3];
	split(src, channels);
	
	Mat blueChannels = channels[0];
	Mat greenChannels = channels[1];
	Mat redChannels = channels[2];

	Mat newsChannels[3] = { blueChannels,greenChannels,redChannels };
	Mat mergeImage;
	merge(newsChannels, 3, mergeImage);
	imshow("merge", mergeImage);
	waitKey(0);
}
#include<opencv2/opencv.hpp>
using namespace cv;
int main()
{
	Mat src = imread("C:\\Users\\441\\Desktop\\ZL\\夏目\\色阶图.png");
	imshow("原图", src);
	std::vector<Mat> channels;
	Mat blueChannels;
	Mat greenChannels;
	Mat redChannels;
	
	split(src, channels);
	blueChannels = channels.at(0);
	greenChannels = channels.at(1);
	redChannels = channels.at(2);

	Mat mergeImage;
	merge(channels, mergeImage);
	imshow("merge", mergeImage);
	waitKey(0);
}

        (3)运行结果

五.图像的亮度和对比度
  1. 定义
    (1)亮度:图像亮度指的是图像的明暗程度。
    (灰度图像,则跟灰度值有关,灰度值越高则图像越亮)
    (2)对比度:图像对比度指的是一幅图像中明暗区域最亮的白和最暗的黑之间不同亮度层级的测量,即指一幅图像灰度反差的大小。
    (灰度图像,设像素值的范围为[a,b],若b-a的值越接近于255,图像对比度越大,看上去图像更清晰;越接近于0,图像对比度越小,图像越不清晰。)
  2. 原理
    g(i, j) = a * f(i, j) + b

>>i和j表示像素位于第i行和第j列
>>f(i, j)表示源图像像素
>>g(i, j)表示输出图像像素
>>a(a > 0)表示增益(gain),常用来控制图像的对比度
>>b表示偏置(bias),常用来控制图像的亮度

 (1)当α=1,β=0时,图像无任何改变;
 (2)当α<1,β=0时,图像对比度下降,图像变暗;
 (3)当α>1,β=0时,图像对比度上升,图像变生动、丰富;
 (4)β上下浮动,图像亮度发生改变,不改变图像对比度;
  1. 访问图像的每一个像素
    ---->系统讲解:访问图像像素的三种方法
Mat new_image = Mat::zeros(image.size(), image.type());
for(int y = 0; y < image.rows; y++)
	for(int x = 0; x < image.cols; x++)
		for(int c = 0; c < 3; c++)
			new_image.at<Vec3b>(y, x)[c] = saturate_cast<uchar>((g_nContrastValue*0.01)*(image.at<Vec3b>(y, x)[c]) + g_nBrightValue); 

>>y为行,x为列,c为R、G、B(分别对应0, 1, 2)其中之一
>>因为运算结果可能会超出像素取值范围 (溢出),或是非整数(如果是浮点数的话),用saturate_cast对结果进行转换,以确保它为有效值
if(data < 0)
        data = 0;
else if(data > 255)
        data = 255;

  1. 代码实现

```clike
#include<opencv2/opencv.hpp>
using namespace cv;
using namespace std;
void onChange(Mat image, float g_nContrastValue, float g_nBrightValue)
{
	Mat new_image = Mat::zeros(image.size(), image.type());
	for (int y = 0; y < image.rows; y++)
		for (int x = 0; x < image.cols; x++)
			for (int c = 0; c < 3; c++)
				new_image.at<Vec3b>(y, x)[c] = saturate_cast<uchar>(g_nContrastValue * (image.at<Vec3b>(y, x)[c]) + g_nBrightValue);
		
	imshow("效果图", new_image);
	waitKey(0);
}
int main()
{
	Mat src = imread("C:\\Users\\441\\Desktop\\ZL\\夏目\\1.jpg");
	imshow("原图", src);

	float g_nContrastValue, g_nBrightValue;
	g_nContrastValue = 10.0;
	g_nBrightValue = 0;
	onChange(src, g_nContrastValue, g_nBrightValue);

	g_nContrastValue = 1.0;
	g_nBrightValue = 30;
	onChange(src, g_nContrastValue, g_nBrightValue);

	g_nContrastValue = 10.0;
	g_nBrightValue = 30;
	onChange(src, g_nContrastValue, g_nBrightValue);

	waitKey(0);
}

5. 运行结果
`原图`
 <img src="https://img-blog.csdnimg.cn/20210220174552926.png"   width="50%" >
 `更改对比度后的效果图`
 <img src="https://img-blog.csdnimg.cn/20210220174836958.png"   width="50%" >
  `更改亮度后的效果图`
 <img src="https://img-blog.csdnimg.cn/20210220174935689.png"   width="50%" >
  `更改对比度和亮度后的效果图`
 <img src="https://img-blog.csdnimg.cn/20210220175011212.png"   width="50%" >







  • 9
    点赞
  • 47
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

X to Y

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

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

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

打赏作者

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

抵扣说明:

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

余额充值