VS2017+opencv还可以这样画女朋友????!!!!!!(基于opencv的蒙太奇图片制作)

2023-10-23 22:40

本文主要是介绍VS2017+opencv还可以这样画女朋友????!!!!!!(基于opencv的蒙太奇图片制作),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

VS2017+opencv制作蒙太奇照片

  • 简介
    • 前提准备
    • 思路
    • 代码解析
    • 完整代码
    • 运行截图
    • 工程使用说明
    • 运行结果
    • 总结

简介

图片的蒙太奇效果,一般称为马赛克图。由很多小图拼接成一个大图。在商业宣传后,宣传自己同时也可以顺便宣传诸多合作商,颇受欢迎。教程展示利用opencv来完成马赛克图的制作。效果图如下:
在这里插入图片描述

前提准备

  1. VS2017的安装 ,参考教程:vs2017安装教程;
  2. OPENCV3.4.1的安装 ,参考教程:vs2017安装opencv教程;
  3. 准备一张目标图片和一堆用于马赛克的图片;

思路

  1. 修改那些提前准备的一堆图片的大小到马赛克般大小。
  2. 计算每一张图片的颜色均值。
  3. 修改目标图片的长宽,使之长宽都是马赛克大小的整数倍。
  4. 一马赛克大小为单位遍历目标图片同时计算这个区域的颜色均值 ,然后和提前准备的用于替换的马赛克图片的均值进行比较,用均值差最接近的替换目标图片。
  5. 用全替换好的图片和最开始准备的目标图片做一个融合,效果更加逼真。

代码解析

思路一代码:

在这里插入图片描述
这一部分主要是修改了用于替换的图片的大小,同时计算了每张图片的颜色均值,然后我定义了一个map类,用均值作为键值,路径作为槽值,为了后文替换目标图片部分做准备

思路二代码:

在这里插入图片描述
这是计算颜色均值的源码

思路三代码:

在这里插入图片描述
在这里插入图片描述
图一就是对目标图片的大小进行一个小小的修改,让他可以由整数个马赛克图来替换而成。
图二是对mat类图片大小修改的源码,xy参数是对传入的image基于(x,y)点向右下取width、height大小的图片。

思路四代码:

for (int i = 0; i < src.rows / height; i++){for (int j = 0; j < src.cols / width; j++){static int n = 0;float error = 0;float lasterror = 10000000;string path = "";Mat tempPieces = spliteTest(src, j * width, i * height, width, height);float ave = cal_mean_stddev(tempPieces);map<int, string>::iterator strmap_iter = strMap.begin();for (; strmap_iter != strMap.end(); strmap_iter++){if (fabsf(ave - strmap_iter->first) < 1){path = strmap_iter->second;break;}error = fabsf(ave - strmap_iter->first);if (error < lasterror){lasterror = error;path = strmap_iter->second;}//cout << strmap_iter->first << ' ' << strmap_iter->second << endl;}Mat tempimg = imread(path);mergeTest(src, tempimg, j * width, i * height, width, height);}}

这部分用简单的算法,以马赛克大小为单位遍历整张目标图,然后计算每部分的颜色均值,然后与我们之前map类中存的键值作比较,然后取颜色均值相近的来替换。

思路五代码:

addWeighted(image, alpha, source, beta, 0.0, image);

最后用结果图和原始图片做一个融合,这样的目的是有的时候我们准备的替换图片数量过少或者颜色太单一,这样会导致目标图上有一些颜色我们是没有准备的从而效果图差异较大。融合也就是用alpha和beta这两个比例系数对mat矩阵的三通道乘上一个系数缩放。具体原理可以百度addWeighted函数源码。

完整代码

#include "pch.h"
#include <iostream>
#include <opencv/cv.hpp>
#include <opencv2/core.hpp>
#include <highgui/highgui.hpp>
#include<map>using namespace cv;
using namespace std;//初始化这些即可
string sourcePath = "101.jpg";//原始图像路径
string repalcePicturePath = "D:\\imgs\\";//用于替换的图片的路径
string pictureFormat = "*.jpg"; //读取格式是.jpg可以换成其他,但是文件夹下要统一
string correctSizePicturePath = "D:\\imgs\\resize\\";//把用于替换的图片修正成小碎片的尺寸存放的路径
string resultPicturePath = "result.jpg";//结果存放的路径以及命名
#define replaceWidth 500//替换像素的宽度
#define replaceHeight 500//替换像素的高度
#define alpha 0.2//图片融合相似度系数,系数越小越接近原始图(0-1范围)
#define beta (1 - alpha)Mat spliteTest(Mat image, int x, int y, int width, int height);
void mergeTest(Mat result, Mat piecse, int x, int y, int width, int height);
float cal_mean_stddev(Mat src);
void convert(Mat &src, int width, int height);int main()
{//计算替换图片的灰度均值//把替换图片的均值和路径存到map中键值是灰度均值,实值是路径//遍历切割结果,计算切片的灰度均值,用替换图片替换并且合并到结果矩阵cout << "开始读取目标图片" << endl;Mat image = imread(sourcePath);//加载原始图像Mat source;image.copyTo(source);cout << "完成读取目标图片" << endl;cout << "开始马赛克转换并修正图片" << endl;convert(image, replaceWidth, replaceHeight);cout << "转换完成" << endl;cout << "开始融合优化" << endl;addWeighted(image, alpha, source, beta, 0.0, image);cout << "优化完成,优化相似度:" << beta << endl;cout << "开始将结果写入本地" << endl;imwrite(resultPicturePath, image);cout << "写入成功,文件路径:" << resultPicturePath << endl;cvWaitKey(0);}Mat spliteTest(Mat image, int x, int y, int width, int height)
{long cols = image.cols;long rows = image.rows;Rect rect(x, y, width, height);Mat nullImage(width, height, CV_8UC3);nullImage = image(rect);return nullImage;
}void mergeTest(Mat result, Mat piecse, int x, int y, int width, int height)
{Rect rect(x, y, width, height);piecse.copyTo(result(rect));
}float cal_mean_stddev(Mat src) {Mat gray, mat_mean, mat_stddev;cvtColor(src, gray, CV_RGB2GRAY); // 转换为灰度图meanStdDev(gray, mat_mean, mat_stddev);float m;//m = mat_stddev.at<double>(0, 0);m = mat_mean.at<double>(0, 0);//cout << "灰度均值是:" << m << endl;return m;
}void convert(Mat &src, int width, int height)
{string pattern = repalcePicturePath + pictureFormat;vector<cv::String> fileName;map<int, string> strMap;//定义map类用均值作为键值来查找文件的路径glob(pattern, fileName, false);//查找符合特定规则的文件路径名//for循环遍历用于替换的图片for (int i = 0; i < fileName.size(); i++){string imgName = fileName[i];Mat temp = imread(imgName);//修改替换图片到马赛克图大小resize(temp, temp, Size(width, height));imwrite(correctSizePicturePath + to_string(i) + ".jpg", temp);//map类用均值作为键值来查找文件的路径pair<float, string> value(cal_mean_stddev(temp), correctSizePicturePath + to_string(i) + ".jpg");strMap.insert(value);}cout << "原始col:" << src.cols << endl;cout << "原始row:" << src.rows << endl;src = (src.cols % width == 0) ? src : spliteTest(src, 0, 0, src.cols - src.cols % width, src.rows);src = (src.rows % height == 0) ? src : spliteTest(src, 0, 0, src.cols, src.rows - src.rows % height);cout << "修正col:" << src.cols << endl;cout << "修正row:" << src.rows << endl;for (int i = 0; i < src.rows / height; i++){for (int j = 0; j < src.cols / width; j++){static int n = 0;float error = 0;float lasterror = 10000000;string path = "";Mat tempPieces = spliteTest(src, j * width, i * height, width, height);float ave = cal_mean_stddev(tempPieces);map<int, string>::iterator strmap_iter = strMap.begin();for (; strmap_iter != strMap.end(); strmap_iter++){if (fabsf(ave - strmap_iter->first) < 1){path = strmap_iter->second;break;}error = fabsf(ave - strmap_iter->first);if (error < lasterror){lasterror = error;path = strmap_iter->second;}}Mat tempimg = imread(path);mergeTest(src, tempimg, j * width, i * height, width, height);}}
}

运行截图

在这里插入图片描述

工程使用说明

**最后为大家准备了整个工程的源码,做一个简单说明如下图:
在这里插入图片描述
我们只需要配置下代码里这些内容即可。
这里不在过多的说明,可以看下我的文件路径。
在这里插入图片描述
在这里插入图片描述

运行结果

原图(拿女朋友开刀吧hhhhh):
在这里插入图片描述
没加图片融合的效果图:

在这里插入图片描述
加图片融合的效果图:
在这里插入图片描述

总结

其实有思路的话还挺好写的。。。过程中最曲折的就是mat类的col和row。。。行和列永远也分不清楚,我是服了,动不动就遍历图片超出了长宽,然后就各种异常,这可能就是vs的魅力吧。
enjoy it~

这篇关于VS2017+opencv还可以这样画女朋友????!!!!!!(基于opencv的蒙太奇图片制作)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/270904

相关文章

基于Python实现一个图片拆分工具

《基于Python实现一个图片拆分工具》这篇文章主要为大家详细介绍了如何基于Python实现一个图片拆分工具,可以根据需要的行数和列数进行拆分,感兴趣的小伙伴可以跟随小编一起学习一下... 简单介绍先自己选择输入的图片,默认是输出到项目文件夹中,可以自己选择其他的文件夹,选择需要拆分的行数和列数,可以通过

Python使用OpenCV实现获取视频时长的小工具

《Python使用OpenCV实现获取视频时长的小工具》在处理视频数据时,获取视频的时长是一项常见且基础的需求,本文将详细介绍如何使用Python和OpenCV获取视频时长,并对每一行代码进行深入解析... 目录一、代码实现二、代码解析1. 导入 OpenCV 库2. 定义获取视频时长的函数3. 打开视频文

springboot项目打jar制作成镜像并指定配置文件位置方式

《springboot项目打jar制作成镜像并指定配置文件位置方式》:本文主要介绍springboot项目打jar制作成镜像并指定配置文件位置方式,具有很好的参考价值,希望对大家有所帮助,如有错误... 目录一、上传jar到服务器二、编写dockerfile三、新建对应配置文件所存放的数据卷目录四、将配置文

利用Python脚本实现批量将图片转换为WebP格式

《利用Python脚本实现批量将图片转换为WebP格式》Python语言的简洁语法和库支持使其成为图像处理的理想选择,本文将介绍如何利用Python实现批量将图片转换为WebP格式的脚本,WebP作为... 目录简介1. python在图像处理中的应用2. WebP格式的原理和优势2.1 WebP格式与传统

Python如何将OpenCV摄像头视频流通过浏览器播放

《Python如何将OpenCV摄像头视频流通过浏览器播放》:本文主要介绍Python如何将OpenCV摄像头视频流通过浏览器播放的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完... 目录方法1:使用Flask + MJPEG流实现代码使用方法优点缺点方法2:使用WebSocket传输视

基于 HTML5 Canvas 实现图片旋转与下载功能(完整代码展示)

《基于HTML5Canvas实现图片旋转与下载功能(完整代码展示)》本文将深入剖析一段基于HTML5Canvas的代码,该代码实现了图片的旋转(90度和180度)以及旋转后图片的下载... 目录一、引言二、html 结构分析三、css 样式分析四、JavaScript 功能实现一、引言在 Web 开发中,

Python如何去除图片干扰代码示例

《Python如何去除图片干扰代码示例》图片降噪是一个广泛应用于图像处理的技术,可以提高图像质量和相关应用的效果,:本文主要介绍Python如何去除图片干扰的相关资料,文中通过代码介绍的非常详细,... 目录一、噪声去除1. 高斯噪声(像素值正态分布扰动)2. 椒盐噪声(随机黑白像素点)3. 复杂噪声(如伪

Python中图片与PDF识别文本(OCR)的全面指南

《Python中图片与PDF识别文本(OCR)的全面指南》在数据爆炸时代,80%的企业数据以非结构化形式存在,其中PDF和图像是最主要的载体,本文将深入探索Python中OCR技术如何将这些数字纸张转... 目录一、OCR技术核心原理二、python图像识别四大工具库1. Pytesseract - 经典O

使用Python和OpenCV库实现实时颜色识别系统

《使用Python和OpenCV库实现实时颜色识别系统》:本文主要介绍使用Python和OpenCV库实现的实时颜色识别系统,这个系统能够通过摄像头捕捉视频流,并在视频中指定区域内识别主要颜色(红... 目录一、引言二、系统概述三、代码解析1. 导入库2. 颜色识别函数3. 主程序循环四、HSV色彩空间详解

OpenCV实现实时颜色检测的示例

《OpenCV实现实时颜色检测的示例》本文主要介绍了OpenCV实现实时颜色检测的示例,通过HSV色彩空间转换和色调范围判断实现红黄绿蓝颜色检测,包含视频捕捉、区域标记、颜色分析等功能,具有一定的参考... 目录一、引言二、系统概述三、代码解析1. 导入库2. 颜色识别函数3. 主程序循环四、HSV色彩空间