康托展开 / 逆康托展开

  先搬一下(戳)维基百科的康托展开(戳):

  康托展开是一个全排列到一个自然数的双射,常用于构建哈希表时的空间压缩。 康托展开的实质是计算当前排列在所有由小到大全排列中的顺序,因此是可逆的。

  由于是双射    所以可以求n的全排列里第k大的排列(逆康托展开)

  (伪)计算原理: 从某个元素找后面比这个元素小的数的个数,再乘以这个位置每一个数字能有的组合方法数(排列 / 阶乘),得出只考虑从这一位开始到末尾比当前小的排列数,然后加起来就是康托展开求的数(追求难懂的巅峰...........看不懂就看看维基.........

    公式:    

        ai 是整数 且 0 <= a< i ,  1 <= i <= n 

        ai 的意义:参悟栗子吧

    栗子:

    例如,3 5 7 4 1 2 9 6 8 展开为 98884。
        因为X=2*8!+3*7!+4*6!+2*5!+0*4!+0*3!+2*2!+0*1!+0*0!=98884.
    解释:
        排列的第一位是3,比3小的数有两个,以这样的数开始的排列有8!个,因此第一项为2*8!
        排列的第二位是5,比5小的数有1、2、3、4,由于3已经出现,因此共有3个比5小的数,这样的排列有7!个,因此第二项为3*7!
        以此类推,直至0*0!

  再贴一下(戳)NOCOW关于康托展开的一页(戳),有代码,看得懂就差不多了

  逆康托展开并不是康托展开完全逆过来:

    1.减去1,得到比该排列小的排列的数量

    2.从高位算起:取摸对应位的阶乘,得到在这一位后面比这一位小的数的个数(所以要注意前面比这一位的数小的要去除)

    

    栗子:

    如n=5,x=96时:
        首先用96-1得到95,说明x之前有95个排列.(将此数本身减去!)
        用95去除4! 得到3余23,说明有3个数比第1位小,所以第一位是4.
        用23去除3! 得到3余5,说明有3个数比第2位小,所以是4,但是4已出现    过,因此是5.
        用5去除2!得到2余1,类似地,这一位是3.
        用1去除1!得到1余0,这一位是2.
        最后一位只能是1.
        所以这个数是45321.

  

  再贴上忘了什么时候写的代码:

 

 1 typedef long long ll;
 2 const ll factorial[11] = {1, 1, 2, 6, 24, 120, 720, 5040, 40320, 362880, 3628800};
 3 int num[101];
 4 ll Cantorex(int n){
 5     for(int i = 0; i < n; ++i) cin >> num[i];
 6     ll sum = 0;
 7     for(int i = 0; i < n; ++i){
 8         int t = 0;
 9         for(int j = i + 1; j < n; ++j) if(num[j] < num[i]) ++t;
10         sum += t * factorial[n-i-1];
11     }
12     return sum;
13 }
14 ll uCantorex(ll sum, int n){
15     memset(num, 0, sizeof(num));
16     sum -= 1;
17     int x = 0;
18     ll ans = 0;
19     for(int i = n - 1; i > 0; --i){
20         ll k = sum / factorial[i] + 1;
21         sum %= factorial[i];
22         while(num[k]) ++k;
23         num[k] = 1;
24         ans = ans * 10 + k;
25     }
26     for(int i = 1; i <= n; ++i)
27         if(!num[i]) ans = ans * 10 + i;
28     return ans;
29 }

  Done!

时间: 2024-10-10 04:32:03

康托展开 / 逆康托展开的相关文章

康托和逆康托展开(转)

1.康托展开的解释 康托展开就是一种特殊的哈希函数 把一个整数X展开成如下形式: X=a[n]*n!+a[n-1]*(n-1)!+...+a[2]*2!+a[1]*1! 其中,a为整数,并且0<=a<i,i=1,2,..,n {1,2,3,4,...,n}表示1,2,3,...,n的排列如 {1,2,3} 按从小到大排列一共6个.123 132 213 231 312 321 . 代表的数字 1 2 3 4 5 6 也就是把10进制数与一个排列对应起来. 他们间的对应关系可由康托展开来找到.

康托展开和逆康托展开

问题:给定的全排列,计算出它是第几个排列? 对于全排列,不清楚的可以参考全排列 方法:康托展开 对于一个长度为 n 的排列 num[1..n], 其序列号 X 为 X = a[1]*(n-1)! + a[2]*(n-2)! +...+ a[i]*(n-i)! +...+ a[n-1]*1! + a[n]*0! 其中a[i]表示在num[i+1..n]中比num[i]小的数的数量 写做伪代码为: Cantor(num[]) X = 0 For i = 1 .. n tp = 0 For j = i

nyist 139 我排第几个&amp;&amp;143 第几是谁(康托展开和逆康托展开)

 我排第几个 时间限制:1000 ms  |  内存限制:65535 KB 难度:3 描述 现在有"abcdefghijkl"12个字符,将其所有的排列中按字典序排列,给出任意一种排列,说出这个排列在所有的排列中是第几小的? 输入 第一行有一个整数n(0<n<=10000); 随后有n行,每行是一个排列: 输出 输出一个整数m,占一行,m表示排列是第几位: 样例输入 3 abcdefghijkl hgebkflacdji gfkedhjblcia 样例输出 1 3027

LightOJ1060 nth Permutation(不重复全排列+逆康托展开)

一年多前遇到差不多的题目http://acm.fafu.edu.cn/problem.php?id=1427. 一开始我还用搜索..后来那时意外找到一个不重复全排列的计算公式:M!/(N1!*N2!*...*Nn!), 然后就靠自己YY出解法,搞了好几天,最后向学长要了数据,然后迷迷糊糊调了,终于AC了. 后来才知道当时想的解法类似于逆康托展开,只是逆康托展开是对于没有重复元素全排列而言,不过有没有重复元素都一个样. 而现在做这题很顺,因为思路很清晰了,另外这做法和数论DP的统计部分有相似之处.

nyoj 139——我排第几个|| nyoj 143——第几是谁? 康托展开与逆康托展开

讲解康托展开与逆康托展开.http://wenku.baidu.com/view/55ebccee4afe04a1b071deaf.html #include<bits/stdc++.h> using namespace std; int fac[20]; int fun(){ fac[0]=1; int i; for(i=1;i<=12;i++){ fac[i]=fac[i-1]*i; } } int main(){ int t,i,j,c,sum,num; char str[15];

数据结构——康托展开与逆康托展开

含义 康托展开是一个全排列到一个自然数的双射,常用于构建哈希表时的空间压缩. 康托展开的实质是计算当前排列在所有由小到大全排列中的顺序,因此是可逆的. 原理 X = s1(n-1)! + s2(n-2)! + s3(n-3)! + …… + sn-1 * 1! + sn * 0! 其中si表示在第i位右边比ai小的数的个数. 我们现在用sl表示第i位左边比ai小的数的个数,sr表示第i位右边比ai小的数的个数,显然可以得到如下等式: ai = sl + sr + 1 故公式中的si可以用上述等式

康拓展开与逆康拓展开

1.康托展开的解释 康托展开就是一种特殊的哈希函数 把一个整数X展开成如下形式: X=a[n]*n!+a[n-1]*(n-1)!+...+a[2]*2!+a[1]*1! 其中,a为整数,并且0<=a<i,i=1,2,..,n {1,2,3,4,...,n}表示1,2,3,...,n的排列如 {1,2,3} 按从小到大排列一共6个.123 132 213 231 312 321 . 代表的数字 1 2 3 4 5 6 也就是把10进制数与一个排列对应起来. 他们间的对应关系可由康托展开来找到.

hdu 1027 Ignatius and the Princess II(正、逆康托)

题意: 给N和M. 输出1,2,...,N的第M大全排列. 思路: 将M逆康托,求出a1,a2,...aN. 看代码. 代码: int const MAXM=10000; int fac[15]; int ans[1005]; int kk; int n,m; vector<int> pq; int main(){ int cn=0; fac[0]=1; while(1){ ++cn; fac[cn]=fac[cn-1]*cn; if(fac[cn]>MAXM){ --cn; break

康拓展开和逆康拓展开

康拓展开和逆康拓展开 康拓展开模板题 复杂度O(\(n^2\))的会tle(看数据就知道了)(虽然某题解说可以,不知道是不是后期加强了数据 然而我还是写了O(\(n^2\))的 #include <cstdio> typedef long long LL; LL f[1000010]; const LL mod = 998244353; int a[1000010], b[1000010]; int main() { f[0] = 1; for(int i = 1; i < 100000