当前位置: 首页 > article >正文

Python实现音频数字水印方法

数字水印技术可以将隐藏信息嵌入到音频文件中而不明显影响音频质量。下面我将介绍几种在Python中实现音频数字水印的方法。

方法一:LSB (最低有效位) 水印

import numpy as np

from scipy.io import wavfile

def embed_watermark_lsb(audio_path, watermark, output_path):

    # 读取音频文件

    sample_rate, audio_data = wavfile.read(audio_path)    

    # 确保是立体声,如果是单声道则转换为立体声

    if len(audio_data.shape) == 1:

        audio_data = np.column_stack((audio_data, audio_data))   

    # 将水印转换为二进制

    watermark_bin = ''.join(format(ord(c), '08b') for c in watermark)

    watermark_bin += '00000000' # 添加结束标记   

    # 检查水印是否适合音频

    if len(watermark_bin) > audio_data.size:

        raise ValueError("水印太大,无法嵌入到音频中")  

    # 嵌入水印到最低有效位

    watermark_index = 0

    for i in range(len(audio_data)):

        for j in range(len(audio_data[i])):

            if watermark_index < len(watermark_bin):

                # 替换最低有效位

                audio_data[i][j] = (audio_data[i][j] & 0xFE) | int(watermark_bin[watermark_index])

                watermark_index += 1

            else:

                break   

    # 保存带水印的音频

    wavfile.write(output_path, sample_rate, audio_data)

def extract_watermark_lsb(audio_path, watermark_length):

    # 读取音频文件

    sample_rate, audio_data = wavfile.read(audio_path)    

    # 提取最低有效位

    watermark_bits = []

    for i in range(len(audio_data)):

        for j in range(len(audio_data[i])):

            watermark_bits.append(str(audio_data[i][j] & 1))   

    # 将比特转换为字节

    watermark = ''

    for i in range(0, len(watermark_bits), 8):

        byte = ''.join(watermark_bits[i:i+8])

        if byte == '00000000': # 遇到结束标记

            break

        watermark += chr(int(byte, 2))   

    return watermark[:watermark_length]

# 使用示例

embed_watermark_lsb('original.wav', '秘密消息', 'watermarked.wav')

extracted = extract_watermark_lsb('watermarked.wav', 4)

print("提取的水印:", extracted)

方法二:频域水印 (DCT变换)

import numpy as np

from scipy.fftpack import dct, idct

from scipy.io import wavfile

def embed_watermark_dct(audio_path, watermark, output_path, alpha=0.01):

    # 读取音频

    sample_rate, audio_data = wavfile.read(audio_path)  

    # 如果是立体声,只使用一个声道

    if len(audio_data.shape) > 1:

        audio_data = audio_data[:, 0]   

    # 将水印转换为二进制

    watermark_bin = ''.join(format(ord(c), '08b') for c in watermark)

    watermark_bin = [int(b) for b in watermark_bin]    

    # 分段处理音频

    segment_size = 1024

    num_segments = len(audio_data) // segment_size

    watermark_length = len(watermark_bin)   

    if num_segments < watermark_length:

        raise ValueError("音频太短,无法嵌入水印")   

    # 嵌入水印

    watermarked_audio = np.copy(audio_data)

    for i in range(watermark_length):

        start = i * segment_size

        end = start + segment_size        

        segment = audio_data[start:end]

        dct_coeffs = dct(segment, norm='ortho')       

        # 修改中频系数嵌入水印

        coeff_index = 100 # 选择一个中频系数

        if watermark_bin[i] == 1:

            dct_coeffs[coeff_index] += alpha * np.abs(dct_coeffs[coeff_index])

        else:

            dct_coeffs[coeff_index] -= alpha * np.abs(dct_coeffs[coeff_index])      

        # 逆DCT变换

        watermarked_segment = idct(dct_coeffs, norm='ortho')

        watermarked_audio[start:end] = watermarked_segment   

    # 保存带水印的音频

    wavfile.write(output_path, sample_rate, watermarked_audio.astype(np.int16))

def extract_watermark_dct(audio_path, original_path, watermark_length):

    # 读取带水印音频和原始音频

    sample_rate, watermarked = wavfile.read(audio_path)

    _, original = wavfile.read(original_path)    

    # 如果是立体声,只使用一个声道

    if len(watermarked.shape) > 1:

        watermarked = watermarked[:, 0]

        original = original[:, 0]   

    segment_size = 1024

    watermark_bits = []    

    for i in range(watermark_length):

        start = i * segment_size

        end = start + segment_size       

        wm_segment = watermarked[start:end]

        orig_segment = original[start:end]        

        wm_dct = dct(wm_segment, norm='ortho')

        orig_dct = dct(orig_segment, norm='ortho')        

        coeff_index = 100

        if wm_dct[coeff_index] > orig_dct[coeff_index]:

            watermark_bits.append('1')

        else:

            watermark_bits.append('0')    

    # 将比特转换为字符串

    watermark = ''

    for i in range(0, len(watermark_bits), 8):

        byte = ''.join(watermark_bits[i:i+8])

        watermark += chr(int(byte, 2))  

    return watermark

# 使用示例

embed_watermark_dct('original.wav', '秘密', 'watermarked_dct.wav', 0.02)

extracted = extract_watermark_dct('watermarked_dct.wav', 'original.wav', 16)

print("提取的水印:", extracted)

方法三:扩频水印

import numpy as np

from scipy.io import wavfile

def generate_pn_sequence(length, seed=42):

    np.random.seed(seed)

    return np.random.choice([-1, 1], size=length)

def embed_watermark_spread_spectrum(audio_path, watermark, output_path, alpha=0.01):

    # 读取音频

    sample_rate, audio_data = wavfile.read(audio_path)    

    # 如果是立体声,只使用一个声道

    if len(audio_data.shape) > 1:

        audio_data = audio_data[:, 0]    

    # 将水印转换为二进制

    watermark_bin = ''.join(format(ord(c), '08b') for c in watermark)

    watermark_bits = np.array([int(b) for b in watermark_bin])

    watermark_bits = 2 * watermark_bits - 1 # 转换为±1   

    # 生成伪随机序列

    pn_length = len(audio_data) // len(watermark_bits)

    pn_sequence = generate_pn_sequence(pn_length)   

    # 创建扩频水印

    spread_watermark = np.repeat(watermark_bits, pn_length)

    spread_watermark = spread_watermark[:len(audio_data)] * pn_sequence[:len(audio_data)]   

    # 嵌入水印

    watermarked_audio = audio_data + alpha * spread_watermark * np.abs(audio_data)

    watermarked_audio = np.clip(watermarked_audio, -32768, 32767) # 确保在16位范围内 

    # 保存带水印的音频

    wavfile.write(output_path, sample_rate, watermarked_audio.astype(np.int16))

def extract_watermark_spread_spectrum(audio_path, original_path, watermark_length, pn_length):

    # 读取音频

    sample_rate, watermarked = wavfile.read(audio_path)

    _, original = wavfile.read(original_path)   

    # 如果是立体声,只使用一个声道

    if len(watermarked.shape) > 1:

        watermarked = watermarked[:, 0]

        original = original[:, 0]    

    # 计算差异

    diff = watermarked - original   

    # 生成相同的伪随机序列

    num_bits = watermark_length * 8

    pn_sequence = generate_pn_sequence(pn_length)   

    extracted_bits = []

    for i in range(num_bits):

        start = i * pn_length

        end = start + pn_length       

        segment_diff = diff[start:end]

        segment_pn = pn_sequence[:len(segment_diff)]        

        correlation = np.sum(segment_diff * segment_pn)

        extracted_bits.append('1' if correlation > 0 else '0')    

    # 将比特转换为字符串

    watermark = ''

    for i in range(0, len(extracted_bits), 8):

        byte = ''.join(extracted_bits[i:i+8])

        watermark += chr(int(byte, 2))    

    return watermark

# 使用示例

embed_watermark_spread_spectrum('original.wav', '秘密', 'watermarked_ss.wav', 0.01)

extracted = extract_watermark_spread_spectrum('watermarked_ss.wav', 'original.wav', 2, 1000)

print("提取的水印:", extracted)

 

## 注意事项

1. **音频质量**:水印嵌入会影响音频质量,需要平衡水印强度和音频质量。

2. **鲁棒性**:不同方法对音频处理的抵抗能力不同:

   - LSB方法脆弱但容量大

   - DCT方法对压缩有一定抵抗能力

   - 扩频方法鲁棒性最强但容量小

3. **安全性**:可以考虑加密水印内容提高安全性

4. **格式支持**:示例中使用WAV格式,因其是无损格式,其他格式可能需要先解码

 扩展建议

1. 添加错误校正码提高水印提取的可靠性

2. 实现盲水印提取(不需要原始音频)

3. 添加同步信号提高对裁剪、时间拉伸的抵抗能力

4. 结合多种技术提高水印的鲁棒性和隐蔽

这些方法可以根据具体需求进行调整和组合,以实现不同场景下的音频数字水印需求。

相关文章:

Python实现音频数字水印方法

数字水印技术可以将隐藏信息嵌入到音频文件中而不明显影响音频质量。下面我将介绍几种在Python中实现音频数字水印的方法。 方法一&#xff1a;LSB (最低有效位) 水印 import numpy as np from scipy.io import wavfile def embed_watermark_lsb(audio_path, watermark, ou…...

快速入手-基于Django-rest-framework的第三方认证插件(SimpleJWT)权限认证扩展返回用户等其他信息(十一)

1、修改serializer.py&#xff0c;增加自定义类 # 自定义用户登录token等返回信息 class MyTokenObtainPair(TokenObtainPairView): def post(self, request, *args, **kwargs): serializer self.get_serializer(datarequest.data) try: serializer.is_valid(raise_exceptio…...

关于IP免实名的那些事

IP技术已成为个人与企业保护隐私、提升网络效率的重要工具。其核心原理是通过中介服务器转发用户请求&#xff0c;隐藏真实IP地址&#xff0c;从而实现匿名访问、突破地域限制等目标。而“免实名”代理IP的出现&#xff0c;进一步简化了使用流程&#xff0c;用户无需提交身份信…...

【SQL性能优化】预编译SQL:从注入防御到性能飞跃

&#x1f525; 开篇&#xff1a;直面SQL的"阿喀琉斯之踵" 假设你正在开发电商系统&#x1f6d2;&#xff0c;当用户搜索商品时&#xff1a; -- 普通SQL拼接&#xff08;危险&#xff01;&#xff09; String sql "SELECT * FROM products WHERE name "…...

Spring容器从启动到关闭的注解使用顺序及说明

Spring容器从启动到关闭的注解使用顺序及说明 1. 容器启动阶段 注解&#xff1a;Configuration、ComponentScan 作用&#xff1a; Configuration&#xff1a;标记配置类&#xff0c;声明Spring应用上下文的配置源。ComponentScan&#xff1a;扫描指定包下的组件&#xff08;B…...

UVM概念面试题100问

1-10:UVM概述 Q1: 什么是UVM? A1: UVM是Universal Verification Methodology的缩写,它是由Accellera标准化的一种用于IC验证的方法学。它提供了一个基类库(BCL),包含通用工具如组件层次结构、事务级模型(TLM)和配置数据库等,使用户能够创建结构化、可重用的验证环境。 Q2:…...

SQL Server从安装到入门一文掌握应用能力。

本篇文章主要讲解,SQL Server的安装教程及入门使用的基础知识,通过本篇文章你可以快速掌握SQL Server的建库、建表、增加、查询、删除、修改等基本数据库操作能力。 作者:任聪聪 日期:2025年3月31日 一、SQL Server 介绍: SQL Server 是微软旗下的一款主流且优质的数据库…...

力扣HOT100之矩阵:54. 螺旋矩阵

这道题之前在代码随想录里刷过类似的&#xff0c;还有印象&#xff0c;我就按照当初代码随想录的思路做了一下&#xff0c;结果怎么都做不对&#xff0c;因为按照代码随想录的边界条件设置&#xff0c;当行数和列数都为奇数时&#xff0c;最后一个元素无法被添加到数组中&#…...

5.1 WPF路由事件以及文本样式

一、路由事件 WPF中存在一种路由事件&#xff08;routed event&#xff09;&#xff0c;该事件将发送到包含该控件所在层次的所有控件&#xff0c;如果不希望继续向更高的方向传递&#xff0c;只要设置e.Handled true即可。 这种从本控件-->父控件->父的父控件的事件&am…...

Python数据可视化-第1章-数据可视化与matplotlib

环境 开发工具 VSCode库的版本 numpy1.26.4 matplotlib3.10.1 ipympl0.9.7教材 本书为《Python数据可视化》一书的配套内容&#xff0c;本章为第1章 数据可视化与matplotlib 本文主要介绍了什么是数据集可视化&#xff0c;数据可视化的目的&#xff0c;常见的数据可视化方式…...

Flutter敏感词过滤实战:基于AC自动机的高效解决方案

Flutter敏感词过滤实战&#xff1a;基于AC自动机的高效解决方案 在社交、直播、论坛等UGC场景中&#xff0c;敏感词过滤是保障平台安全的关键防线。本文将深入解析基于AC自动机的Flutter敏感词过滤实现方案&#xff0c;通过原理剖析实战代码性能对比&#xff0c;带你打造毫秒级…...

20250331-vue-组件事件1触发与监听事件

触发与监听事件 1 在组件的模板表达式中&#xff0c;可以直接使用 $emit 方法触发自定义事件(例如&#xff1a;在 v-on 的处理函数中)&#xff1a; 子组件代码 <template><button click"$emit(someEvent)">点击</button> </template><…...

Odoo/OpenERP 和 psql 命令行的快速参考总结

Odoo/OpenERP 和 psql 命令行的快速参考总结 psql 命令行选项 选项意义-a从脚本中响应所有输入-A取消表数据输出的对齐模式-c <查询>仅运行一个简单的查询&#xff0c;然后退出-d <数据库名>指定连接的数据库名&#xff08;默认为当前登录用户名&#xff09;-e回显…...

Vue中使用antd-table组件时,树形表格展开配置不生效-defaultExpandedRowKeys-默认展开配置不生效

defaultExpandedRowKeys属性 defaultExpandAllRows这个属性仅仅是用来设置默认值的,只在第一次渲染的时候起作用,后续再去改变,无法实现响应式 解决方案一 a-table表格添加key属性,当每次获取值时,动态改变key,以达到重新渲染的效果 <a-table:key="tableKey"…...

VRRP交换机三层架构综合实验

题目要求&#xff1a; 1&#xff0c;内网Ip地址使用172.16.0.0/16分配 说明可以划分多个子网&#xff0c;图中有2个VLAN&#xff0c;可以根据VLAN划分 2&#xff0c;sw1和SW2之间互为备份 互为备份通常通过VRRP&#xff08;虚拟路由冗余协议&#xff09;来实现。VRRP会在两个…...

基于卷积神经网络的眼疾识别系统,resnet50,efficentnet(pytorch框架,python代码)

更多图像分类、图像识别、目标检测、图像分割等项目可从主页查看 功能演示&#xff1a; 眼疾识别系统resnet50&#xff0c;efficentnet&#xff0c;卷积神经网络&#xff08;pytorch框架&#xff0c;python代码&#xff09;_哔哩哔哩_bilibili &#xff08;一&#xff09;简介…...

基于srpingboot智慧校园管理服务平台的设计与实现(源码+文档+部署讲解)

技术范围&#xff1a;SpringBoot、Vue、SSM、HLMT、Jsp、PHP、Nodejs、Python、爬虫、数据可视化、小程序、安卓app、大数据、物联网、机器学习等设计与开发。 主要内容&#xff1a;免费功能设计、开题报告、任务书、中期检查PPT、系统功能实现、代码编写、论文编写和辅导、论…...

【力扣hot100题】(026)合并两个有序链表

可以创建一个新链表记录答案&#xff1a; /*** Definition for singly-linked list.* struct ListNode {* int val;* ListNode *next;* ListNode() : val(0), next(nullptr) {}* ListNode(int x) : val(x), next(nullptr) {}* ListNode(int x, ListNode *…...

TCP网络编程与多进程并发实践

一、引言 在网络编程中&#xff0c;TCP&#xff08;传输控制协议&#xff09;是一种面向连接的、可靠的、基于字节流的传输层通信协议。而多进程并发则是一种提高服务器处理能力的有效手段&#xff0c;允许服务器同时处理多个客户端的请求。本文将详细介绍如何使用 TCP 协议进…...

【前端】一文掌握 Vue 3 指令用法(vue3 备忘清单)

文章目录 入门介绍创建应用应用实例通过 CDN 使用 Vue使用 ES 模块构建版本模板语法文本插值原始 HTMLAttribute 绑定布尔型 Attribute动态绑定多个值使用 JavaScript 表达式仅支持表达式(例子都是无效)调用函数指令 Directives参数 Arguments绑定事件动态参数动态的事件名称修…...

visio导出pdf公式变形

情况描述导出为pdf后&#xff0c;mathtype写的公式就变形了 但是导出为png和jpg就是正常 解决方法就是 需要下载一个Adobe Acrobat...

【学习笔记】计算机网络(六)

第6章应用层 文章目录 第6章应用层6.1 域名系统DNS6.1.1 域名系统概述6.1.2 互联网的域名结构6.1.3 域名服务器域名服务器的分区管理DNS 域名服务器的层次结构域名服务器的可靠性域名解析过程-两种查询方式DNS 高速缓存机制 6.2 文件传送协议6.2.1 FTP 概述6.2.2 FTP 的基本工作…...

做一个多级动态表单,可以保存数据和回显数据

<template> <div class"two"> <button class"save" click"saveBtn">保存数据</button> <button class"sd" click"showBtn">回显数据</button> <div class"all" click&quo…...

量子退火与机器学习(2):少量实验即可找到新材料,黑盒优化➕量子退火

使用量子退火和因子分解机设计新材料 这篇文章是东京大学的一位博士生的毕业论文中的主要贡献。 结合了黑盒优化和量子退火&#xff0c;是融合的非常好的一篇文章&#xff0c;在此分享给大家。 https://journals.aps.org/prresearch/abstract/10.1103/PhysRevResearch.2.0133…...

WPF中的Adorner基础用法详解与实例

WPF中的Adorner基础用法详解与实例 Adorner&#xff08;装饰器&#xff09;是WPF中一个强大的功能&#xff0c;它允许开发者在现有UI元素之上叠加额外的视觉效果或交互功能&#xff0c;而不会影响原有布局系统。本文将详细介绍Adorner的基础概念、核心用法以及实际应用示例。 …...

【React】基于 React+Tailwind 的 EmojiPicker 选择器组件

1.背景 React 写一个 EmojiPicker 组件&#xff0c;基于 emoji-mart 组件二次封装。支持添加自定义背景 、Emoji 图标选择&#xff01;并在页面上展示&#xff01; 2.技术栈 emoji-mart/data 、emoji-mart : emoji 图标库、元数据 tailwindcss: 原子化 CSS 样式库 antd : 组…...

02-Docker 使用

docker:快速构建、运行、管理应用的工具,可以帮助我们下载应用镜像,创建并运行镜像的容器,从而快速部署应用 1、部署mysql 先停掉虚拟机中的MySQL,确保你的虚拟机已经安装Docker,且网络开通的情况下,执行下面命令即可安装MySQL(注意:若服务器上已经有mysql 占用了330…...

html5时钟升级!支持切换深浅模式 Canvas实现现代化动态时钟

HTML5 Canvas实现现代化动态时钟 这里写目录标题 HTML5 Canvas实现现代化动态时钟项目介绍技术实现1. 项目架构2. Canvas绘图实现2.1 表盘绘制2.2 刻度绘制2.3 指针绘制 3. 动画效果4. 主题切换 项目亮点技术要点总结项目收获改进方向结语 项目介绍 本项目使用HTML5 Canvas技术…...

MOE-1 基本认识

解读一下MOE架构&#xff0c;部分内容图片参考自油管。 首先来简单了解一下什么是MoE&#xff08;Mixture of Experts&#xff0c;专家混合&#xff09; MoE&#xff08;Mixture of Experts&#xff09;是一种深度学习架构&#xff0c;其核心思想是通过**多个专家网络&#xf…...

【C++接入大模型】WinHTTP类封装:实现对话式大模型接口访问

一、类设计概述 近期准备用C做一些大预言模型方面的开发&#xff0c;先期计划实现C调用公共的大模型Web接口&#xff0c;因为之前没做过C的Web开发&#xff0c;经验少&#xff0c;所以对比了一些主流的框架&#xff0c;包括实际测试验证。以下是Windows平台下主流C HTTP库的对…...