数字水印 | Python 基于离散小波变换 DWT 的图像水印嵌入(上)

🍍原文: 基于 dwt (离散小波变换) 实现彩色图像水印嵌入部分_1.0

🍍写在前面: 本文在原文的基础上进行了代码补全。



正文

本文的内容主要为:水印图像经过 A r n o l d \mathsf{Arnold} Arnold 置乱算法后,通过离散小波变换进行嵌入。

在这里插入图片描述

c A \mathsf{cA} cA 等为原始图像的小波系数, c a \mathsf{ca} ca 等为水印图像的小波系数,嵌入公式如下:

{ c A 3 ′ = c A 3 + a 1 × c a 1 c H 3 ′ = c H 3 + a 2 × c h 1 c V 3 ′ = c V 3 + a 3 × c v 1 c D 3 ′ = c D 3 + a 4 × c d 1 \left\{\begin{matrix} cA'_3 = cA_3 + a_1 \times ca_1\\ cH'_3 = cH_3 + a_2 \times ch_1 \\ cV'_3 = cV_3 + a_3 \times cv_1 \\ cD'_3 = cD_3 + a_4 \times cd_1 \end{matrix}\right. cA3=cA3+a1×ca1cH3=cH3+a2×ch1cV3=cV3+a3×cv1cD3=cD3+a4×cd1

为了验证方便,后文代码将上述公式中的三级小波 c A 3 \mathsf{cA_3} cA3 变换与一级小波 c a 1 \mathsf{ca_1} ca1 变换的嵌入,简化为了一级小波 c A 1 \mathsf{cA_1} cA1 变换与一级小波 c a 1 \mathsf{ca_1} ca1 变换的嵌入。当然,嵌入效果不会很好。

看得出来嵌入效果不是很好😇

为了使结果仍然为彩色图像,本文在小波变换前将原始图像的 R , G , B \mathsf{R,G,B} R,G,B 通道分离,仅在 B ( b l u e ) \mathsf{B(blue)} B(blue) 通道中嵌入水印。嵌入完成后,再将 R , G , B \mathsf{R,G,B} R,G,B 三个通道合并。



1 代码说明

1.1 Arnold

A r n o l d \mathsf{Arnold} Arnold 置乱算法如下,即粘即用:

def arnold(img, s):  # s置乱次数
    r, c, d = img.shape
    img = img[:, :, 0]
    p = np.zeros((r, c), np.uint8)
    a = 1
    b = 1
    for _s in range(s):
        for i in range(r):
            for j in range(c):
                x = (i + b * j) % r
                y = (a * i + (a * b + 1) * j) % c
                p[x, y] = img[i, j]
        img = np.copy(p)  # 深复制
    return p

以上代码就是对 A r n o l d \mathsf{Arnold} Arnold 公式的实现。如果不知道 A r n o l d \mathsf{Arnold} Arnold 公式是什么,那么自然是看不懂的😇

参考自博客:Python 基于位平面的信息隐藏算法 阿诺德置乱算法



1.2 读取图像

分别读取 I m g \mathsf{Img} Img 原始图像和 w a t e r I m g \mathsf{waterImg} waterImg 水印图像:

Img = cv2.imread('white_bear.jpg')
# 原始图像调序
b, g, r = cv2.split(Img)
Img = cv2.merge([r, g, b])

waterImg = cv2.imread('uestc_logo.jpg')
# 水印图像调序
b, g, r = cv2.split(waterImg)
waterImg = cv2.merge([r, g, b])

cv2 读取图片时的通道顺序为 B、G、R,而 PIL 显示图片时的通道顺序为 R、G、B,因此显示出来的图片颜色会改变,需要对图像通道进行调序。

参考自博客:CV2 读取图片,图片颜色发生改变解决方案



1.3 小波变换

首先修剪原始图像为水印图像的大小,然后分离出原始图像的 B \mathsf{B} B 通道:

img = cv2.resize(Img, (r, c))  # 修剪原始图像
(b, g, r) = cv2.split(img)  # 分离通道

接着,直接调用库函数 pywt.dwt2 实现小波变换:

# 水印图像一级小波变换
coeffs1 = pywt.dwt2(waterImg, 'haar')
ca1, (ch1, cv1, cd1) = coeffs1

# 原始图像B通道一级小波变换
coeffs2 = pywt.dwt2(b, 'haar')
ca2, (ch2, cv2, cd2) = coeffs2

如果没有 p y w t \mathsf{pywt} pywt 库则需要进行安装:

pip install PyWavelets


1.4 嵌入水印

就是对文首的嵌入公式的实现:

# 自定义嵌入系数,可用随机数处理
a1 = 0.1
a2 = 0.2
a3 = 0.1
a4 = 0.1

ca2 = ca2 + ca1 * a1
ch2 = ch2 + ch1 * a2
cv2 = cv2 + cv1 * a3
cd2 = cd2 + cd1 * a4


1.5 小波逆变换

第一句代码才是小波逆变换:

newImg = pywt.idwt2((ca2, (ch2, cv2, cd2)), "haar")
merged = np.ones(img.shape, dtype=np.uint8)
merged[:, :, 0] = r
merged[:, :, 1] = g
merged[:, :, 2] = newImg

后面的代码是在对通道进行合并。可以看出,嵌入了水印的 n e w I m g \mathsf{newImg} newImg 被我们放在了 B \mathsf{B} B 通道,而 R , G \mathsf{R,G} R,G 通道保持原样。



2 完整代码

import cv2
import pywt
import numpy as np
import matplotlib.pyplot as plt


def arnold(img, s):  # s是置乱次数
    r, c, d = img.shape
    img = img[:, :, 0]
    p = np.zeros((r, c), np.uint8)
    a = 1
    b = 1
    for _s in range(s):
        for i in range(r):
            for j in range(c):
                x = (i + b * j) % r
                y = (a * i + (a * b + 1) * j) % c
                p[x, y] = img[i, j]
        img = np.copy(p)  # 深复制
    return p


Img = cv2.imread('white_bear.jpg')  # 原始图像
b, g, r = cv2.split(Img)
Img = cv2.merge([r, g, b])

water = cv2.imread('uestc_logo.jpg')  # 只是为了后面的展示
b, g, r = cv2.split(water)
water = cv2.merge([r, g, b])

waterImg = cv2.imread('uestc_logo.jpg')  # 水印图像
b, g, r = cv2.split(waterImg)
waterImg = cv2.merge([r, g, b])
waterImg = arnold(waterImg, 5)  # Arnold置乱

# 原始图像尺寸数据
R = Img.shape[0]
C = Img.shape[1]

# 水印图像尺寸数据
r = waterImg.shape[0]
c = waterImg.shape[1]

img = cv2.resize(Img, (r, c))  # 修剪原始图像
(b, g, r) = cv2.split(img)  # 分离通道

# 水印图像一级小波变换
coeffs1 = pywt.dwt2(waterImg, 'haar')
ca1, (ch1, cv1, cd1) = coeffs1

# 原始图像B通道一级小波变换
coeffs2 = pywt.dwt2(b, 'haar')
ca2, (ch2, cv2, cd2) = coeffs2

# 自定义嵌入系数,可用随机数处理
a1 = 0.1
a2 = 0.2
a3 = 0.1
a4 = 0.1

ca2 = ca2 + ca1 * a1
ch2 = ch2 + ch1 * a2
cv2 = cv2 + cv1 * a3
cd2 = cd2 + cd1 * a4

# 对小波系数进行逆变换
newImg = pywt.idwt2((ca2, (ch2, cv2, cd2)), "haar")
merged = np.ones(img.shape, dtype=np.uint8)
merged[:, :, 0] = r
merged[:, :, 1] = g
merged[:, :, 2] = newImg

plt.subplot(2, 2, 1)
plt.title("Watermark", fontsize=12, loc="center")
plt.axis('off')
plt.imshow(water)

plt.subplot(2, 2, 2)
plt.title("Arnold Watermark", fontsize=12, loc="center")
plt.axis('off')
plt.imshow(waterImg)

plt.subplot(2, 2, 3)
plt.title("Original", fontsize=12, loc="center")
plt.axis('off')
plt.imshow(Img)

plt.subplot(2, 2, 4)
plt.title("Watermarked", fontsize=12, loc="center")
plt.axis('off')
plt.imshow(merged)

plt.savefig('test.jpg', dpi=400)
plt.show()


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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

参加了深圳线下组织的商业沙龙,收获颇丰

能参加本次商业沙龙也本是一连串的事情导致的,听我慢慢道来: 大概在4.13之前无意收到忘了谁发的短信了,说有个啥AI的报告会,然后我就报名了: 会上大佬还是挺多的,来了好多各界的老板和政府的领导&#xff…

配置 IDEA 识别自定义规则的 Dockerfile 文件

目录 配置所在位置解决方案其他 配置所在位置 打开 IntelliJ IDEA,然后转到顶部菜单中的 “File” > “Settings”(Windows/Linux)或 “IntelliJ IDEA” > “Preferences”(macOS)。 在弹出的设置窗口中&#x…

10分钟了解Flink SQL使用

Flink 是一个流处理和批处理统一的大数据框架,专门为高吞吐量和低延迟而设计。开发者可以使用SQL进行流批统一处理,大大简化了数据处理的复杂性。本文将介绍Flink SQL的基本原理、使用方法、流批统一,并通过几个例子进行实践。 1、Flink SQL基…

AI论文写作生成器-AI自动生成论文-10分钟/万字论文

在当今这个高速发展的信息时代,科技的进步无疑给我们的工作、学习乃至生活带来了翻天覆地的变化。学术研究领域也不例外,近期一款新型的论文自动写作神器诞生了,它凭借人工智能技术的优势,为学者和研究人员撰写学术论文提供了极大…

c++opencv Project3 - License Plate Detector

俄罗斯车牌识别案例:实时识别车牌,并且读取到指定文件夹中。 惯例先展示结果图: 对于摄像头读取图片进行车牌匹配,原理和人脸识别其实是一致的。 利用训练好的模型进行匹配即可。可参考: 对视频实现人脸识别-CSDN博…

综合模型及应用(图论学习总结部分内容)

文章目录 前言六、综合模型及应用(以题目总结为主)分层图思想(包括拆点建图) e g 1 : 通信线路 eg1:通信线路 eg1:通信线路​​​[A-Telephone Lines](https://ac.nowcoder.com/acm/contest/1055/A)(蓝书例题) e g 2 : 小雨坐地铁 eg2:小雨坐地铁 eg2:小雨坐地铁​ [1012-小雨坐…

六甲基二硅烷用途广泛 可以工业副产物为原料进行生产

六甲基二硅烷用途广泛 可以工业副产物为原料进行生产 六甲基二硅烷,化学式为C6H18Si2,外观为无色透明液体状,不溶于水,可溶于乙醚、乙二醇、丙酮、苯、二甲苯、二甲醚等多种有机溶剂,有刺激性,有高度易燃性…

Agilent MSO9404A、Keysight MSO9404A示波器,4 GHz,4 通道,20 GSa/s

Agilent MSO9404A、Keysight MSO9404A、HP MSO9404A 示波器,4 GHz,4 通道,20 GSa/s Keysight MSO9404A 示波器配备 15 英寸 XGA 显示屏,封装深度仅为 9 英寸(23 厘米),重量仅为 26 磅&#xff…

传统鞋业如何转型?3D数字化技术让鞋业品牌焕发新机!

数字经济时代,3D数字化技术在各行业都得到广泛应用,这其中,传统的鞋服行业的发展也受到了3D数字化技术的影响,产生了深刻的变化,越来越多的鞋企品牌开始尝试3D数字化营销。 比如,时尚运动品牌VANS就在官网上…

【Obsidian】视频笔记插件Media Extended的强大功能

我将开设一个专栏,介绍当下最好用的笔记软件Obsidian的使用经验和技巧。欢迎持续关注。 摘要:本文将首先向您介绍一款功能强大的笔记软件Obsidian,然后为您详细解析Obsidian的一款实用插件——Media Extended,帮助您更好地利用Obs…

【2024年5月备考新增】】 考前篇(1)《官方平台 - 考生模拟练习平台操作指南》

1 登录 登录中国计算机技术职业资格网(https://www.ruankao.org.cn),点击服务园地的【模拟练习】。 温馨提示:实名认证通过且注册成功的考生方可登录模拟练习。 2 下载模拟作答系统 温馨提示: 点击“下载”按钮,下载对应的模拟作答系统。未报名成功的考生不允许下载…

JPA@Entry报错Could not determine recommended JdbcType for Java type

问题很明显,无法自动决定类型,那就手动告诉该字段。 一、直接上解决方案 如果是一对一的关系用 OneToOne 如果是一对多的关系用 OneToMany 如果是多对一的关系用 ManyToOne 二、另一个无空构造函数的问题 使用注解后,注解报错找不到空的…

mac定时任务、自启动任务

https://quail.ink/mynotes/p/mac-startup-configuration-detailed-explanation <?xml version"1.0" encoding"UTF-8"?> <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.d…

Visual Studio 中.net8.0(以前叫NET Core)框架和.net framewok 框架有什么区别?

更新vs到2022版本后&#xff0c;新建项目时就多出不少选项&#xff0c;这里来给大家分享下.net8.0&#xff08;以前叫NET Core&#xff09;框架和.net framewok的区别 如下图&#xff0c;不带后缀的就是使用.net8.0。 .net framewok框架选项&#xff1a; 正文开始&#xff1a;…

Ci24R1 (SOP8)2.4GHz无线收发一体、双向系统的智能家居芯片

Ci24R1 &#xff08;SOP8&#xff09;工作范围在2.4GHzISM频段&#xff0c;专为低系统应用成本的无线场合设计&#xff0c;集成嵌入式ARQ基带协议引擎的无线收发器芯片。它的工作频率范围为2400MHz-2525MHz&#xff0c;共有126个1MHz带宽的信道。 Ci24R1 &#xff08;SOP8&…

ThreadLocal 源码详解

概述 ThreadLocal是一个java提供的本地线程副本变量工具类。主要用于将私有线程和该线程存放的副本对象做一个映射&#xff0c;各个线程之间的变量互不干扰&#xff0c;在高并发场景下&#xff0c;可以实现无状态的调用&#xff0c;特别适用于各个线程依赖不通的变量值完成操作…

外卖点餐小程序平台源码系统 自由切换 轻松管理 附带源码代码包以及系统搭建教程

系统概述 外卖点餐小程序平台源码系统是一款集点餐、支付、配送、评价等功能于一体的综合性平台。该系统采用先进的互联网技术&#xff0c;支持多种支付方式&#xff0c;包括微信支付、支付宝等&#xff0c;同时支持多平台使用&#xff0c;包括微信小程序、支付宝小程序等。商…

游戏找不到steam_api64.dll如何解决,介绍5种简单有效的方法

面对“找不到steam_api64.dll&#xff0c;无法继续执行代码”的问题&#xff0c;许多游戏玩家或软件使用者可能会感到手足无措。这个错误提示意味着你的计算机系统在尝试运行某个游戏或应用程序时&#xff0c;无法定位到一个至关重要的动态链接库文件——steam_api64.dll&#…

大模型与AIGC应用相关问题 模型大型

最近经常被问&#xff0c;你看“万亿的模型都出来了&#xff0c;你们训练的千亿模型是不是落伍了&#xff1f;”我想说&#xff1a;“虽然都叫超大模型&#xff0c;但是类型是不一样的&#xff0c;虽说每一类模型训出来都不容易&#xff0c;不过澄清一下概念还是必要的”。 大…

C# WinForm —— 17 MaskedTextBox 介绍

1. 简介 本质是文本框&#xff0c;但它可以通过掩码来区分输入的正确与否&#xff0c;可以控制输入的格式、长度 主要应用场景是&#xff1a;需要格式化输入信息的情况 2. 常用属性 属性解释(Name)控件ID&#xff0c;在代码里引用的时候会用到,一般以 mtxt 开头AsciiOnly是否…