【轉】康拓展開

———本文轉自:http://www.cnblogs.com/1-2-3/archive/2011/04/25/generate-permutation-part2.html

1、康托展開
  康托展開的公式是 X=an*(n-1)!+an-1*(n-2)!+...+ai*(i-1)!+...+a2*1!+a1*0! 其中,ai為當前未出現的元素中是排在第幾個(從0開始)。
  這個公式可能看著讓人頭大,最好舉個例子來說明一下。例如,有一個數組 s = ["A", "B", "C", "D"],它的一個排列 s1 = ["D", "B", "A", "C"],現在要把 s1 映射成 X。n 指的是數組的長度,也就是4,所以
X(s1) = a4*3! + a3*2! + a2*1! + a1*0!
關鍵問題是 a4、a3、a2 和 a1?等于啥?
a4 = "D" 這個元素在子數組 ["D", "B", "A", "C"] 中是第幾大的元素。"A"是第0大的元素,"B"是第1大的元素,"C" 是第2大的元素,"D"是第3大的元素,所以 a4 = 3。
a3 = "B" 這個元素在子數組 ["B", "A", "C"] 中是第幾大的元素。"A"是第0大的元素,"B"是第1大的元素,"C" 是第2大的元素,所以 a3 = 1。
a2 = "A" 這個元素在子數組 ["A", "C"] 中是第幾大的元素。"A"是第0大的元素,"C"是第1大的元素,所以 a2 = 0。
a1 = "C" 這個元素在子數組 ["C"] 中是第幾大的元素。"C" 是第0大的元素,所以 a1 = 0。(因為子數組只有1個元素,所以a1總是為0)
所以,X(s1) = 3*3! + 1*2! + 0*1! + 0*0! = 20。

2、通過康托逆展開生成全排列
  如果已知 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 的只有第一組。可以使用輾轉相除的方法得到 ai,如下圖所示:

知道了a4、a3、a2、a1的值,就可以知道s1[0] 是子數組["A", "B", "C", "D"]中第3大的元素 "D",s1[1] 是子數組 ["A", "B", "C"] 中第1大的元素"B",s1[2] 是子數組 ["A", "C"]?中第0大的元素"A",s[3] 是子數組 ["C"] 中第0大的元素"C",所以s1 = ["D", "B", "A", "C"]。
這樣我們就能寫出一個函數,它可以返回? s 的第 m 個排列。

?

以下是代碼(以不超過10位的字符串為例),有所修改:

 1 #include<iostream>
 2 #include<cstdio>
 3 #include<cstring>
 4 #include<string>
 5 using namespace std;
 6 const int maxn = 10;
 7 char s[maxn+1];
 8 //計算x的階乘
 9 int Cal(int x)
10 {
11     int ans = 1;
12     for (int i = 1; i <= x; i++) ans *= i;
13     return ans;
14 }
15 //得到字符在字符串其后字符中為第幾大
16 int ID(char* c, char*s,int len)
17 {
18     int re = 0;
19     for (char*t = c + 1; t < s + len; t++)
20     {
21         if (*c >= *t)re++;
22     }
23     return re;
24 }
25 //得到字符串康拓展開值(從0開始)
26 int V_CantorExpansion(char *s)
27 {
28     int ret = 0;
29     int len = strlen(s);
30     for (int i = 0; i < len; i++)
31     {
32         ret += ID(s+i,s,len)*Cal(len - i - 1);
33     }
34     return ret;
35 }
36 //求k個字符(升序)第n個全排列
37 string Inv_CantorExpansion(char*s, int n)
38 {
39     int len = strlen(s);
40     string ini = s;
41     string ret;
42     for (int i = len - 1; i>= 0; --i)
43     {
44         int pos = n / Cal(i);
45         ret.push_back(ini[pos]);
46         ini.erase(pos, 1);
47         n %= Cal(i);
48     }
49     return ret;
50 }
51 int main()
52 {
53 
54     while (1)
55     {
56         printf("輸入不超過10位的字符串計算康拓展開值:\n");
57         scanf("%s", s);
58         printf("%d\n", V_CantorExpansion(s));
59         printf("輸入不超過10位的初始字符串(升序)計算第n個全排列:\n");
60         scanf("%s", s);
61         int n;
62         scanf("%d", &n);
63         printf("%s\n", Inv_CantorExpansion(s, n).c_str());
64     }
65     return 0;
66 }
View Code

?

轉載于:https://www.cnblogs.com/ivan-count/articles/7466044.html

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/257560.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/257560.shtml
英文地址,請注明出處:http://en.pswp.cn/news/257560.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

java類排序

1、實現Comparator接口 public static class ComparatorImpl implements Comparator<Element>{Overridepublic int compare(Element o1, Element o2) {if(o1.unitPrice > o2.unitPrice)return 1;else if(o1.unitPrice < o2.unitPrice){return -1;}else{return 0;}}…

java jni so_java 用jni調用so全過程

這幾天一直在研究JNI的開發過程&#xff0c;順便把NDK環境搭建一起總結下。在windows環境下開發jni需要c/c編譯器的支持&#xff0c;網絡上我看很多人使用cygwin。呵呵我不是很喜歡使用它&#xff0c;感覺安裝起來挺麻煩的。我使用GNUStep&#xff0c;下載地址http://www.gnust…

ios開發之 -- 自動輪播圖創建

這里是oc版本的&#xff0c;簡單記錄下&#xff1a; 具體代碼如下&#xff1a; 1&#xff0c;準備 #define FRAME [[UIScreen mainScreen] bounds] #define WIDTH FRAME.size.width #define HEIGHT FRAME.size.height 2&#xff0c;具體實現 //scrollview的添加_bigScrollView…

學習進度(2016.3.13)

第二周所花時間&#xff08;包括上課&#xff09;14小時代碼量&#xff08;行&#xff09;138行博客量&#xff08;篇&#xff09;4篇了解到的知識點動態數組的定義初始化和使用&#xff0c;指定范圍獲得隨機數轉載于:https://www.cnblogs.com/zzcs/p/5272365.html

binaryoperator java_BinaryOperatorT接口的用法示例

java Function函數中的BinaryOperator接口用于執行lambda表達式并返回一個T類型的返回值&#xff0c;下面的BinaryOperator用法示例讓你簡單了解一下。import java.util.function.BinaryOperator;public class TestDemo {public static void main(String[] args) {BinaryOperat…

線性表的順序存儲結構之順序表類的實現_Java

在上一篇博文——線性表接口的實現_Java中&#xff0c;我們實現了線性表的接口&#xff0c;今天讓我們來實現線性表的順序存儲結構——順序表類。 首先讓我們來看下順序表的定義&#xff1a; 線性表的順序存儲是用一組連續的內存單元依次存放線性表的數據元素&#xff0c;元素在…

Linux下安裝jdk

參考于&#xff1a;http://www.cnblogs.com/caosiyang/archive/2013/03/14/2959087.html 一、準備階段 ①下載jdk-6u45-linux-i586.bin&#xff0c;通過xftp上傳至Linux系統中 ②在命令行執行 ./jdk-6u45-linux-i586.bin&#xff0c;生成目錄jdk1.6.0_45 ③移動到/usr/share下&…

JDK source 之 ArrayList 需要注意事項

線程安全 ArrayList內部沒有實現原子性操作&#xff0c;所以是非線程安全的。如果需要在線程安全的環境下使用List的話&#xff0c;需要使用Vector 或者CopyOnWriteArrayList&#xff0c;具體場景&#xff0c;自行深入了解。 擴容算法 // minCapacity 為需要的最小容量 private…

為Tiny4412設備驅動在proc目錄下添加一個可讀版本信息的文件

http://blog.csdn.net/morixinguan/article/details/77808088 上節&#xff0c;我們明白了proc文件系統的作用&#xff0c;接下來我們在友善之臂已經寫好的led驅動的基礎上&#xff0c;在proc目錄下創建一個文件夾&#xff0c;然后加入led驅動的版本信息讀取。 我們在init函數的…

java audiorecord_Android 錄音實現(AudioRecord)

上一篇文章介紹了使用 MediaRecorder 實現錄音功能 Android錄音實現(MediaRecorder) &#xff0c;下面我們繼續看看使用 AudioRecord 實現錄音功能。AudioRecord首先看看Android幫助文檔中對該類的簡單概述: AndioRecord 類的主要功能是讓各種 Java 應用能夠管理音頻資源&#…

SqlServer中的數據類型UniqueIdentifier

SqlServer中的數據類型UniqueIdentifier究竟是什么東東&#xff1f;該類型一般用來做為主鍵使用&#xff0c;可用SQL語法的newid()來生成一個唯一的值。我想請問的是&#xff0c;這個值是一個長整型的數據值呢&#xff0c;還是個其他的什么值&#xff1f;我在程序中該怎樣去控制…

《架構探險——從零開始寫Java Web框架》這書不錯,能看懂的入門書

這書適合我。 哈哈&#xff0c;結合 以前的知識點&#xff0c;勉強能看懂。 講得細&#xff0c;還可以參照著弄出來。 希望能堅持 完成啦。。。 原來&#xff0c;JSTL就類似于DJANGO中的模板。 而servlet類中的res,req&#xff0c;玩了DJANGO就覺得好熟悉啦。。。&#xff1a;&…

java 生成 tar.gz_一文教您如何通過 Java 壓縮文件,打包一個 tar.gz Filebeat 采集器包...

一、背景最近&#xff0c;小哈主要在負責日志中臺的開發工作, 等等&#xff0c;啥是日志中臺&#xff1f;俺只知道中臺概念&#xff0c;這段時間的確很火&#xff0c;但是日志中臺又是用來干啥的&#xff1f;這里小哈盡量地通俗的說下日志中臺的職責&#xff0c;再說日志中臺之…

腳本安裝smokeping

我將提供兩種方法來安裝smokeping&#xff0c;一種是大家常用的普通安裝&#xff0c;另一種是用腳本下自動化安裝的&#xff0c;僅供大家學習&#xff0c;參考!普通安裝&#xff1a;centos 5.4下安裝smokeping需要的軟件:(1)httpd(2)rrdtool(3)smokeping(4)fping(5)libwww-perl…

強烈推薦:Android史上最強大的自定義任務軟件Tasker

強烈推薦&#xff1a;Android史上最強大的自定義任務軟件Taskerhttp://bbs.mumayi.com/thread-28387-1-1.html(出處: 木螞蟻手機樂園) Android上的Tasker絕對稱得上是Android系統的神器之一&#xff0c;與Auto Memory Manager不同&#xff0c;Tasker不是加速型的軟件&#xff0…

配置文件*.xml中 classpath: 與 classpath*: 的區別

首先classpath 指的是WEB-INF下面的classes目錄&#xff0c;所有src目錄下面的java、xml、properties等文件編譯后都會在此,classes在eclipse的項目目錄下是看不到的&#xff0c;它存在于部署在服務器上的項目目錄WEB-INF下 classpath:指的是第一個classpath路徑&#xff0c;也…

原型模式 java 深淺_JAVA設計模式---原型模式--淺客隆和深克隆

JAVA淺克隆和深克隆淺克隆&#xff1a;被復制對象的所有變量和原來相同&#xff0c;而所有的對其他對象的引用仍指向原對象。即如果復制的對象修改復制對象的變量&#xff0c;原對象不會改變。而修改引用的對象&#xff0c;二者均會發生改變。深復制(克隆)&#xff1a;被復制對…

SocketErrorCode:10022

在編寫.net的網絡服務器時&#xff0c;我使用了裸socket來實現。在windows上&#xff0c;或者在linux上通過.net core來跑時都沒有什么問題&#xff0c;但是通過mono運行調用socket.Bind()時卻總是報ErrorCode為10022的SocketException&#xff0c;表示參數無效。通過命令netst…

request.RequestContextListener

由于是使用spring mvc來做項目&#xff0c;因此脫離了HttpServletRequest作為參數&#xff0c;不能夠直接使用request&#xff0c;要想使用request可以使用下面的方法&#xff1a; 在web點xml中配置一個監聽 [html] view plaincopyprint?<listener> <listen…

poj1741 Tree 點分治

入門題&#xff0c;算是對樹分治有了初步的理解吧。 #include<iostream> #include<cstdio> #include<cstring> #include<cstdlib> #include<algorithm> #include<vector> #define REP(i,a,b) for(int ia;i<b;i) #define MS0(a) memset(…