离散哈特莱变换(DHT)及快速哈特莱变换(FHT)学习

离散哈特莱变换(DHT)及快速哈特莱变换(FHT)学习


说在前边

最近复习\(DSP\)的时候,发现了一个号称专门针对离散实序列的变换,经分析总运算量为普通\(FFT\)的几乎一半,而且完全没有复数。这么强的吗?于是花了一个下午,去学习了一下。。。于是去图书馆翻了几乎所有的\(dsp\)课本。。。发现了这本书 西安电子科技大学出版社《数字信号处理》第二版!竟然花了一节在讲\(DHT\)\(FHT\)!竟然还有附录FORTRAN代码(虽然一堆错)!这里把这个东西稍微普及一下。。。不过估计没人用的上


离散哈特莱变换(DHT)

引入

对于序列\(x(n)\)\(N\)\(DFT\)具有简单的共轭对称性,即\[X(N - k) = X^*(k), k = 0, 1 ... ,N-1 \]
所以只要计算\(X(k)\)的前\(N/2\)个值,则后\(N/2\)可以通过上式求得,\(X(k)\)\(N/2\)个复数正好对应\(N\)个实数数据(\(N\)点实序列\(DFT\),也可以通过把\(N\)个实数,压成\(N/2\)个复数,再利用共轭对称性求解,这里不做详解)。由此可见,一个\(N\)点实序列的\(DFT\),完全可由\(N\)个实数数据确定。由此,我们引出一种直接对于实序列进行实数域变换的离散哈特莱变换(\(DHT\))。\(DHT\)\(x(n)\)看成实数数据,而不是像\(DFT\)一样看作虚部为\(0\)的复数,因此节省了一半的空间,运算效率也提升了近一倍,且\(DFT\)\(DHT\)之间存在简单的关系,容易实现相互转换。

预备知识

  1. \(DFT\)的意义
  2. \(FFT\)实现
  3. 最好学过\(DSP\)

定义

\(x(n), n = 0, 1,..., N-1\),为一实序列,其\(DHT\)定义为
\[ X_H(k) = DHT[ x(n)] = \sum _{n=0}^{N-1} x(n)cas(\frac{2\pi}{N}kn), k = 0,1, ... , N-1 \]
式中\(cas(a) = cos(a) + sin(a)\)逆变换(\(IDHT\))为
\[ x(n) = DHT[ X_H(k)] = \frac{1}{N}\sum _{n=0}^{N-1} X(k)cas(\frac{2\pi}{N}kn), n = 0,1, ... , N-1 \]
\(DHT\)的正交证明:

\[ \sum_{k=0}^{N-1} cas(\frac{2\pi}{N}kn)cas(\frac{2\pi}{N}km) = \sum_{k=0}^{N-1}[cos(\frac{2\pi}{N}k(n-m)) + sin(\frac{2\pi}{N}k(n+m))] = \left\{\begin{array}{cc} N, & k = 0\\ 0, & k \neq 0 \\ \end{array}\right. \]

DHT与DFT的关系

\(X(k)\)表示实序列\(x(n)\)\(DFT\),用\(X_H(k)\)表示\(x(n)\)\(DHT\),分别用\(X_{He}(k)\)\(X_{Ho}(k)\)表示\(X_H(k)\)的偶对称分量与奇对称分量,即
\[ X_H(k) = X_{He}(k) + X_{Ho}(k) \]

其中

\[ X_{He}(k) = \frac{1}{2}[X_H(k) + X_H(N-k)]\\ X_{Ho}(k) = \frac{1}{2}[X_H(k) - X_H(N-k)]\\ X(k) = X_{He}(k) - jX_{Ho}(k)\\ X_H(k) = Re[X(k)] - Im[X(k)]\\ X(k) = \frac{1}{2}[X_H(k)+X_H(N-k)] - \frac{1}{2}j[X_H(k)-X_H(N-k)] \]

利用上面这些性质,我们可以很容易的将\(DFT\)\(DHT\)进行变换,并且又因为这个原因\(X_H(k)\)的周期为\(N\)(隐含周期性)。

DHT的优点

  1. \(DHT\)为实值,避免了复数运算
  2. \(DHT\)正反变换形式基本一致
  3. \(DHT\)\(DFT\)的转换容易实现

DHT的性质

  1. 线性性
  2. \(DHT\)不改变\(x(n)\)奇偶性
  3. 循环卷积定理

\(x_1(n) \leftrightarrow X_{1H}(k), ~~x_2(n) \leftrightarrow X_{2H}(k)\)
\[ x_1(n) \otimes x_2(n) \leftrightarrow X_{2H}(k)X_{1He}(k) + X_{2H}(N-k)X_{1Ho}(k) \]

\[ x_1(n) \otimes x_1(n) \leftrightarrow X_{1H}(k)X_{2He}(k) + X_{1H}(N-k)X_{2Ho}(k) \]

值得注意的是,相比于\(DFT\)的卷积定理,\(DHT\)的运算更加复杂,这一点把这个算法的在竞赛中的实用性大大降低了。。。毕竟我们就是为了加速卷积,不过他空间上的优势还是很名明显的


快速哈特莱变换(FHT)

基2 DIT-FHT算法

\(FFT\)算法一样,\(FHT\)也可以,用基\(4\),基\(8\),分裂基等方式优化,这里我们讨论最基础的基\(2\) 快速\(DHT\)算法。
\(x(n)\)\(N=2^M\)\(DHT\)
\[ X_H(k) = \sum_{n=0}^{N-1}x(n)cas(\frac{2\pi}{N}kn) \]
\(x(n)\)进行奇偶抽取
\[ x_0(r) = x(2r)\\ x_1(r) = x(2r+1) \]
带入后,与\(DFT\)比较可得,\(cas(\frac{2\pi}{N}k(2r+1))\)不是指数函数,所以我们通过
\[ cas(a + b) = cas(a)cas(b) + cas(-a)sin(b) \]
推导可得:
\[ X_H(k) = \sum_{r=0}^{\frac{N}{2}-1} x_0(r)cas(\frac{2\pi}{N/2}rk) + cos(\frac{2\pi}{N}k)\sum_{r=0}^{\frac{N}{2}-1}x_1(r)cas(\frac{N}{2}rk) \\ + sin(\frac{2\pi}{N}k)\sum_{r=0}^{\frac{N}{2}-1}x_1(r)cas(-\frac{2\pi}{N/2}rk) \]
\(X_{0H}(k) = DHT[x_0(n)]\)\(X_{1H}(k) = DHT[x_1(n)]\),可以写成:
\[ X_H(k) = X_{0H}(k) + cos(\frac{2\pi}{N}k)X_{1H}(k)+sin(\frac{2\pi}{N}k)X_{1H}(\frac{N}{2}-1) \]

相比于\(DIT-DFT\)该式中,多了一项\(X_{1H}(\frac{N}{2}-k)\)。所以可用\(X_{0H}(k)\),\(X_{1H}(k)\),\(X_{0H}(\frac{N}{2}-k)\),\(X_{0H}(\frac{N}{2}+k)\)四个点同址计算出\(X_{H}(k)\),\(X_{H}(\frac{N}{2}+k)\),\(X_{H}(\frac{N}{2}-k)\),\(X_{H}(N-k)\),与\(FFT\)中的蝶形运算类似,我们叫这种运算“哈特莱蝶形”

\(C(k)=cos(\frac{2\pi}{N}k)\),\(S(k)=sin(\frac{2\pi}{N}k)\),当\(N \geq 8\)时,有
\[ X_H(k) = X_{0H}(k) + [C(k)X_{1H}(k)+S(k)X_{1H}(\frac{N}{2}-k)]\\ X_H(\frac{N}{2}+k) = X_{0H}(k) - [C(k)X_{1H}(k)+S(k)X_{1H}(\frac{N}{2}-k)]\\ X_H(\frac{N}{2}-k) = X_{0H}(\frac{N}{2}-k) - [S(k)X_{1H}(k)-C(k)X_{1H}(\frac{N}{2}-k)]\\ X_H(N-k) = X_{0H}(\frac{N}{2}-k) - [S(k)X_{1H}(k)-C(k)X_{1H}(\frac{N}{2}-k)] \]

信号流图

DHT

基2 DIT-FHT的运算量(只考虑蝶形变换)

\(N = 2^M\)
乘法次数:\(NM - 3N + 4\)
加法次数:$ \frac{3}{2}NM - \frac{3}{2}N + 2$

应用

利用\(FHT\)以及循环卷积定理,与DFT类似,我们就可以通过循换卷积来求出两个实序列的线性卷积。

代码 [UR#34]多项式乘法

#include <cstdio>
#include <cmath>
#include <algorithm>
#define DXT(X,Y) ( XT = (X) , X = ((XT) + (Y)) , Y = ((XT) - (Y)) )
using namespace std;int N1 , N2 , N4 , L1 , L2 , L3 , L4, n , m , rev[2000001];
double XT , A , E , CC1 , SS1 , T1 , T2 , a[2000001] , b[2000001] , c[2000001];void FHT( double X[] , int N , int M ) {for(int i = 0 ; i < N ; ++i) if(i < rev[i]) swap(X[i] , X[rev[i]]);for(int i = 0 ; i < N ; i += 2) DXT(X[i] , X[i+1]);N2 = 1; --M;if(M <= 0) return;while( M-- ) {N4 = N2 , N2 = N4 + N4 , N1 = N2 + N2;E = 6.283185307179586 / N1;for(int j = 0 ; j < N ; j += N1) {L2 = j + N2 , L3 = j + N4 , L4 = L2 + N4;DXT(X[j] , X[L2]) , DXT(X[L3] , X[L4]) , A = E;for(int i = 1 ; i < N4 ; ++i , A += E) {L1 = j + i , L2 = j - i + N2;L3 = L1 + N2 , L4 = L2 + N2;CC1 = cos(A) , SS1 = sin(A);T1 = X[L3] * CC1 + X[L4] * SS1;T2 = X[L3] * SS1 - X[L4] * CC1;XT = X[L1] , X[L1] = XT + T1 , X[L3] = XT - T1;XT = X[L2] , X[L2] = XT + T2 , X[L4] = XT - T2;}}}
}int main() {scanf("%d%d" , &n , &m);for(int x , i = 0 ; i <= n ; ++i) scanf("%d" , &x) , a[i] = (double)x;for(int x , i = 0 ; i <= m ; ++i) scanf("%d" , &x) , b[i] = (double)x;int nn = n + m + 1 , L = 0 , tmp;for(tmp = 1 ; tmp < nn ; tmp <<= 1, ++L) ; nn = tmp;for(int i = 0 ; i < nn ; ++i)rev[i] = (rev[i >> 1] >> 1 ) | ((i & 1) << (L - 1));FHT(a , nn , L);   FHT(b , nn , L);c[0] = b[0] * a[0];for(int i = 1; i < nn; ++i) {T1 = a[i] , T2 = a[nn - i];c[i] = (b[i] * (T1 + T2) + b[nn - i] * (T1 - T2)) * 0.5;}FHT(c  , nn , L);for(int i = 0 ; i <= n + m ; ++i) c[i] /= nn;for(int i = 0 ; i <= n + m ; ++i) printf("%d " , (int)(c[i] + 0.5));return 0;
}

这份代码比我想像的要慢好多,主要原因是循环卷积定理里面运算的增多,以及蝶形变化中较为麻烦的计算四个位置,还有每次使用\(cosA\)\(sinA\)都是很大的开销,曾经尝试使用乘法代替每次计算\(cos\)\(sin\)但是点数\(N\)很大时,精度问题会被放大的很严重,另一个原因是我不会卡常...但是内存上看起来还是比较优秀的,拜托善于优化的同学优化一下,或者有更好的实现的话,也请告诉我。。。不过看起来还是没人会用啊。。。法法塔的历史地位还是很难被替代的吧。。。改日会补上信号流图。~掰掰

转载于:https://www.cnblogs.com/RRRR-wys/p/10090007.html

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

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

相关文章

P2495-[SDOI2011]消耗战【虚树,dp】

正题 题目链接:https://www.luogu.com.cn/problem/P2495 题目大意 nnn个点的一棵树&#xff0c;mmm次给出一些点&#xff0c;要求割掉最小权值的边使得这些点不和111号点联通。 解题思路 根据这些给出的点构造一棵虚树&#xff0c;然后直接dpdpdp求解即可。 codecodecode #i…

【贪心】Sunscreen(poj 3614/luogu 2887)

Sunscreen poj 3614 luogu 2887 题目大意&#xff1a; 有n个人&#xff0c;每个人要求选一个价值在minniminn_iminni​到maxximaxx_imaxxi​的物品&#xff0c;现在有m件物品&#xff0c;每件的价值是spfispf_ispfi​&#xff0c;可以选covericover_icoveri​次&#xff0c…

.NET Core 2.1中的分层编译(预览)

如果您是.NET性能的粉丝&#xff0c;最近有很多好消息&#xff0c;例如.NET Core 2.1中的性能改进和宣布.NET Core 2.1&#xff0c;但我们还有更多的好消息。分层编译是一项重要的新特性功能&#xff0c;我们可以作为预览供任何人试用&#xff0c;从.NET Core 2.1开始。在我们测…

Codeforces 刷题记录(已停更)

Codeforces 每日刷题记录 (已停更) 打‘’是一些有启发意义的题目&#xff0c;部分附上一句话题解&#xff0c;每日更新3题&#xff0c;大部分题目较水。 DayIDProblemTutorialNote11CF1073E状压&#xff0c;数位dp&#xff0c;官方题解std骚操作\(~\)2CF1072A\(~\)3CF1072B24C…

牛客练习赛60 ~ 斩杀线计算大师

题目传送 时间限制&#xff1a;C/C 1秒&#xff0c;其他语言2秒 空间限制&#xff1a;C/C 262144K&#xff0c;其他语言524288K Special Judge, 64bit IO Format: %lld 题目描述 算术能力是每个炉石玩家必不可少的&#xff0c;假设现在有三种伤害卡&#xff0c;伤害值分别是a,b…

【贪心】Stall Reservations(luogu 2859/poj 3190)

Stall Reservations luogu 2859 poj 3190 题目大意&#xff1a; 有n头牛&#xff0c;每头牛都有自己的挤奶时间&#xff0c;挤奶时间内每头牛用一个奶棚&#xff0c;现在问最少需要多少个奶棚 输入样例 5 1 10 2 4 3 6 5 8 4 7输出样例 4 1 2 3 2 4数据范围 1⩽N⩽50,00…

选择IT事业,意味着终身学习

八月&#xff0c;炎阳如火。 前几天书记找我交流&#xff0c;问我离职的原因&#xff0c;我跟他仔细的分析了一下我的职业发展规划和我对于未来的预期&#xff0c;书记也向我分析了一下他所认为的原因&#xff0c;他说&#xff0c;无外乎是三个原因&#xff1a;第一个是钱的问…

P4103-[HEOI2014]大工程【虚树,dp】

正题 题目链接:https://www.luogu.com.cn/problem/P4103 题目大意 nnn个点的一棵树&#xff0c;mmm次给出一些点&#xff0c;求最近点对&#xff0c;最远点对和所有点对的距离和。 解题思路 先构造一棵虚树&#xff0c;然后在上面dpdpdp统计答案即可。 codecodecode #includ…

牛客网【每日一题】Shortest Path 4月3日题目精讲 DFS

题号 NC13886 Shortest Path 西南交通大学第十三届ACM决赛 题意&#xff1a; 一棵偶数节点的树&#xff0c;分成n/2对&#xff0c;两两一组&#xff0c;所有组的路径之和最小是多少&#xff1f; 题解&#xff1a; 如果两个点之间相连将另外两个相连的点覆盖&#xff0c;那么完全…

是男人就过 8 题--Pony.AI 题 - A String Game

是男人就过 8 题--Pony.AI 题 - A String Game 题目来源 题意&#xff1a;给一个串t以及n个t的子串s&#xff0c;两个人每轮可以选择一个s在他的后边添加一个字符满足得到的新串仍是t的子串&#xff0c;第一个不能操作的人输。 做法&#xff1a;对s串建SAM&#xff0c;在一个子…

使用Jexus服务器运行Asp.Net Core2.0程序

前段时间写了篇关于.net core跨平台部署的文章。https://my.oschina.net/lichaoqiang/blog/1861977主要讲述了&#xff0c;利用NginxCentOSSupervisor.NetCore2.1&#xff0c;来运行.net core程序&#xff0c;感兴趣的朋友可以看一下。今天向大家介绍.net core使用jexus服务器的…

【结论】棋盘(jzoj 2297)

棋盘 jzoj 2297 题目大意&#xff1a; 在棋盘上有一个特殊的象&#xff0c;他可以向四个方向行走若干步&#xff08;左上&#xff0c;左下&#xff0c;右上&#xff0c;右下&#xff09;&#xff0c;现在问从某一个点是否能到另外一个点 输入样例 5 1 1 2 2 2 3 2 2 1 2 4…

P3233-[HNOI2014]世界树【虚树,倍增】

正题 题目链接:https://www.luogu.com.cn/problem/P3233 题目大意 nnn个点的一棵树&#xff0c;mmm次选出一些点作为关键点。每个树上的点会对最近的关键点做贡献&#xff0c;求每个关键点的贡献。 解题思路 显然是虚树&#xff0c;考虑如何在虚树上求贡献&#xff0c;我们发…

牛客每日一题3.31 城市网络 树上倍增

牛客网 时间限制&#xff1a;C/C 2秒&#xff0c;其他语言4秒 空间限制&#xff1a;C/C 262144K&#xff0c;其他语言524288K 64bit IO Format: %lld 题目描述 有一个树状的城市网络&#xff08;即 n 个城市由 n-1 条道路连接的连通图&#xff09;&#xff0c;首都为 1 号城市&…

RRRR_wys' Blog 3.0 准备上线啦!

RRRR_wys Blog 3.0 准备上线啦&#xff01; 今年马上要过完啦&#xff0c;打算在年前把博客翻翻新之前的布局太复杂了&#xff0c;感觉很视觉疲劳&#xff0c;这一版我打算能删就删完善了\(markdown\)还有一些地方要修&#xff0c;放假再说辣在vj上交了道cf&#xff0c;有惊喜 …

【ST表】栈(jzoj 2295)

栈 jzoj 2295 题目大意&#xff1a; 有一个A数组&#xff0c;一个B数组和一个栈&#xff0c;现在把A数组的数存入栈&#xff08;操作1&#xff09;&#xff0c;然后再从栈中取出来放在B数组&#xff0c;但取出可以从栈顶&#xff08;操作2&#xff09;&#xff0c;也可以从栈…

CSPNOIP2020总结

这里是目录友链CSPpartCSP\ \ partCSP partCSP−S1CSP-S1CSP−S1CSP−S2CSP-S2CSP−S2游记PartPartPart后面PartPartPart总结PartPartPartNOIPpartNOIP\ \ partNOIP partDay0Day\ \ 0Day 0Day1Day\ 1Day 1友链 stoorz的CSPNOIP游记是这个逼强行拉我加友链\tiny\color{white…

WebApiClient的JsonPatch局部更新

1. 文章目的随着WebApiClient的不断完善&#xff0c;越来越多开发者选择WebApiClient替换原生的HttpClient&#xff0c;本文将介绍使用WebApiClient来完成JsonPatch提交的新特性。2. json patch介绍在服务端WebApi开发的时候&#xff0c;如果设计一个更新登录用户的个人信息的接…

SPOJ1812 LCS2

SPOJ1812 LCS2 题意&#xff1a;给n个串&#xff0c;求最长公共子串 做法&#xff1a;对第一个串建\(SAM\)&#xff0c;拿剩余的串类似于求\(LCS\)的在上面跑&#xff0c;对于当前这个串&#xff0c;求出可以到达每个状态的最长子串长度&#xff0c;然后&#xff0c;每个状态对…

【每日一题】4月1日题目 Rinne Loves Edges

牛客网 链接&#xff1a;https://ac.nowcoder.com/acm/problem/22598 来源&#xff1a;牛客网 题目&#xff1a; 时间限制&#xff1a;C/C 1秒&#xff0c;其他语言2秒 空间限制&#xff1a;C/C 131072K&#xff0c;其他语言262144K 64bit IO Format:%lld 题目描述 Rinne 最近…