使用 OpenCV 实现哈哈镜效果

news2025/7/21 2:36:43

在计算机视觉和图像处理领域,OpenCV 提供了非常强大的图像几何变换能力,不仅可以用于纠正图像,还能制造各种“有趣”的视觉效果。今天,我们就来实现一个经典的“哈哈镜”效果,让图像像在游乐园里一样被拉伸、压缩、扭曲,创造出令人发笑的面部或形体变形。

🎯 什么是“哈哈镜”?

“哈哈镜”是一种非线性扭曲镜面,会在不同区域产生放大或缩小的视觉错觉。我们可以用数学变换模拟出类似的效果,比如:

  • 水平凹面/凸面:图像左右边缘被拉伸或收缩

  • 垂直凹面/凸面:图像上下边缘被拉伸或压缩

  • 中心凹面/凸面:图像向内或向外膨胀

  • 水波扰动:从若干中心点向外扩散波纹,模拟水面晃动感


🔧 技术实现原理

我们将使用 OpenCV 的 remap 函数,它允许我们通过两个映射矩阵 map_xmap_y,定义每个输出像素应该对应输入图像的哪个位置。

关键在于如何构造这两个映射矩阵,让它们产生扭曲效果。


🧪 示例代码:中心凸面效果(鱼眼)

import cv2
import numpy as np

def funhouse_effect(frame):
    h, w = frame.shape[:2]
    map_y, map_x = np.indices((h, w), dtype=np.float32)

    # 计算图像中心
    cx, cy = w // 2, h // 2

    # 构造相对坐标
    dx = map_x - cx
    dy = map_y - cy
    r = np.sqrt(dx**2 + dy**2)
    r_max = np.max(r)

    # 控制扭曲强度
    k = 0.0008  # 越大越扭曲(中心凸出)

    scale = 1 + k * (r**2)  # 非线性放大
    map_x = cx + dx * scale
    map_y = cy + dy * scale

    # 保证映射范围合法
    map_x = np.clip(map_x, 0, w - 1)
    map_y = np.clip(map_y, 0, h - 1)

    return cv2.remap(frame, map_x, map_y, interpolation=cv2.INTER_LINEAR)

frame = cv2.imread("face.jpg")
output = funhouse_effect(frame)
cv2.imwrite("distorted.jpg", output)


📚 多种哈哈镜效果

你可以基于上面的思路实现更多效果:

效果类型扭曲方式示意说明
水平凹面scale = 1 - k * ((x-cx)/cx)^2中心宽、边窄
垂直凸面scale = 1 + k * ((y-cy)/cy)^2中心鼓起
中心凹面scale = 1 - k * r^2边缘大、中心小
随机水波扰动sin(r * 频率 + 相位) 叠加扰动水波纹起伏感


🔄 通用框架:FrameObject 封装

为了在实时视频或处理多个帧时使用,我们可以封装为如下类:

class FrameObject:
    def __init__(self):
        self.mode = 'random_wave'  # 选择效果

    def do(self, frame, device):
        h, w = frame.shape[:2]
        map_y, map_x = np.indices((h, w), dtype=np.float32)
        cx, cy = w // 2, h // 2
        dx = map_x - cx
        dy = map_y - cy
        r = np.sqrt(dx**2 + dy**2)

        if self.mode == 'center_fisheye':
            scale = 1 + 0.0006 * (r**2)
            map_x = cx + dx * scale
            map_y = cy + dy * scale

        elif self.mode == 'horizontal_cave':
            scale = 1 - 0.0012 * ((dx / cx) ** 2)
            map_x = cx + dx * scale
            map_y = map_y

        elif self.mode == 'random_wave':
            for _ in range(np.random.randint(1, 4)):
                wave_cx = np.random.randint(w // 4, 3 * w // 4)
                wave_cy = np.random.randint(h // 4, 3 * h // 4)
                ddx = map_x - wave_cx
                ddy = map_y - wave_cy
                rr = np.sqrt(ddx**2 + ddy**2)
                phase = np.random.uniform(0, 2 * np.pi)
                displacement = 8 * np.sin(rr * 0.05 + phase)
                map_x += displacement * (ddx / (rr + 1e-6))
                map_y += displacement * (ddy / (rr + 1e-6))

        map_x = np.clip(map_x, 0, w - 1)
        map_y = np.clip(map_y, 0, h - 1)
        return cv2.remap(frame, map_x, map_y, interpolation=cv2.INTER_LINEAR, borderMode=cv2.BORDER_REFLECT)


🎥 应用场景

  • 互动镜像设备(如景区搞笑自拍)

  • 视频滤镜制作(社交媒体)

  • 教学演示图像几何变换原理

  • 图像增强(用作数据增强的一种方式)


🧠 总结

使用 OpenCV,我们可以轻松实现各种非线性图像变换来模拟“哈哈镜”效果。本质上是通过构建合适的映射矩阵 map_xmap_y,来控制每个像素的位置变换。配合正弦波、极坐标缩放、指数函数等,你可以无限创造各种扭曲方式。


如果你对某种特定变形方式感兴趣,或者想将其用于实时视频流、交互系统中,欢迎留言交流!🎉

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

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

相关文章

node-DeepResearch开源ai程序用于深入调查查询,继续搜索、阅读网页、推理,直到找到答案

​一、软件介绍 文末提供程序和源码下载 node-DeepResearch开源ai程序用于深入调查查询,继续搜索、阅读网页、推理,直到找到答案。 重要提示 与 OpenAI/Gemini/Perfasciity 的“深度研究”不同,我们只专注于通过迭代过程找到正确的答案 。我…

C#、C++、Java、Python 选择哪个好

选择哪种语言取决于具体需求:若关注性能和底层控制选C、若开发企业级应用选Java、若偏好快速开发和丰富生态选Python、若构建Windows生态应用选C#。 以Python为例,它因语法简洁、开发效率高、应用广泛而在AI、数据分析、Web开发等领域大放异彩。根据TIOB…

OpenGL Chan视频学习-8 How I Deal with Shaders in OpenGL

bilibili视频链接: 【最好的OpenGL教程之一】https://www.bilibili.com/video/BV1MJ411u7Bc?p5&vd_source44b77bde056381262ee55e448b9b1973 函数网站: docs.gl 说明: 1.之后就不再整理具体函数了,网站直接翻译会更直观也…

机器学习课程设计报告 —— 基于口红数据集的情感分析

目录 一、课程设计目的 二、数据预处理及分析 2.1 数据预处理 2.2 数据分析 三、特征选择 3.1 特征选择的重要性 3.2 如何进行特征选择 3.3 特征选择的依据 3.4 数据集的划分 四、模型训练与模型评估 4.1 所有算法模型不调参 4.2 K-近邻分类模型 4.3 GaussianNB模…

Windows安装Docker部署dify,接入阿里云api-key进行rag测试

一、安装docker 1.1 傻瓜式安装docker Get Docker | Docker Docs Docker原理(图解秒懂史上最全)-CSDN博客 官网选择好windows的安装包下载,傻瓜式安装。如果出现下面的报错,说明主机没有安装WSL 1.2 解决办法 安装 WSL | Mic…

Dify中 SYSTEM, USER, ASSISTANT 的关系、职责与使用方法

在Dify这类对话式AI应用构建平台中,SYSTEM, USER, ASSISTANT 这三种消息类型共同定义了与大型语言模型(LLM)交互的结构和上下文。它们的关系可以理解为: SYSTEM: 扮演着“导演”或“场景设定者”的角色。USER: 扮演着“提问者”或“任务发起者”的角色。ASSISTANT: 扮演着“…

【ArcGIS Pro草履虫大师】空间地图系列

地图系列是根据单个布局来构建的页面集合。 正常情况下,一个布局只能导出一个页面,通过地图系列则可以通过不同的视图、动态元素,构建并导出多个页面。 地图系列发展自ArcMap的【数据驱动页面】功能。 ArcGIS Pro中有3个地图系列&#xff…

1. 数据结构基本概念 (1)

本文部分ppt、视频截图来自:[青岛大学-王卓老师的个人空间-王卓老师个人主页-哔哩哔哩视频] 1. 数据结构基本概念 1.1 研究内容 数据结构是一门研究非数值计算的程序设计中计算机操作队形以及他们之间关系和操作的核心课程,学习的主要内容如下&#x…

函数抓取图片microsoft excel与wps的区别

microsoft excel 写出index函数 找到图片所在的位置 INDEX(员工数据库!$H:$H,MATCH(Sheet1!$B$3,员工数据库!$A:$A,0))将index函数定义为名称 插入截图 插入-屏幕截图-屏幕剪辑 选中给截图插入定义的公式 WPS 直接写公式抓取

openpi π₀ 项目部署运行逻辑(三)——策略推理服务器 serve_policy.py

π₀ 主控脚本都在 scripts 中: 其中,serve_policy.py 是 openpi 中的策略推理服务端脚本,作用为:启动一个 WebSocket 服务器,加载预训练策略模型,等待外部请求(如来自 main.py 的控制程序&…

基于vue框架的独居老人上门护理小程序的设计r322q(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。

系统程序文件列表 项目功能:用户,护理人员,服务预约,服务评价,服务类别,护理项目,请假记录 开题报告内容 基于Vue框架的独居老人上门护理小程序的设计开题报告 一、研究背景与意义 (一)研究背景 随着社会老龄化的加剧,独居老…

【前端】Hexo一键生成目录插件推荐_放入Hexo博客

效果 使用 安装 npm install hexo-auto-toc插件会自动对<article>包含下的所有内容进行解析&#xff0c;自动生成目录。如果你的文章页面结构中内容没被<article>包裹&#xff0c;需要自行添加它&#xff08;即blog文件夹下的index.html&#xff09;查看效果 hex…

宫格导航--纯血鸿蒙组件库AUI

摘要&#xff1a; 宫格导航(A_GirdNav)&#xff1a;可设置导航数据&#xff0c;建议导航项超过16个&#xff0c;可设置“更多”图标指向的页面路由。最多显示两行&#xff0c;手机每行最多显示4个图标&#xff0c;折叠屏每行最多6个图标&#xff0c;平板每行最多8个图标。多余图…

RNN 循环神经网络:原理与应用

一、RNN 的诞生背景 传统神经网络&#xff08;如 MLP、CNN&#xff09;在处理独立输入时表现出色&#xff0c;但现实世界中存在大量具有时序依赖的序列数据&#xff1a; 自然语言&#xff1a;"我喜欢吃苹果" 中&#xff0c;"苹果" 的语义依赖于前文 "…

若依框架 账户管理 用户分配界面解读

下载下来若依网站后 先对 后端代码进行解读 首先项目架构&#xff1a; 一般用 admin 这个比较多进行二次开发 其他 rouyi-common,rouyi-framework:为公共部分 rouyi-generator&#xff1a;代码生成部分 ruoyi-quartz&#xff1a;定时任务 ruoyi-system&#xff1a;系统任务 …

文档贡献 | 技术文档贡献流程及注意事项(保姆级教程)

内容目录 一、注册流程 二、创建分支&#xff08;Fork&#xff09; 三、使用GitLab界面更新文件的MR流程 四、使用Git命令行工具更新文件的MR流程 五、注意事项 一、注册流程 1、注册页面 在长安链平台注册页面&#xff0c;输入手机号码 &#xff0c;点击 “获取验证码”…

open-vscode-server +nodejs 安装

GitCode - 全球开发者的开源社区,开源代码托管平台GitCode是面向全球开发者的开源社区,包括原创博客,开源代码托管,代码协作,项目管理等。与开发者社区互动,提升您的研发效率和质量。https://gitcode.com/gh_mirrors/op/openvscode-server/?utm_sourceartical_gitcode&ind…

知行之桥如何将消息推送到钉钉群?

在钉钉平台中&#xff0c;机器人主要分为企业机器人和自定义机器人两类。本文将重点介绍如何通过自定义机器人&#xff0c;实现将知行之桥 EDI 系统的通知消息高效推送至钉钉群&#xff0c;帮助企业第一时间掌握业务动态。 一、在钉钉群中添加自定义机器人 在需要接收知行之桥…

09《从依赖管理到容器化部署:Maven 全链路实战笔记,解锁 Java 项目自动化构建的终极奥秘》

目录 一、Maven 核心基础强化 &#xff08;一&#xff09;Maven 架构与工作原理 1. 核心组件解析 2. 工作流程图示​编辑 &#xff08;二&#xff09;项目结构深度实践 1. 标准目录扩展说明 2. 多模块项目典型结构示例​编辑 二、依赖管理高级进阶 &#xff08;一&…

<el-date-picker>组件传参时,选中时间和传参偏差8小时

遇到一个bug&#xff0c;不仔细看&#xff0c;都不一定能发现&#xff0c;bug描述&#xff1a;我们有一个搜索框&#xff0c;里面有一个时间选择器&#xff0c;当我使用<el-date-picker>时&#xff0c;我发现当我选择时分秒之后&#xff0c;显示都正常&#xff0c;但是当…