C++实现网格水印之调试笔记(六)—— 提取完成

昨天在修改了可以调试出来的错误之后,提取出的水印和嵌入的仍然相去甚远。这个时候我觉得有必要整理一下嵌入和提取的整个过程。

嵌入过程:

Step1,嵌入的时候对网格的拉普拉斯矩阵L进行特征值分解,得到特征值和特征向量;

Step2,将特征向量单位化;

Step3,解方程:R . E = V(1),E,V已知求出R

其中R是频谱系数矩阵,E是单位化的特征向量矩阵,V是顶点坐标矩阵

Step4,修改R的R‘,其中R‘= R + △ (2),R,△已知,求R‘

Step5,根据矩阵方程(1)求出V‘, R‘. E = V‘(3),R‘, E已知,求V‘

再来重温一下提取过程:

Step1,将原始水印M和网格水印M‘对齐(放入同一坐标系下)

Step2,计算M的频谱系数R和M‘的频谱系数R‘

Step3,根据下式计算Qj即 △

Step4, 再根据下式计算出嵌入水印的符号,这里相当于得到的是嵌入的水印b‘,只是记为了aj

Step5,最后根据下式得出原始水印序列b

注意,在计算M‘的频谱系数R‘的时候,

用的是R‘. E‘= V‘(4),E‘,V‘已知,求R‘

对比(3)和(4)式,

R‘. E = V‘(3)

R‘. E‘ = V‘(4)

可以看到两个等式中有两个量是相同的,即R‘和V‘而E和E‘很可能是不相等的,所以用(4)式计算出的R‘和(2)R‘= R + △ (2)式中的R‘是不同的,自然提取出来的 △ 也不正确。所以我觉得,是这篇论文本身有问题。。。

我觉得正确的提取方法应该是用式(3),输入水印网格,然后获取V‘矩阵,用原始矩阵的E,两者结合求出R‘,与原始水印的R相减才能得到△

为了验证我的想法,我首先将原始网格中的E矩阵写入文件

//将单位化的特征向量写入文件中

ofstream NormEfile;

NormEfile.open("D:\\Ne.txt",ios_base::out );

if ( NormEfile)

{

for ( int i = 0; i < m_vertexNum; i++)

{

for (int j = 0; j < m_vertexNum; j++)

{

NormEfile<<E_matrix(i,j)<<" ";

}

}

}

NormEfile.close();

但是在读取的过程中遇到了问题。

Matlab计算出的R

获取出的R矩阵

也就是说 wRs应该是获取出的R矩阵的第一行

wRs = R_matrix.col(0);

输出wRs和R矩阵的第一行,结果却便了。。。

结果是循环的索引值写错了,修改后,结果如下:

但是Qi的值仍然错误

调试的时候发现,Ru读取的时候也出错了。。。

改正后问题仍然没有得到解决,观察上面的输出,估计问题是出在Ru上,输出Ru进行观察

原来是每次打开的文件都是Rs,忘记改成相应的文件了。。。。

修改之后,仍然错误。检查读出的pRu

文件中的Ru如下,显然读取错误了。

再检查pRs

读取的结果是正确的。明明是一样的读取方式,为啥会出错?

结果是下面一个判断写错了。。。

今天犯的很多错误都是这方面的,总结出来的教训是一定要仔细。

修改了上述错误后,终于提取出来了!

记录下将二维数组读出文件的方法(写入在前面有记录)

//从文件中将原始网格的E矩阵读取出来

double *pE = new double[m_vertexNum * m_vertexNum];

ifstream NERfile;

NERfile.open("D:\\Ne.txt",ios_base::in );

if (NERfile)

{

// read into memory

NERfile.seekg (0, NERfile.end);

int length = NERfile.tellg();

NERfile.seekg (0, NERfile.beg);

char *buffer = new char[length];

NERfile.read(buffer, length);

NERfile.close();

// parse into array

std::istringstream iss(buffer);

int i = 0;

while (iss >> pE[i++]);

delete [] buffer;

// print or use it.

}

NERfile.close();

int ecount = 0;

for ( int i = 0; i < m_vertexNum; i++ )

{

for ( int j = 0; j < m_vertexNum; j++ )

{

E_matrix(i,j) = pE[ecount++];

}

}

总算提取成功了。

值得注意的是,当模型的顶点个数为奇数时,产生的Wb和P的位数可能比顶点个数多一些,这是因为码片速率和原始的水印序列a的乘积很难刚好等于顶点个数。所以在读取Wb和P的时候应该改写一下while语句,如下:

while ( i<m_vertexNum )

{

iss >> RB[i++];

}

最后,new出来的数组最好用delete语句释放掉。

时间: 2024-08-28 10:31:06

C++实现网格水印之调试笔记(六)—— 提取完成的相关文章

C++实现网格水印之调试笔记(六)——补充

调用matlab生成的网格水印特征向量矩阵 ? 从文件中读取的原始网格的特征向量矩阵 ? 好吧,之前得出的结果不正确是因为代码写错了.因为实现论文中的提取方案时代码写错了,自己想了另外一个方法,结果方向两者在本质上是一样的.

C++实现网格水印之调试笔记(五)—— 提取出错

在实现提取水印的过程中,遇到了一些问题 首先还是根据论文中的思路来梳理一下整个提取流程 读入两个模型,一个原始模型ori_mesh, 一个水印模型wm_mesh. 将两个模型对齐(即放在同一个坐标系下) 分别计算两个模型的频谱系数,记ori_mesh频谱系数为Rs, Rt, Ru, 记wm_mesh的频谱系数为wRs, wRt, wRu. 根据下式计算Qj 再根据下式计算出嵌入水印的符号,这里相当于得到的是嵌入的水印b',只是记为了aj 最后根据下式得出原始水印序列b ? 根据上述步骤,我一开始

C++实现网格水印之调试笔记(四)—— 完成嵌入

接下来的问题是,当模型是对称的时候,结果是符合预期的,但是当模型是不对称的时候,结果是错误的,如下: 输入: 顶点:233 输出: 这又是什么鬼...,我的马呢!!! 看来逻辑上还是有错误 注意这时候C++输出的调试信息如下: 错误提示为:Input to EIG must not contain NaN,然后是一堆烫烫烫... 还有一个提示是: Matrix is close to singular or badly scaled. Results may be inaccurate. RCO

C++实现网格水印之调试笔记(三)—— 初有结果

错误: error C2338: THE_BRACKET_OPERATOR_IS_ONLY_FOR_VECTORS__USE_THE_PARENTHESIS_OPERATOR_INSTEAD 这种错误不会直接定位到行,要自己慢慢看代码,找出不正确的访问. 原因:使用[]访问引起的错误,eigen中矩阵的访问不用用[] 解决方法:将[ ]改为( ) 修改错误之后,编译通过了,但运行时会中断. 调试时,遇到代码不会跳转的问题 当在调试模式下代码运行到eigenDef.calLapMatrix()时,

java之jvm学习笔记六-十二(实践写自己的安全管理器)(jar包的代码认证和签名) (实践对jar包的代码签名) (策略文件)(策略和保护域) (访问控制器) (访问控制器的栈校验机制) (jvm基本结构)

java之jvm学习笔记六(实践写自己的安全管理器) 安全管理器SecurityManager里设计的内容实在是非常的庞大,它的核心方法就是checkPerssiom这个方法里又调用 AccessController的checkPerssiom方法,访问控制器AccessController的栈检查机制又遍历整个 PerssiomCollection来判断具体拥有什么权限一旦发现栈中一个权限不允许的时候抛出异常否则简单的返回,这个过程实际上比我的描述要复杂 得多,这里我只是简单的一句带过,因为这

JavaScript特效(调试笔记)

JavaScript特效 一.在网页上显示当前的时间日期,例如:“2016年3月26日 星期六”. js源代码: 1 function getTime() { 2 var today = new Date(); //返回当日的日期和时间. 3 var year = today.getFullYear(); //获得当前的年份 4 var month = today.getMonth() + 1; //获得当前的月份 5 var day = today.getDate(); //获得当前的日期 6

python之raw_input()(学习笔记六)

python之raw_input()(学习笔记六) 我们经常使用raw_input()读取用户的输入,如下例子所示: >>> name = raw_input('please input your name:'),截图如下: 下面简单说下,raw_input()与if搭配使用,脚本如下: #!/usr/bin/env python # -*- coding:utf-8 -*- birth = raw_input('birth:') if birth < 2000: print '0

《格蠹汇编》调试笔记

以前也就是把Windbg作为一个调试所写驱动的调试器.只进行源码级的调试,运用的比较浅显. 最近研读张银奎老师的<软件调试>获益良多,刚好与之配套的<格蠹汇编>提供了老师大量的调试经验以及实验环境.不拿来好好实践一番简直浪费. 0X01调试笔记之侦查广告插件 俗话说:工欲善其事,必先利其器.对于调试领域来说,更是如此.我们不但要在自己的计算机中安装有WinDbg,还应当使用JIT调试的方法.因为对于Windows系统中的应用程序的崩溃问题,JIT调试是非常有效的办法,而且JIT调试

USB wifi调试笔记

本文以realtek 8192CU WiFi模块为例,介绍USB wifi在Jelly Bean 4.1的调试笔记. 1.WIFI打不开现象概述 WiFi打不开是指您在UI的settings下选中WiFi选项,将其置为on(打开),但最终结果是自动变成off(关闭).正在打开Wi-Fi.正在扫描等一系列不能正常打开WiFi的提示. 2.问题分析流程 WiFi打不开从硬件和软件两方面进行定位,一般是先硬件,然后软件.总体分析流程如图: 3.确认硬件可以识别 8192CU WiFi模块是基于usb接