哈哈哈哈哈哈镜~
时间:2022-07-23
本文章向大家介绍哈哈哈哈哈哈镜~,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。
我绝对是无聊爆炸了,所以我又丧心病狂处理二次元图片了。
今天基于像素变换来实现图像的哈哈镜变换,效果就是下面这样了:
哈哈镜分两种,一种是挤压,一种是放大。分别对应凹函数和凸函数。
输入一副图像,首先设置缩放中心center,我们取原图鼻子处为中心。
设置图像上任意一点到中心点的相对坐标tx= x-cx,ty=y-cy。
左边为挤压哈哈镜,对应像素映射函数:
//变换后新的坐标
x = cos(atan2(ty , tx))* 12*(sqrt(tx*tx + ty*ty))+cx
y = sin(atan2(ty , tx))* 12*(sqrt(tx*tx + ty*ty))+cy
公式中的常数12代表强度,越大则图像越扭曲。
自定义挤压函数(C++版)(Python版可以点击文末阅读原文查看):
Mat MinFrame(Mat frame) {
Mat srcImage;
frame.copyTo(srcImage);
int radius = 400;//定义哈哈镜的半径
int height = frame.rows;
int width = frame.cols; //图片的长宽
Point2d center;//人脸中心
center.x = 130;
center.y = 180;
int newX, newY;//变换后的坐标
for (int x = 0; x < width; x++)
for (int y = 0; y < height; y++) {
double tX = x - center.x;
double tY = y - center.y;
double theta = atan2(tY, tX);
radius = sqrt((tX * tX) + (tY * tY)); //与上面一样,计算公式不一样
int newR = sqrt(radius) *8;
newX = int(center.x + (newR * cos(theta)));
newY = int(center.y + (newR * sin(theta)));
if (newX<0 && newX>width)
newX = 0;
if (newY<0 && newY>height)
newY = 0;
if (newX < width && newY < height) {
srcImage.at<Vec3b>(y, x)[0] = frame.at<Vec3b>(newY, newX*3)[0]; //将计算后的坐标移动到原坐标
srcImage.at<Vec3b>(y, x)[1] = frame.at<Vec3b>(newY, newX*3+1)[1];
srcImage.at<Vec3b>(y, x)[2] = frame.at<Vec3b>(newY, newX*3+2)[2];
}
else {
srcImage.at<Vec3b>(y, x)[0] = frame.at<Vec3b>(y, x)[0]; //将计算后的坐标移动到原坐标
srcImage.at<Vec3b>(y, x)[1] = frame.at<Vec3b>(y, x)[1];
srcImage.at<Vec3b>(y, x)[2] = frame.at<Vec3b>(y, x)[2];
}
}
return srcImage;
}
右边为放大哈哈镜,对应像素映射函数:
//变换后的新坐标
x = (tx/2)*(sqrt(tx*tx + ty*ty)/radius)+cx
y = (ty/2)*(sqrt(tx*tx + ty*ty)/radius)+cy
自定义放大函数(C++版):
Mat MaxFrame(Mat frame) {
Mat srcImage;
frame.copyTo(srcImage);
int radius = 400;//定义哈哈镜的半径
int height = frame.rows;
int width = frame.cols; //图片的长宽
Point2d center;//图片中心
center.x = 130;//人脸中心像素坐标
center.y = 180;
int newX, newY;//变换后的坐标
int real_radius = int(radius / 2.0); //计算公式
for (int x = 0; x < width; x++)
for (int y = 0; y < height; y++) {
int tX = x - center.x;
int tY = y - center.y;
int distance = tX * tX + tY * tY;
if (distance < radius*radius) {
newX = int(tX / 2.0);
newY = int(tY / 2.0);
newX = int(newX * (sqrt(distance) / real_radius));
newY = int(newY * (sqrt(distance) / real_radius));
newX = int(newX + center.x);
newY = int(newY + center.y);
if (newX < width && newY < height) {
srcImage.at<Vec3b>(y, x)[0] = frame.at<Vec3b>(newY, newX)[0]; //将计算后的坐标移动到原坐标
srcImage.at<Vec3b>(y, x)[1] = frame.at<Vec3b>(newY, newX)[1];
srcImage.at<Vec3b>(y, x)[2] = frame.at<Vec3b>(newY, newX)[2];
}
}
else {
srcImage.at<Vec3b>(y, x)[0] = frame.at<Vec3b>(y, x)[0]; //将计算后的坐标移动到原坐标
srcImage.at<Vec3b>(y, x)[1] = frame.at<Vec3b>(y, x)[1];
srcImage.at<Vec3b>(y, x)[2] = frame.at<Vec3b>(y, x)[2];
}
}
return srcImage;
}
至于这俩函数怎么来的,,我也说不出啥门道来,,欢迎评论告知哈。
再看最后一眼:
THE END
- Java魔法堂:JUnit4使用详解
- Windows魔法堂:解决“由于启动计算机时出现页面文件配置问题.......”
- Java魔法堂:自定义和解析注解
- 【Hibernate那点事儿】—— Hibernate应该了解的知识
- 【Hibernate那点事儿】—— Hibernate应该了解的知识
- Java魔法堂:以Windows服务的形式运行Java程序
- 【Hibernate那点事儿】—— Hibernate知识总结
- java获取网页源代码
- Java魔法堂:注解用法详解——@SuppressWarnings
- 日志那点事儿——slf4j源码剖析
- 给Dreamweaver插上Svn的翅膀
- MyBatis魔法堂:ResultMap详解
- 腾讯海量监控体系经验分享
- 【Spring开发】—— Spring Core
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- Java NIO Selector 详解
- 【机器学习基础】一文搞懂机器学习里的L1与L2正则化
- 【深度学习】深入理解LSTM
- 短视频商城源码,两种方式实现点击出现弹窗显示
- 【50期】基础考察:ClassNotFoundException 和 NoClassDefFoundError 有什么区别
- MySQL 8.0新特性 — 管理端口
- 从单词嵌入到文档距离 :WMD一种有效的文档分类方法
- 程序员技术选型:写Go还是Java?
- 用python爬虫简单网站却有 “多重思路”--猫眼电影
- 对开发者而言,GitHub Profile可能是比简历更好的求职利器
- R语言中GLM(广义线性模型),非线性和异方差可视化分析
- leetcode树之二叉树的深度
- PowerBI DAX 库存余量模型与计算
- 20种小技巧,玩转Google Colab
- 图深度学习入门教程(九)——图滤波神经网络模型