日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) > 编程资源 > 编程问答 >内容正文

编程问答

数学--数论--康托展开与逆康托展开

發(fā)布時(shí)間:2023/12/15 编程问答 40 豆豆
生活随笔 收集整理的這篇文章主要介紹了 数学--数论--康托展开与逆康托展开 小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

康托展開

可以理解為把一個(gè)全排列映射到一個(gè)數(shù)上面,因?yàn)槿帕腥绻凑諒男〉酱蠡蛘邚拇蟮叫?#xff0c;肯定是有一個(gè)確定的序列的。

一般是從小到大的序列個(gè)數(shù)。我們就是要求出這個(gè)序列的位置。,想法很簡(jiǎn)答,就是求出前面比他小的個(gè)數(shù)就可以了。

理解為一個(gè)每位都是階乘進(jìn)位的數(shù)轉(zhuǎn)化為10進(jìn)制的數(shù)。思路如下:
先準(zhǔn)備求每一位的階乘,然后從高位開始統(tǒng)計(jì)后面有多少個(gè)數(shù)比他小記錄這個(gè)個(gè)位數(shù),然后乘以后面?zhèn)€數(shù)的階乘,再把它累加起來。

x[i]表示第i位后面比他小的個(gè)數(shù),那么

∑1Nx[i]?Fac[N?i]\sum_{1}^{N}x[i]*Fac[N-i] 1N?x[i]?Fac[N?i]
這樣就能求出比他小的有多少個(gè)了,也能求出他是第幾個(gè)序列。

//求出階乘 void init(){Fac[0] = 1;for(int i=1;i<=N;++i){Fac[i] = Fac[i-1]*i;} }int kangtuo(int n,char a[]) {int i,j,t,sum;sum=0;for( i=0; i<n ;++i){t=0;for(j=i+1;j<n;++j)if( a[i]>a[j] )++t;sum+=t*fac[n-i-1];}return sum+1; }

逆康托展開

相當(dāng)于知道序列位置求這個(gè)位置的數(shù)。
想法也很簡(jiǎn)單,因?yàn)閷?duì)于每位的Fac[N-i]都比后面說有的和都大,所以用pos/Fac[N-1]求得的就是x[i],同理pos%Fac[N-i]就是后面的和。

我們維護(hù)一個(gè)序列st始終按照從小到大排列,那么已知某位置的x[i],那么這個(gè)位置的數(shù)就是st[x[i]+1]。

void init(){Fac[0] = 1;for(int i=1;i<=N;++i){Fac[i] = Fac[i-1]*i;} } void reverse_kangtuo(int n,int k,char s[]) {int i, j, t, vst[8]={0};--k;for (i=0; i<n; i++){t = k/fac[n-i-1];for (j=1; j<=n; j++)if (!vst[j]){if (t == 0) break;--t;}s[i] = '0'+j;vst[j] = 1;k %= fac[n-i-1];} }

總結(jié)

以上是生活随笔為你收集整理的数学--数论--康托展开与逆康托展开的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。