康托展開

原博客
  康托展開的公式是 X=an*(n-1)!+an-1*(n-2)!+...+ai*(i-1)!+...+a2*1!+a1*0! 其中份汗,ai為當(dāng)前未出現(xiàn)的元素中是排在第幾個(從0開始)昔逗。
  這個公式可能看著讓人頭大,最好舉個例子來說明一下如贷。
例如,有一個數(shù)組 s = ["A", "B", "C", "D"]诗力,它的一個排列 s1 = ["D", "B", "A", "C"]烙心,現(xiàn)在要把 s1 映射成 X。n 指的是數(shù)組的長度选泻,也就是4冲粤,所以
X(s1) = a4*3! + a3*2! + a2*1! + a1*0!
關(guān)鍵問題是 a4、a3页眯、a2 和 a1 等于啥梯捕?
a4 = "D" 這個元素在子數(shù)組 ["D", "B", "A", "C"] 中是第幾大的元素。"A"是第0大的元素窝撵,"B"是第1大的元素傀顾,"C" 是第2大的元素,"D"是第3大的元素碌奉,所以 a4 = 3短曾。
a3 = "B" 這個元素在子數(shù)組 ["B", "A", "C"] 中是第幾大的元素。"A"是第0大的元素赐劣,"B"是第1大的元素嫉拐,"C" 是第2大的元素,所以 a3 = 1魁兼。
a2 = "A" 這個元素在子數(shù)組 ["A", "C"] 中是第幾大的元素婉徘。"A"是第0大的元素,"C"是第1大的元素,所以 a2 = 0盖呼。
a1 = "C" 這個元素在子數(shù)組 ["C"] 中是第幾大的元素儒鹿。"C" 是第0大的元素,所以 a1 = 0塌计。(因為子數(shù)組只有1個元素挺身,所以a1總是為0)
所以,X(s1) = 3*3! + 1*2! + 0*1! + 0*0! = 20

A B C | 0
A C B | 1
B A C | 2
B C A | 3
C A B | 4
C B A | 5

通過康托逆展開生成全排列

如果已知 s = ["A", "B", "C", "D"]锌仅,X(s1) = 20章钾,能否推出 s1 = ["D", "B", "A", "C"] 呢?
  因為已知 X(s1) = a4*3! + a3*2! + a2*1! + a1*0! = 20热芹,所以問題變成由 20 能否唯一地映射出一組 a4贱傀、a3、a2伊脓、a1府寒?如果不考慮 ai 的取值范圍,有
3*3! + 1*2! + 0*1! + 0*0! = 20
2*3! + 4*2! + 0*1! + 0*0! = 20
1*3! + 7*2! + 0*1! + 0*0! = 20
0*3! + 10*2! + 0*1! + 0*0! = 20
0*3! + 0*2! + 20*1! + 0*0! = 20
等等报腔。但是滿足 0 <= ai <= n-1 的只有第一組株搔。
可以使用輾轉(zhuǎn)相除的方法得到 ai,如下圖所示:



知道了a4纯蛾、a3纤房、a2、a1的值翻诉,就可以知道s1[0] 是子數(shù)組["A", "B", "C", "D"]中第3大的元素 "D"炮姨,s1[1] 是子數(shù)組 ["A", "B", "C"] 中第1大的元素"B",s1[2] 是子數(shù)組 ["A", "C"] 中第0大的元素"A"碰煌,s[3] 是子數(shù)組 ["C"] 中第0大的元素"C"舒岸,所以s1 = ["D", "B", "A", "C"]。

#include<iostream>
#include<algorithm>
#include<vector>
#include<string>
#include<cstdlib>
using namespace std;
class Cantor
{
private:
    string str;//要康托展開的字符串
    char *indexStr;//str排序后的字符串
    int len;//字符串長度
    int *fact;//階乘
    void Init();
    void toCharArray();

public:
    Cantor(string str):str(str),len(str.length())
    {
        Init();
    };
    int encode(string ss);
    string decode(int val);
};
void Cantor::Init()
{
    fact=new int[len];
    fact[0]=fact[1]=1;
    for(int i=2;i<len;i++)
    {
        fact[i]=i*fact[i-1];
    }
    toCharArray();
}
void Cantor::toCharArray()
{
    indexStr=new char[len];
    for(int i=0;i<len;i++)
    {
        indexStr[i]=str[i];
    }
    sort(indexStr,indexStr+len);
}
int Cantor::encode(string ss)//返回字符串在全排列中的字典位置芦圾,從0開始
{
    int cnt,pos=0;
    for(int i=0;i<len;i++)
    {
        cnt=0;
        for(int j=i;j<len;j++)
        {
            if(ss[i]>ss[j]) cnt++;
        }
        pos+=cnt*fact[len-i-1];
    }
    return pos;
}
string Cantor::decode(int val)//通過字符串在全排列中的字典位置找到相應(yīng)的字符串
{
    div_t divResult;//求商和余數(shù)的數(shù)據(jù)結(jié)構(gòu)
    string order,res;
    vector<int> num;
    for(int i=0;i<len;i++)
    {
        num.push_back(i);
    }
    for(int i=len-1;i>=0;i--)
    {
        divResult=div(val,fact[i]);
        order.push_back(num[divResult.quot]);
        num.erase(num.begin()+divResult.quot);
        val=divResult.rem;
    }
    for(int i=0;i<len;i++)
    {
        res.push_back(indexStr[order[i]]);
    }
    return res;
}
int main(){
    Cantor cantor("ABCD");

    cout<<cantor.encode("DBAC")<<endl;
    cout<<cantor.decode(cantor.encode("DBAC"))<<endl;

    cout<<cantor.encode("BCAD")<<endl;
    cout<<cantor.decode(cantor.encode("BCAD"))<<endl;
}

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末蛾派,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子个少,更是在濱河造成了極大的恐慌碍脏,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,378評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件稍算,死亡現(xiàn)場離奇詭異典尾,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)糊探,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,356評論 2 382
  • 文/潘曉璐 我一進(jìn)店門钾埂,熙熙樓的掌柜王于貴愁眉苦臉地迎上來河闰,“玉大人,你說我怎么就攤上這事褥紫〗裕” “怎么了?”我有些...
    開封第一講書人閱讀 152,702評論 0 342
  • 文/不壞的土叔 我叫張陵髓考,是天一觀的道長部念。 經(jīng)常有香客問我,道長氨菇,這世上最難降的妖魔是什么儡炼? 我笑而不...
    開封第一講書人閱讀 55,259評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮查蓉,結(jié)果婚禮上乌询,老公的妹妹穿的比我還像新娘。我一直安慰自己豌研,他們只是感情好妹田,可當(dāng)我...
    茶點故事閱讀 64,263評論 5 371
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著鹃共,像睡著了一般鬼佣。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上霜浴,一...
    開封第一講書人閱讀 49,036評論 1 285
  • 那天晶衷,我揣著相機(jī)與錄音,去河邊找鬼坷随。 笑死房铭,一個胖子當(dāng)著我的面吹牛驻龟,可吹牛的內(nèi)容都是我干的温眉。 我是一名探鬼主播,決...
    沈念sama閱讀 38,349評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼翁狐,長吁一口氣:“原來是場噩夢啊……” “哼类溢!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起露懒,我...
    開封第一講書人閱讀 36,979評論 0 259
  • 序言:老撾萬榮一對情侶失蹤闯冷,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后懈词,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體蛇耀,經(jīng)...
    沈念sama閱讀 43,469評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,938評論 2 323
  • 正文 我和宋清朗相戀三年坎弯,在試婚紗的時候發(fā)現(xiàn)自己被綠了纺涤。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片译暂。...
    茶點故事閱讀 38,059評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖撩炊,靈堂內(nèi)的尸體忽然破棺而出外永,到底是詐尸還是另有隱情,我是刑警寧澤拧咳,帶...
    沈念sama閱讀 33,703評論 4 323
  • 正文 年R本政府宣布伯顶,位于F島的核電站,受9級特大地震影響骆膝,放射性物質(zhì)發(fā)生泄漏祭衩。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,257評論 3 307
  • 文/蒙蒙 一谭网、第九天 我趴在偏房一處隱蔽的房頂上張望汪厨。 院中可真熱鬧,春花似錦愉择、人聲如沸劫乱。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,262評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽衷戈。三九已至,卻和暖如春层坠,著一層夾襖步出監(jiān)牢的瞬間殖妇,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,485評論 1 262
  • 我被黑心中介騙來泰國打工破花, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留谦趣,地道東北人。 一個月前我還...
    沈念sama閱讀 45,501評論 2 354
  • 正文 我出身青樓座每,卻偏偏與公主長得像前鹅,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子峭梳,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,792評論 2 345

推薦閱讀更多精彩內(nèi)容

  • 1. 關(guān)于診斷X線機(jī)準(zhǔn)直器的作用舰绘,錯誤的是()。 (6.0 分) A. 顯示照射野 B. 顯示中心線 C. 屏蔽多...
    我們村我最帥閱讀 10,270評論 0 5
  • 【1】7葱椭,9捂寿,-1,5孵运,( ) A秦陋、4;B治笨、2驳概;C粪小、-1;D抡句、-3 分析:選D探膊,7+9=16;9+(-1)=8待榔;(...
    Alex_bingo閱讀 18,817評論 1 19
  • 這兩天太煩燥了逞壁,對我好的老想假裝不喜歡,對我不好的一直生著人家的悶氣锐锣。其實我的性格缺陷才比較大吧腌闯。 這樣心情一直不...
    象三旬閱讀 330評論 0 0
  • 同事到底是怎樣一種人際關(guān)系,以前從來沒考慮過雕憔。 剛從大學(xué)出來就到了部隊研究所工作姿骏,周圍的人也勉強(qiáng)算...
    尼航閱讀 274評論 0 1
  • 或許,我是在寫 一片被晚風(fēng)沉醉的花 總試圖用一些詞語 講述屬于這個冬天的故事 望穿秋水斤彼。也未能尋到稱心的話 當(dāng)你還...
    雨心兒forever閱讀 379評論 0 0