非局部均值滤波(磨皮)

news2025/1/19 22:15:56

note

根据掩膜区域与其他区域的相似程度来赋予掩膜区域像素权重

相似程度由均方差来衡量

code

/*
 \brief 矩阵求邻和
 \param type=1,列方向;type=2,行方向
*/
static void MyCumSum(Mat& src, Mat& res, int type) {
	if ((src.channels() > 1) || (res.channels() > 1)) {
		return;
	}
	src.copyTo(res);
	res.convertTo(res, CV_64FC1);
	if (type == 1) {
		for (int i = 0; i < src.cols; ++i) {
			for (int j = 1; j < src.rows; ++j) {
				res.at<double>(j,i) = res.at<double>(j-1,i) + res.at<double>(j,i);
			}
		}
	}
	else if (type == 2) {
		for (int i = 0; i < src.rows; ++i) {
			for (int j = 1; j < src.cols; ++j) {
				res.at<double>(i,j) = res.at<double>(i,j-1) + res.at<double>(i,j);
			}
		}
	}
}

void GetWeight(Mat& distance, Mat& weight, double h) {
	for (int i = 0; i < weight.rows; ++i) {
		for (int j = 0; j < weight.cols; ++j) {
			double w = distance.at<double>(i,j);
			weight.at<double>(i,j) = exp(-1*w/h/h);
		}
	}
}

void MyNonLocalMeanFilter(Mat& src, Mat& res, int ds = 2, int Ds = 5, double h = 10) {
	int srcType = src.type();
	if (srcType != CV_64FC1) {
		src.convertTo(src, CV_64FC1);
		res.convertTo(res, CV_64FC1);
	}

	int m = src.rows;
	int n = src.cols;
	int offset = ds+Ds;

	Mat paddedImg;
	copyMakeBorder(src, paddedImg, offset, offset, offset, offset, BORDER_REFLECT);	// 上下左右镜像复制矩阵边界

	Mat sumImage(m, n, CV_64FC1, Scalar(0));
	Mat sumWeight(m, n, CV_64FC1, Scalar(0));
	Mat maxWeight(m, n, CV_64FC1, Scalar(0));

	Rect rc;
	rc.x = Ds;
	rc.y = Ds;
	rc.width = m+ds;
	rc.height = n+ds;
	Mat image;	// 和搜索图比较使用的原图
	paddedImg(rc).copyTo(image);

	int M = image.rows;
	int N = image.cols;

	// 移动搜索框
	for (int r = -Ds; r <= Ds; r++) {
		for (int s = -Ds; s <= Ds; s++) {
			if (r == 0 && s == 0) {
				continue;
			}

			// 求差值积分图
			Rect recTmp;
			recTmp.x = Ds+s;
			recTmp.y = Ds+r;
			recTmp.width = n+ds;
			recTmp.height = m+ds;
			Mat wimage;	// 某个搜索框对应的搜索图
			paddedImg(recTmp).copyTo(wimage);

			Mat diff;	// 差值平方图
			subtract(image, wimage, diff);
			multiply(diff, diff, diff);

			Mat J;
			MyCumSum(diff, J, 1);	// 列方向求邻和
			MyCumSum(J, J, 2);	// 行方向求邻和

			recTmp.x = N-n;
			recTmp.y = M-m;
			recTmp.width = n;
			recTmp.height = m;
			Mat matTmp1;
			J(recTmp).copyTo(matTmp1);

			recTmp.x = 0;
			recTmp.y = 0;
			recTmp.width = n;
			recTmp.height = m;
			Mat matTmp2;
			J(recTmp).copyTo(matTmp2);

			recTmp.x = 0;
			recTmp.y = M-m;
			recTmp.width = n;
			recTmp.height = m;
			Mat matTmp3;
			J(recTmp).copyTo(matTmp3);

			recTmp.x = N-n;
			recTmp.y = 0;
			recTmp.width = n;
			recTmp.height = m;
			Mat matTmp4;
			J(recTmp).copyTo(matTmp4);

			// 计算距离
			Mat distance;
			add(matTmp1, matTmp2, distance);
			subtract(distance, matTmp3, distance);
			subtract(distance, matTmp4, distance);
			double var = (2*ds+1) * (2*ds+1);
			distance = distance / var;

			// 计算权重并获得单个偏移下的加权图像
			Mat weight(distance.rows, distance.cols, CV_64FC1, Scalar(0));
			GetWeight(distance, weight, h);

			recTmp.x = ds;
			recTmp.y = ds;
			recTmp.width = n;
			recTmp.height = m;
			Mat matTmp5;
			wimage(recTmp).copyTo(matTmp5);

			multiply(weight, matTmp5, matTmp5);
			add(sumImage, matTmp5, sumImage);

			add(sumWeight, weight, sumWeight);

			maxWeight = cv::max(maxWeight, weight);
		}
	}
	
	rc.x = ds;
	rc.y = ds;
	rc.width = n;
	rc.height = m;
	Mat matTmp;
	image(rc).copyTo(matTmp);
	multiply(maxWeight, matTmp, matTmp);
	add(sumImage, matTmp, sumImage);
	add(sumWeight, maxWeight, sumWeight);
	divide(sumImage, sumWeight, res);

	src.convertTo(src, srcType);
	res.convertTo(res, srcType);
}

test

 

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/731277.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

关于impacket套件中的atexec.py在使用的时候报error: unrecognized arguments: 的解决方案

定位到库文件C:\Users\LC\AppData\Local\Programs\Python\Python311\Lib\argparse.py 加一个return args就行了 用完记得改回来&#xff0c;不然会影响其他脚本的功能

算法分析和设计选择和判断

算法分析和设计选择和判断 目录 平时测试 1.算法概述 1.1渐进符号的概念 1.2求时间复杂度 1.3比较时间复杂度 1.4时间复杂度定义 1.5算法的有效性 1.6算法的性质 1.7顺序搜索法 1.8渐进算法分析 2.递归和分治策略 2.1归并排序 2.2归并的空间复杂度 2.3插入排序 2.4递归 2.5归并…

Docker容器与HSM整合的好处

Docker透过容器使创建、部署和运行应用程式变得更加容易。容器允许开发人员可以将一个应用程式与它所需要的所有部分封包&#xff0c;然后将应用程式及其组件作为一个单一的套装软体。 Docker面临的挑战 : ● 遭受特权用户滥用的风险 Docker环境或容器内的权限配置不当可能会导…

Redis常见数据结构

文章目录 前言一、Redis通用命令二、String类型三、Key的层级结构四、Hash类型五、List类型六、Set类型七、SortedSet类型 前言 Redis是一个key-value的数据库&#xff0c;key一般是String类型&#xff0c;但是value的类型多种多样 在学习Redis不同数据类型时&#xff0c;我们…

ChatGPT付费创作系统V2.1.0独立版 +WEB端+ H5端 + 小程序端+文心一言

ChatGPT付费创作系统V2.1.0提供最新的对应版本小程序端&#xff0c;上一版本增加了PC端绘画功能&#xff0c;绘画功能采用其他绘画接口-意间AI&#xff0c;本版新增了百度文心一言接口。后台一些小细节的优化及一些小BUG的处理&#xff0c;前端进行了些小细节优化&#xff0c;针…

C#学习之路-类型转换

类型转换从根本上说是类型铸造&#xff0c;或者说是把数据从一种类型转换为另一种类型。 C# 中的类型转换可以分为两种&#xff1a;隐式转换和显式转换。 隐式类型转换 隐式转换是指将一个较小范围的数据类型转换为较大范围的数据类型时&#xff0c;编译器会自动完成类型转换…

chatgpt生成pygame opengl实现旋转用图片填充的3d三角形

import pygame from pygame.locals import * from OpenGL.GL import * from OpenGL.GLU import *def draw_triangle():vertices ((0, 2, 0), # 顶点1(-2, -2, 0), # 顶点2(2, -2, 0) # 顶点3)tex_coords ((1, 2), # 顶点1的纹理坐标(1, 1), # 顶点2的纹理坐标(2, …

如何使用SU插件一键生成通道图

如何使用SU插件一键生成通道图 通道图有什么作用&#xff1f; 以材质ID为例&#xff0c;最大的作用就是在PS通过选择材质ID图上的颜色&#xff0c;将颜色区域转化为蒙版&#xff0c;进而可以调节颜色区域所对应的效果图区域&#xff0c;原理类似于PS自带的通道&#xff1a;而A…

css 3D Demo

<!DOCTYPE html> <html> <head> <meta charset"utf-8"> <title></title> <style> body{ /* 该属性必须在父或大于父的级别上添加,保证其子,孙等盒子可以以3D视角…

langchain调用chatGLM2纪实

一、科学上网要注意&#xff1a; 域名全代和全局代理&#xff08;网卡&#xff09;&#xff0c;都要打开。这样conda install特别快。 二、安装langchain 1、 conda install langchain 2、 conda install openai 注意&#xff1a; 使用pip install和conda install 是不同…

mysql如何运行 mysql 基本命令

一、 mysql如何运行 1. 成功安装mysql 点击进入–》如何安装 2. 确保mysql服务启动&#xff1a; 按下win&#xff0c;输入服务&#xff0c;然后找到mysql&#xff0c;启动服务&#xff1a; 3. 配置环境变量&#xff08;可跳过&#xff09; 点击进入–》如何配置环境变量…

SpringBoot项目出现Failed to configure a DataSource错误时解决方法

若在运行SpringBoot项目时&#xff0c;出现如下错误&#xff1a; Description:Failed to configure a DataSource: url attribute is not specified and no embedded datasource could be configured.Reason: Failed to determine a suitable driver classAction:Consider the…

1.前言和介绍

目录 前言1. 说在前面2. 介绍3. 拓展&#xff1a;vscode3.1 文件传输3.2 高效调试配置3.2.1 配置launch.json3.2.2 配置task.json3.2.3 配置settings.json3.2.4 配置c_cpp_properties.json 前言 从零学习算法部署-TensorRT篇 杜老师推出的 tensorRT从零起步高性能部署 课程&…

7、添加billboard小旗子

本节演示添加billboard小旗子&#xff0c;使用第四节click的例子&#xff0c;通过鼠标点击地图添加billboard添加小旗子。 1、修改鼠标点击事件mouseUp&#xff0c;在函数中创建BillboardCollection对象并添加到地图中。鼠标点击时获取鼠标位置&#xff0c;转换成经纬度然后添…

Mycat【Mycat分片技术(水平拆分-分表、ER表、全局表)】(五)-全面详解(学习总结---从入门到深化)

目录 Mycat分片技术_水平拆分-分表 Mycat分片技术_ER表 Mycat分片技术_全局表 Mycat分片技术_水平拆分-分表 前言 上面谈到垂直切分只是把表按模块划分到不同数据库&#xff0c;但没有解决单表大数据量的问题。 水平拆分 相对于垂直拆分&#xff0c;水平拆分不是将表做分…

宝塔 安装/使用Apollo(阿波罗)多环境部署-图文小白教程

官方安装教程手册&#xff1a;Apollo apollo&#xff08;阿波罗&#xff09;是一款可靠的分布式配置管理中心&#xff0c;诞生于携程框架研发部&#xff0c;能够集中化管理应用不同环境、不同集群的配置&#xff0c;配置修改后能够实时推送到应用端&#xff0c;并且具备规范的…

Android 视频直播提拉流 嵌入式硬件 流媒体开发详细内容

1 Linux 系统编程网络编程基础 2 Linux 网络编程流媒体服务器&#xff0c;客户端开发实践 3 Android流媒体客户端 FFmpeg OpenGL ES 开发实践 4 Android H.264 AAC 封装mp4开发实战 5 流媒体开发实战之Rtmp推流 6 流媒体开发实战之RTSP推流 7 流媒体开发实战之UDP 8 P2P点对点项…

BERT 论文精读与理解

1.论文题目 BERT: Pre-training of Deep Bidirectional Transformers for Language Understanding 2.论文摘要 本文引入了一种名为 BERT 的新语言表示模型&#xff0c;它代表 Transformers 的双向编码器表示。与最近的语言表示模型&#xff08;Peters et al., 2018a&#xf…

【云原生】 一文了解Docker到底是什么?

目录 1.docker是什么&#xff1f; 2.为什么需要docker&#xff1f; 3.docker特点 4.docker架构 5.云计算中的服务包括三个层面 6.传统虚拟化架构 7.容器架构 8.docker系统架构 Docker 守护进程 Docker 客户端 Docker 仓库 Docker 对象 Images&#xff08;镜像&#xff09; Cont…

数据库基本操作及表的创建

目录标题 1.数据库基本操作创建数据库查询数据库使用数据库查询默认数据库编码方式查询默认数据库校验规则删除数据 2.表的基本操作创建一个包含所有类型的表 1.数据库基本操作 创建数据库 查询数据库 使用数据库 查询默认数据库编码方式 查询默认数据库校验规则 删除数据 2.表…