OpenCV C++案例实战十八《抖音特效——“蓝线挑战”》

OpenCV C++案例实战十八《抖音特效——“蓝线挑战”》


前言

本文将使用OpenCV C++ 实现抖音上的特效“蓝线挑战”。虽然看起来觉得很牛的样子,但如果了解其中的原理就非常简单了。本案例是我自己对于这个特效实现过程的理解,仅供参考。
算法原理可以分为三个流程:
1、将视频(图像)从(顶->底)或(左->右)逐行(列)扫描图像。
2、将扫描完成的行(列)像素重新生成定格图像。
3、使用原帧图像像素填充未扫描到的像素。
接下来就具体来看看是如何实现的吧。

一、图像扫描

首先第一步,拿到一个视频(很多帧图像)可以简单的看成图像处理。我们需要将图像从顶到底逐行进行像素扫描,当然也可以从左到右逐列扫描,这要看你想要实现什么样的效果。在这里,我实现的是从上到下逐行扫描。效果如图所示。

请添加图片描述

二、生成定格图像

所谓生成定格图像就是将我们每扫描到的行像素重新进行绘制。

	//从顶向下逐行扫描图像
	if (h < height)
	{
		h++;
		//将扫描到的图像像素进行重新绘制,生成新图像
		for (int j = 0; j < width; j++)
		{
			for (int c = 0; c < 3; c++)
			{
				temp.at<Vec3b>(h, j)[c] = canvas.at<Vec3b>(h, j)[c];
			}
		}
		//绘制扫描过程
		line(canvas, Point(0, h), Point(width, h), Scalar(255, 255, 0), 2);
	}

请添加图片描述
如图所示,这是使用上面代码段实现的逐行扫描生成定格图像。从效果上看,已经得到了我们想要的大致效果了。不过现在的问题是,经扫描到的行有像素填充,未扫描到的行还是漆黑一片。所以接下来我们需要做的就是将未扫描到的行用原图进行填充。具体请看源码注释。

三、图像混合

//将两幅图像进行线性混合
bool Linear_Blend(Mat src1, Mat src2, Mat& dst)
{
	/*
	参数说明:
	src1:生成的定格图像。由于生成的定格图像是从顶往下逐行扫描,故在扫描线下的像素为0
	src2:原视频帧图像
	dst:新生成的定格图像。
	算法原理:经扫描到的像素用src1进行填充,未扫描到的像素用src2进行填充,这样就可以得到我们所要的效果了。
	*/

	for (int i = 0; i < src1.rows; i++)
	{
		for (int j = 0; j < src1.cols; j++)
		{
			for (int c = 0; c < 3; c++)
			{
				if (src1.at<Vec3b>(i, j)[0] != 0)
				{
					dst.at<Vec3b>(i, j)[c] = src1.at<Vec3b>(i, j)[c];
				}
				else
				{
					dst.at<Vec3b>(i, j)[c] = src2.at<Vec3b>(i, j)[c];
				}
			}
		}
	}

	return true;
}

四、效果显示

请添加图片描述
如上图所示,至此我们已经完成了案例所想要的效果。请参考源码,注释也比较详细了。

五、源码

#include<iostream>
#include<opencv2/opencv.hpp>
using namespace std;
using namespace cv;

/*
抖音特效:蓝线挑战
算法原理:
	1、将视频(图像)从(顶->底)或(左->右)逐行(列)扫描图像。
	2、将扫描完成的行(列)像素重新生成定格图像
	3、使用原帧图像像素填充未扫描到的像素
*/

//将两幅图像进行线性混合
bool Linear_Blend(Mat src1, Mat src2, Mat& dst)
{
	/*
	参数说明:
	src1:生成的定格图像。由于生成的定格图像是从顶往下逐行扫描,故在扫描线下的像素为0
	src2:原视频帧图像
	dst:新生成的定格图像。
	算法原理:经扫描到的像素用src1进行填充,未扫描到的像素用src2进行填充,这样就可以得到我们所要的效果了。
	*/

	for (int i = 0; i < src1.rows; i++)
	{
		for (int j = 0; j < src1.cols; j++)
		{
			for (int c = 0; c < 3; c++)
			{
				if (src1.at<Vec3b>(i, j)[0] != 0)
				{
					dst.at<Vec3b>(i, j)[c] = src1.at<Vec3b>(i, j)[c];
				}
				else
				{
					dst.at<Vec3b>(i, j)[c] = src2.at<Vec3b>(i, j)[c];
				}
			}
		}
	}

	return true;
}

int main()
{
	VideoCapture capture;
	capture.open("test.avi");
	if (!capture.isOpened())
	{
		cout << "can not open the camera!" << endl;
		system("pause");
		return -1;
	}

	int width = capture.get(CAP_PROP_FRAME_WIDTH);//视频帧宽
	int height = capture.get(CAP_PROP_FRAME_HEIGHT);//视频帧高

	//保存视频
	VideoWriter writer;
	int fourcc = writer.fourcc('m', 'p', '4', 'v'); //视频编码
	Size size(capture.get(CAP_PROP_FRAME_WIDTH), capture.get(CAP_PROP_FRAME_HEIGHT));
	writer.open("result.avi", fourcc, 30, size, true);

	int h = 0;//定义变量,代表当前扫描高度

	//用于生成定格照
	Mat temp = Mat::zeros(Size(width, height), CV_8UC3);
	
	Mat frame;
	while (capture.read(frame))
	{
		//将图像拷贝一份,用于每帧更新
		Mat canvas = frame.clone();

		//从顶向下逐行扫描图像
		if (h < height)
		{
			h++;
			//将扫描到的图像像素进行重新绘制,生成新图像
			for (int j = 0; j < width; j++)
			{
				for (int c = 0; c < 3; c++)
				{
					temp.at<Vec3b>(h, j)[c] = canvas.at<Vec3b>(h, j)[c];
				}
			}
			//绘制扫描过程
			line(canvas, Point(0, h), Point(width, h), Scalar(255, 255, 0), 2);
		}

		Mat result = Mat::zeros(frame.size(), frame.type());//蓝线挑战最终定格图
		Linear_Blend(temp, canvas, result); //将两张图像进行像素叠加

		//writer.write(temp);//进行视频保存

		imshow("定格图像", temp);
		imshow("原视频帧", canvas);
		imshow("蓝线挑战", result);

		char key = waitKey(10);
		if (key == 27) break;
	}

	capture.release();
	system("pause");
	return 0;
}

总结

本文使用OpenCV C++ 实现抖音特效“蓝线挑战”,关键步骤有以下几点。
1、将图像进行逐行扫描
2、将扫描到的像素逐行生成定格图像
3、将定格图像与原图像进行像素叠加。

  • 10
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
课程目的:OpenCV是应用非常广泛的开源视觉处理库,在图像处理、计算机视觉和自动驾驶中有着非常重要的作用。课程设计特色:(课程当前为第一期)1、C++与Python双语教学Python语言是在计算机视觉中应用最多的一种语言,在工作中,深度学习模型的训练基本上都是使用Python语言编写的训练代码。OpenCV在这个过程中用于图像的预处理(例如图像读取、数据增强)和后处理,还可以用于显示处理的结果,功能强大,使用方便。但是在功能的部署的时候,不管是部署在服务端还是PC端,开发语言基本上用的是C++,所以如何有效的使用OpenCV进行模型或者功能的部署尤为重要。C++语言应用的好坏,在面试中可以看出一个面试者的工程实践能力的强弱,两种语言的开发掌握好了可以使工作如虎添翼。2、全模块讲解我出版了一本图书《学习OpenCV4:基于Python的算法实战》,虽然这本书是写的基于Python的算法实战,但是实际上这本书有详细的介绍算法的C++接口,还有一些C++方向的案例,是以Python为主。图书出版的时候就想双语写作,只是限于篇幅没有成行。本课程不仅采用双语教学,更是对C++的每个模块都做讲解,我们知道,很多的书其实只讲imgproc,如果你翻开一本书图像的形态学运算和图像滤波都是作为独立章节讲解的,那么这本书基本上就可以确定是只是讲解了imgproc模块,但是其他的模块在工作中也有很重要的作用。例如:core模块定义了C++的基本数据结构和基本运算(如四则运算);highgui模块是可视化与交互的模块;feature2d是特征点与特征匹配相关算法所在的模块;ml是机器学习相关的模块;dnn是深度学习相关的模块,可以使用OpenCV进行深度学习模型的部署。这些是很多的书和课程都不会讲的。3、讲解细致本课程会从环境搭建开始讲解,环境搭建尤为重要。从我多年的授课经验总结来看,如果只是给了代码,很多的入门用户环境问题处理不好的话,后面的学习很难进行下去,甚至会丧失学习的信心。4、会讲解C++和Python的开发语法问题是入门用户的一大难关,特别是C++语言。大学只是教授了C语言相关的内容,C++很多同学只懂一点皮毛,所以写代码步履维艰,我们在讲解代码的过程中会顺带讲解C++和Python的内容。我们还会讲解编译相关知识,还有库的装载与链接,这些是学校里不会教的,目前也几乎没有课程讲解。5、讲师经验丰富我讲解过C++和OpenCV的多个课程,广受学员好评。我出版过两本图书《深度学习计算机视觉实战》和《学习OpenCV4》,两本书都是细致入微的讲解,主要针对的就是初学者,所以能够很好的处理课程的难易程度。6、讲义准备充分讲义准备的充分细致,标识清楚明确,重点和疑难点突出。
OpenCV是一款非常强大的计算机视觉库,其中包含了很多功能强大的图像处理和计算机视觉算法。而在这个系列的第三篇文章中,我们将重点介绍如何在OpenCV中绘制图形和文本。 OpenCV中绘制图形的函数包括cv2.line(直线),cv2.rectangle(矩形),cv2.circle(圆),cv2.ellipse(椭圆)和cv2.putText(文本)。这些函数的使用非常简单,只需要指定图像,开始点和结束点(或中心点和半径),颜色和线宽即可。 例如,我们可以使用cv2.rectangle函数在图像上绘制一个矩形: ``` import cv2 import numpy as np img = np.zeros((512,512,3), np.uint8) # 创建一个黑色的图像 cv2.rectangle(img,(384,0),(510,128),(0,255,0),3) # 绘制矩形 cv2.imshow("image",img) cv2.waitKey(0) cv2.destroyAllWindows() ``` 在这个例子中,我们首先创建了一个512x512的黑色图像,然后使用cv2.rectangle函数在图像上绘制了一个矩形,并将其显示出来。在这里,矩形的左上角坐标是(384,0),右下角坐标是(510,128),颜色是绿色(0,255,0),线宽为3。 除了绘制图形,我们也可以使用cv2.putText函数在图像上绘制文本。这个函数的使用也非常简单,只需要指定图像,文本内容,位置,字体,字体大小和颜色即可。 例如,我们可以使用cv2.putText函数在图像上写入一个“Hello World!”的文本: ``` import cv2 import numpy as np img = np.zeros((512,512,3), np.uint8) # 创建一个黑色的图像 font = cv2.FONT_HERSHEY_SIMPLEX # 设置字体 cv2.putText(img,'Hello World!',(10,500), font, 1,(255,255,255),2,cv2.LINE_AA) # 绘制文本 cv2.imshow("image",img) cv2.waitKey(0) cv2.destroyAllWindows() ``` 在这个例子中,我们使用cv2.putText函数在左下角绘制了一个“Hello World!”的文本,并将其显示出来。在这里,我们首先选取了字体类型为cv2.FONT_HERSHEY_SIMPLEX,然后指定了文本的位置(10,500),字体大小为1,颜色为白色(255,255,255),线宽为2,并且使用了cv2.LINE_AA参数,以实现更好的文本渲染效果。 综上所述,OpenCV中可以很方便地对图像进行绘制和文本处理,这对于图像处理和计算机视觉应用来说是非常重要的一步。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

Zero___Chen

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

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

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

打赏作者

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

抵扣说明:

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

余额充值