倒位序算法(C#实现)
倒位序算法(C#实现)
原序数(十进制) | 原序数(二进制) | 倒位序(二进制) | 倒位序(十进制) |
---|---|---|---|
0 | 000 | 000 | 0 |
1 | 001 | 100 | 4 |
2 | 010 | 010 | 3 |
3 | 011 | 110 | 6 |
4 | 100 | 001 | 1 |
5 | 101 | 101 | 5 |
6 | 110 | 011 | 3 |
7 | 111 | 111 | 7 |
在编写FFT算法的时候,会用到倒位序,倒位序的计算方法:将原序数表示成二进制,然后将二进制首尾反转,得到新的序数序列,即需要的倒位序。
以下称正常排序为正序,倒位序为倒序,十进制数字后面加d标识,二进制数字加b标识。
首先我们剖析一下倒位序的生成规则:
以长度为\(2^3\)的序列为例:
正序是相对于上一个序数加1d,对应于二进制加上001b,那么对应于倒序就是二进制加上100b,而且是向右边进位。以下是长度为8的倒位序生成过程:
二进制 | 十进制 |
---|---|
\(000b\) | \(0d\) |
\(000b + 100b = 100b\) | \(4d\) |
\(100b + 100b = 010b\) | \(2d\) |
\(010b + 100b = 110b\) | \(6d\) |
\(\cdots\) |
可以总结出长度为\(2^3\)的倒位序序列,从第\(i\)个倒位序数字\(d_i(十进制)\)生成第\(i+1\)个倒位序数字的规则如下:
- 情况1:第\(i\)个序数倒位加1没有进位,形如\(0**b\)这样的(二进制0打头),相当于十进制加了\(100b=4d\),故第\(i+1\)个倒位序数字为\(d_{i+1}=d_i + 4\);
- 情况2:第\(i\)个序数倒位加1有进位,形如\(1**b\)这样的(二进制1打头),可以通过减去\(100b=4d\)来实现最高位变成0,但是有进位,所以又要加上\(010b=2d\),这时要判断二进制第二位是否有进位,判断方法是比较原数字减去了\(4d\)后是否大于等于\(2d\),如果大于等于\(2d\),则它必然形如\(01*b\),加上\(010b\)后有进位,所以先减去\(010b\),然后再判断加上\(001d\)会不会进位;如果小于小于\(2d\),则它一定形如\(00*b\),所以直接加上010b=2d,输出结果。
下面是利用C#输出长度为\(2^n\)的倒位序序列的函数:
//输出长度为2^n的倒位序序列
public int[] reverse(int n)
{
int len = Convert.ToInt32(Math.Pow(2, n));
int k ;
int[] rank = new int[len];
rank[0] = 0;
for (int i = 1; i < len; i++)
{
k = len / 2;
if (rank[i - 1] < k)
{
rank[i] = rank[i - 1] + k;
}
else
{
rank[i] = rank[i - 1];
while (rank[i] >= k)
{
rank[i] = rank[i] - k;
k = k / 2;
}
rank[i] = rank[i] + k ;
}
}
return rank;
}