OpenCV实现数字水印的相关函数和示例代码

OpenCV计算机视觉开发实践:基于Qt C++ - 商品搜索 - 京东

实现数字水印的相关函数

用OpenCV来实现数字水印功能,需要使用一些位操作函数,我们需要先了解一下这些函数。

1. bitwise_and函数

bitwise_and函数是OpenCV中的位运算函数之一,用于对两幅二值图像进行按位与操作。具体来说,对于每个像素,将两幅输入图像相应位置的像素值分别进行按位与运算,输出的结果图像的对应像素值即为这两幅输入图像对应像素值的按位与结果。

bitwise_and函数的语法如下:

void bitwise_and(InputArray src1, InputArray src2,OutputArray dst, InputArray mask = noArray());

其中,src1和src2表示要进行按位与操作的两幅输入图像;mask是可选参数,如果指定了掩膜,则只对掩膜对应位置的像素进行按位与操作;dst表示按位与运算的结果。

【例14.1】创建空白图像进行按位与操作

   打开Qt Creator,新建一个控制台项目,项目名称是test。

   在main.cpp中输入如下代码:

#include "opencv2/opencv.hpp"
using namespace cv;int main()
{// 空白图像创建Mat m1 = Mat::zeros(Size(256, 256), CV_8UC3);Mat m2 = Mat::zeros(Size(256, 256), CV_8UC3);// 在图像内添加矩阵rectangle(m1, Rect(100, 100, 80, 80), Scalar(255, 255, 0), -1, LINE_8, 0);rectangle(m2, Rect(150, 150, 80, 80), Scalar(0, 255, 255), -1, LINE_8, 0);imshow("m1", m1);imshow("m2", m2);Mat dst;bitwise_and(m1, m2, dst); // 进行与操作,结果存于dst中imshow("result", dst);waitKey();return 0;
}

通过bitwise_and函数就能得到与操作的结果dst。

   运行程序,结果如图14-1所示。

图14-1

再看个示例,对现有图片进行按位与操作,代码如下:

【例14.2】对现成图像进行按位与操作

   打开Qt Creator,新建一个控制台项目,项目名称是test。

   在main.py中输入如下代码:

#include "opencv2/opencv.hpp"
using namespace cv;int main()
{Mat dog,cat,img_and;resize(imread("cat.png"),cat, Size(400, 360));resize(imread("dog.png"),dog, Size(400, 360));bitwise_and(cat,dog,img_and); // 与运算 1 & 1 = 1, 其它为0imshow("result",img_and);waitKey(0);return 0;
}

   保存程序并运行,结果如图14-2所示。

可以看出,与运算的结果是使图像变小,最后的图像也会偏暗。

2. bitwise_or函数

在OpenCV中进行或运算使用bitwise_or函数,其声明如下:

void bitwise_or(InputArray src1, InputArray src2, OutputArray dst, InputArray mask = noArray());

其中输入参数src1和src2可为灰度图或彩色图,src1和src2大小需相同;输出参数dst存放或运算的结果,尺寸和类型与src保持一致;掩膜mask可通俗理解为一个遮罩,只对mask设定的有效区域进行操作。

图14-2

或运算 0 | 0 = 0,其他情况为1。下面将猫和狗的图片进行或运算。

【例14.3】对现成图像进行按位或操作

   打开Qt Creator,新建一个控制台项目,项目名称是test。

   在main.cpp中输入如下代码:

#include "opencv2/opencv.hpp"
using namespace cv;int main()
{Mat dog,cat,img_and;resize(imread("cat.png"),cat, Size(400, 360));resize(imread("dog.png"),dog, Size(400, 360));bitwise_or(cat,dog,img_and); // 或运算 1 & 1 = 1,其他为0imshow("result",img_and);waitKey(0);return 0;
}

   保存程序并运行,结果如图14-3所示。

图14-3

可以看出,或运算的结果是使图像变大,最后的图像也偏亮。

代码实现数字水印

前面讲解了数字水印的嵌入过程和提取过程,步骤比较清晰。本节将根据这些步骤,通过代码来实现数字水印的嵌入和提取。

【例14.4】实现数字水印的嵌入和提取

   打开Qt Creator,新建一个控制台项目,项目名称是test。

   在main.cpp中输入如下代码:

#include "opencv2/opencv.hpp"
using namespace cv;int main()
{Mat gray1H7,dst,getWatermark;Mat src= imread("src.bmp",0);// 读取水印图像Mat watermark= imread("watermark.bmp",0);// 将水印图像内的正值处理为1,以方便嵌入,相当于把水印图像变成二值图像for(int i=0;i<watermark.rows;i++){for(int j=0;j<watermark.cols;j++){if(watermark.at<uchar>(i,j)>0)watermark.at<uchar>(i,j)=1;}}// 读取原始载体图像的行和列int r=src.rows;int c=src.cols;// ----------------------嵌入过程------------------------// 生成元素值都是254的数组Mat t254 = Mat::ones(r, c, CV_8UC1) * 254;// 获取gray1H7图像的高七位bitwise_and(src,t254,gray1H7);// 将watermark嵌入gray1H7内bitwise_or(gray1H7,watermark,dst);// ----------------------提取过程--------------------------// 生成元素值都是1的数组Mat t1=  Mat::ones(r, c, CV_8UC1);// 从目标载体图像内提取水印图像bitwise_and(dst,t1,getWatermark);// 将水印图像内的正值处理为255,以方便显示for(int i=0;i<getWatermark.rows;i++){for(int j=0;j<getWatermark.cols;j++){if(getWatermark.at<uchar>(i,j)>0)getWatermark.at<uchar>(i,j)=255;}}// ---------显示结果--------imshow("srcImg",src);imshow("watermark",watermark*255); // 当前watermark内最大值为1imshow("dstImg",dst);imshow("getWatermark",getWatermark);waitKey();destroyAllWindows();waitKey(0);return 0;
}

可以看出,上述代码是按照嵌入过程的步骤和提取过程的步骤来实现的。我们把一副水印图像(watermark.bmp)嵌入原始载体图像(src.bmp)中变为目标载体图像(也称含水印的载体图像)dst,然后从dst中提取出水印数据存于getWatermark中,最后显示4种图像。

   运行程序,结果如图14-4所示。

从图14-4中可以看到,原始载体图像(src)和含水印的载体图像(dst)肉眼是看不出区别的。下面再看水印图像,如图14-5所示。

图14-4

图14-5

在图14-5中,左边是原来的数字水印图像,右边是从目标载体图像中提取出来的数字水印图像,可以发现两者并没有变化。这样我们就实现了把一副数字水印图像嵌入载体图像再提取出的过程。

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

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

相关文章

基于Python的计算机科学研究话题管理系统的设计与实现 - 爬虫

标题:基于Python的计算机科学研究话题管理系统的设计与实现 - 爬虫 内容:1.摘要 本文聚焦于基于Python的计算机科学研究话题管理系统的爬虫部分。背景是随着计算机科学研究的快速发展&#xff0c;相关话题数据海量且分散&#xff0c;人工管理效率低。目的是设计并实现一个能高…

告别手动解析!借助 CodeBuddy 快速开发网页源码提取工具

作为一名长期从事 Web 开发的程序员&#xff0c;我们在日常工作中&#xff0c;时不时会需要查看网页的源代码。这么做的目的通常是为了排查前端渲染的问题、分析接口返回的数据结构&#xff0c;或者就是单纯地想快速提取页面中的某些信息&#xff0c;比如文章链接、图片地址&am…

为什么要在 input() 后加 .strip()?

strip() 是 Python 字符串的一个方法&#xff0c;用于去除字符串开头和结尾的空白字符&#xff08;包括空格、制表符 \t、换行符 \n 等&#xff09;。 为什么要在 input() 后加 .strip()&#xff1f; 用户在输入时&#xff0c;可能会不小心在开头或结尾输入空格&#xff0c;例…

【日撸 Java 300行】Day 14(栈)

目录 Day 14&#xff1a;栈 一、栈的基本知识 二、栈的方法 1. 顺序表实现栈 2. 入栈 3. 出栈 三、代码及测试 拓展&#xff1a; 小结 Day 14&#xff1a;栈 Task&#xff1a; push 和 pop 均只能在栈顶操作.没有循环, 时间复杂度为 O(1). 一、栈的基本知识 详细的介…

dotnet core c#调用Linux c++导出函数

1.声明C++导出函数 platform_export.h // // Created by dev on 5/6/25. //#ifndef PLATFORM_EXPORT_H #define PLATFORM_EXPORT_H #if defined(_WIN32)#ifdef LIB_EXPORTS#define LIB_API __declspec(dllimport)#else#define LIB_API __declspec(dllimport)#endif #else#ifde…

SparkSQL操作Mysql

前面的课程我们学习了如何从csv文件中读入数据&#xff0c;这相当于是对csv这种类型的数据的操作。那么接下来&#xff0c;我们一起看看&#xff0c;如何写Spark程序来操作mysql数据库。先来给大家介绍一下我们这节课的主要学习内容&#xff1a; &#xff08;1&#xff09;安装…

语言学中的对象语言与元语言 | 概念 / 区别 / 实例分析

注&#xff1a;英文引文&#xff0c;机翻未校。 语言学中的“对象语言”和“元语言” 刘福长 现代外语 1989年第3期&#xff08;总第45期&#xff09; 在阅读语言学著作时&#xff0c;我们有时会遇到这样两个术语&#xff1a;对象语言&#xff08;object language&#xff0…

livenessProbe 和 readinessProbe 最佳实践

在 Kubernetes 中&#xff0c;livenessProbe 和 readinessProbe 是确保应用高可用性的关键机制&#xff0c;但配置不当可能导致应用频繁重启或流量中断。以下是配置这两个探针的最佳实践&#xff1a; 1. 核心区别与作用 探针类型目的失败后果livenessProbe检测应用是否 存活&…

集成管理工具Gitlab

GitLab 是一个功能强大的开源代码托管和协作平台&#xff0c;集成 GitLab 可以显著提升团队的开发效率。下面我将为你介绍如何集成 GitLab&#xff0c;包括安装配置和基本使用流程。 一、GitLab 安装与配置 GitLab 有多种安装方式&#xff0c;推荐使用官方 Omnibus 包安装&am…

Electron-Vue3、Electron-React、Electron-Angular打造舆情监控系统项目

Electron是一个跨平台的桌面应用开发框架&#xff0c;可以让我们用html css js的技术开发跨平台桌面上可以安装的软件。视频详解: Electron教程 ElectronVue跨平台桌面软件开发教程-2024年更新&#xff08;大地老师&#xff09; 从Electron环境搭建开始到手把手教你调试、Elect…

08.webgl_buffergeometry_attributes_none ,three官方示例+编辑器+AI快速学习

本实例主要讲解内容 这个Three.js示例展示了无属性几何体渲染技术&#xff0c;通过WebGL 2的gl_VertexID特性和伪随机数生成算法&#xff0c;在着色器中动态计算顶点位置和颜色&#xff0c;而不需要在CPU端预先定义几何体数据。 核心技术包括&#xff1a; WebGL 2的顶点ID特…

Ubuntu 22.04搭建OpenStreeMap地址解析服务(保姆级教程)

1.数据准备 1.1.全球数据 下载地址&#xff1a;https://planet.openstreetmap.org/ 1.2.特定区域的数据 下载地址&#xff1a;Geofabrik Download Server 2.安装必要的软件包 2.1.更新系统软件包 sudo apt updatesudo apt upgrade 2.2.安装所需要的软件包 执行下面的命…

Ubuntu 22.04.5 LTS上部署Docker及相关优化

以下是在Ubuntu 22.04.5 LTS上部署Docker及相关优化的步骤&#xff1a; 安装Docker 更新系统&#xff1a;在安装Docker之前&#xff0c;先确保系统是最新的&#xff0c;执行以下命令&#xff1a;sudo apt update sudo apt upgrade -y安装依赖包&#xff1a;安装一些必要的依赖…

React -> AI组件 -> 调用Ollama模型, qwen3:1.7B非常聪明

使用 React 搭建一个现代化的聊天界面&#xff0c;支持与 Ollama 本地部署的大语言模型进行多轮对话。界面清爽、功能完整&#xff0c;支持 Markdown 渲染、代码高亮、<think> 隐藏思考标签、流式渐进反馈、暗黑模式适配等特性。 &#x1f9e9; 核心功能亮点 ✅ 模型选择…

vue2/3 中使用 @vue-office/docx 在网页中预览(docx、excel、pdf)文件

1. 安装依赖&#xff1a; #docx文档预览组件npm install vue-office/docx vue-demi0.14.6#excel文档预览组件npm install vue-office/excel vue-demi0.14.6#pdf文档预览组件npm install vue-office/pdf vue-demi0.14.6 vue2.6版本或以下还需要额外安装 vue/composition-api …

【应用密码学】实验五 公钥密码2——ECC

一、实验要求与目的 1.复习CCC基本概念&#xff0c;并根据实验平台提供的资料完成验证性实验。 2.编程练习&#xff1a;以书上例题小模数p为例编程实现ECC的基本运算规则。 二、实验内容与步骤记录&#xff08;只记录关键步骤与结果&#xff0c;可截图&#xff0c;但注意排版…

rust-candle学习笔记9-使用tokenizers加载qwen3分词,使用分词器处理文本

参考&#xff1a;about-pytorch&#xff0c; about-tokenizers 在魔搭社区链接下载qwen3的tokenizer.json文件 添加依赖库&#xff1a; cargo add tokenizers tokenizers库初体验&#xff1a; use tokenizers::tokenizer::{self, Result, Tokenizer};fn main() -> Resu…

【MySQL】存储引擎 - ARCHIVE、BLACKHOLE、MERGE详解

&#x1f4e2;博客主页&#xff1a;https://blog.csdn.net/2301_779549673 &#x1f4e2;博客仓库&#xff1a;https://gitee.com/JohnKingW/linux_test/tree/master/lesson &#x1f4e2;欢迎点赞 &#x1f44d; 收藏 ⭐留言 &#x1f4dd; 如有错误敬请指正&#xff01; &…

5.Redission

5.1 前文锁问题 基于 setnx 实现的分布式锁存在下面的问题&#xff1a; 重入问题&#xff1a;重入问题是指 获得锁的线程可以再次进入到相同的锁的代码块中&#xff0c;可重入锁的意义在于防止死锁&#xff0c;比如 HashTable 这样的代码中&#xff0c;他的方法都是使用 sync…

C语言主要标准版本的演进与核心区别的对比分析

以下是C语言主要标准版本的演进与核心区别的对比分析 K&R C&#xff08;1978年&#xff09; 定位‌&#xff1a;非标准化的原始版本&#xff0c;由Brian Kernighan和Dennis Ritchie定义 特性‌&#xff1a; 基础语法&#xff1a;函数声明无参数列表&#xff08;如int func…