【康拓展开】及其在求全排列第k个数中的应用

时间:2022-09-30 09:24:01

题目:给出n个互不相同的字符, 并给定它们的相对大小顺序,这样n个字符的所有排列也会有一个顺序. 现在任给一个排列,求出在它后面的第i个排列.
这是一个典型的康拓展开应用,首先我们先阐述一下什么是康拓展开。

(1)康拓展开

  所谓康拓展开是指把一个整数X展开成如下形式:

  X=a[n]*(n-1)!+a[n-1]*(n-2)!+...+a[i]*(i-1)!+...+a[2]*1!+a[1]*0!。(其中,a为整数,并且0<=a[i]<i(1<=i<=n))

(2)应用实例

  {1,2,3,4,...,n}表示1,2,3,...,n的排列如 {1,2,3} 按从小到大排列一共6个:123 132 213 231 312 321。他们间的对应关系可由康托展开来找到。

  1324是{1,2,3,4}排列数中第几个大的数:
  第一位是1小于1的数没有,是0个 0*3! ;
  第二位是3小于3的数有1和2,但1已经在第一位了,即1未出现在前面的低位当中,所以只有一个数2 1*2! ;
  第三位是2小于2的数是1,但1在第一位,即1未出现在前面的低位当中,所以有0个数 0*1! ;
  所以比1324小的排列有0*3!+1*2!+0*1!=2个,1324是第三个大数。
其代码实现为:
 #include <iostream>
using namespace std; int Cantor(int *s,int n); //康托展开,判断给定的排列位于全排列中的第几个
long int fac[]={,,,,,,,,,}; //表示阶乘运算的结果
//long int fac[]={0!,1!,2!,3!,4!,5!,6!,7!,8!,9!}; int main(int argc,char *argv)
{
int s[]={,,,}; //表示排列2134
int len=; //表示数列中数字数目
int index=Cantor(s,len);
cout<<index<<endl;
return ;
}
int Cantor(int *s,int n)
{
int i,j,num,temp;
num=;
for(i=;i<n;i++)
{
temp=; //temp记录当前数位前面的低数位中小于当前位数上的数字的个数
for(j=i+;j<n;j++)
if(s[j]<s[i])
temp++;
num+=fac[n--i]*temp; //乘以相应的阶乘
}
return num;
}

如何判断给定一个位置,输出该位置上的数列,康拓展开的逆运算,例如:

  {1,2,3,4,5}的全排列,并且已经从小到大排序完毕,请找出第96个数:
 

   首先用96-1得到95

   用95去除4! 得到3余23,即有3个数比该数位上的数字小,则该数位的数字为4;
   用23去除3! 得到3余5,即有3个数比该数位上的数字小,理应为4,但4已在前面的高位中出现过,所以该数位的数字为5;
   用5去除2!得到2余1,即有2个数比该数位上的数字小,则该数位的数字为3;
   用1去除1!得到1余0,即有1个数比该数位上的数字小,则该数位的数字为2;
   最后一个数只能是1;
   所以这个数是45321
其代码实现:
 #include <iostream>
using namespace std; void CantorReverse(int index,int *p,int n); //康托展开逆用,判断给定的位置中的排列
long int fac[]={,,,,,,,,,}; //表示阶乘运算的结果
//long int fac[]={0!,1!,2!,3!,4!,5!,6!,7!,8!,9!}; int main(int argc,char *argv)
{
int len=;
int *s=(int *)malloc(len*sizeof(int));
CantorReverse(,s,len); //有数字{12345}组成的所有排列中,求出第96个排列的顺序
for(int i=;i<len;i++)
cout<<s[i];
cout<<endl;
free(s);
return ;
}
void CantorReverse(int index,int *p,int n)
{
index--; //勿丢
int i,j;
bool hash[]={};
for(i=;i<n;i++)
{
int tmp=index/fac[n--i]; //tmp表示有tmp个数字比当前位置上的数字小
for(j=;j<=tmp;j++)
if(hash[j]) tmp++;
p[i]=tmp+;
hash[tmp]=;
index%=fac[n--i];
}
return;
}

(2)题目解决

  通过以上分析,则本章开头提出的题目就迎刃而解了,先通过给定的序列,求出所在位置,再加上i,得到 i 以后的位置,最后根据位置求出序列。相信大家能自己写出程序,在此就不具体写出了。
 
 
 
 
 
 
 

【康拓展开】及其在求全排列第k个数中的应用的更多相关文章

  1. &lbrack;LeetCode&rsqb;60&period; Permutation Sequence求全排列第k个

    /* n个数有n!个排列,第k个排列,是以第(k-1)/(n-1)!个数开头的集合中第(k-1)%(n-1)!个数 */ public String getPermutation(int n, int ...

  2. 剑指Offer面试题:27&period;最小的k个数

    一.题目:最小的k个数 题目:输入n个整数,找出其中最小的k个数.例如输入4.5.1.6.2.7.3.8这8个数字,则最小的4个数字是1.2.3.4. 这道题是典型的TopK问题,其最简单的思路莫过于 ...

  3. 寻找最小&lpar;最大&rpar;的k个数

    题目描述:输入n个整数,输出其中最小的k个元素. 例如:输入1,2,3,4,5,6,7,8这8个数字,则最小的4个数字为1,2,3,4. 思路1:最容易想到的方法:先对这个序列从小到大排序,然后输出前 ...

  4. 算法系列:寻找最大的 K 个数

    Copyright © 1900-2016, NORYES, All Rights Reserved. http://www.cnblogs.com/noryes/ 欢迎转载,请保留此版权声明. -- ...

  5. 剑指offer面试题30:最小的k个数

    一.题目描述 输入n个整数,找出其中最小的K个数.例如输入4,5,1,6,2,7,3,8这8个数字,则最小的4个数字是1,2,3,4,. 二.解题思路 1.思路1 首先对数组进行排序,然后取出前k个数 ...

  6. 第2章 数字之魅——寻找最大的K个数

    寻找最大的K个数 问题描述 在面试中,有下面的问答: 问:有很多个无序的数,我们姑且假定它们各不相等,怎么选出其中最大的若干个数呢? 答:可以这样写:int array[100] …… 问:好,如果有 ...

  7. 九度OJ 1371 最小的K个数 -- 堆排序

    题目地址:http://ac.jobdu.com/problem.php?pid=1371 题目描述: 输入n个整数,找出其中最小的K个数.例如输入4,5,1,6,2,7,3,8这8个数字,则最小的4 ...

  8. 【算法】数组与矩阵问题——找到无序数组中最小的k个数

    /** * 找到无序数组中最小的k个数 时间复杂度O(Nlogk) * 过程: * 1.一直维护一个有k个数的大根堆,这个堆代表目前选出来的k个最小的数 * 在堆里的k个元素中堆顶的元素是最小的k个数 ...

  9. 【算法与数据结构】在n个数中取第k大的数(基础篇)

    (转载请注明出处:http://blog.csdn.net/buptgshengod) 题目介绍            在n个数中取第k大的数(基础篇),之所以叫基础篇是因为还有很多更高级的算法,这些 ...

随机推荐

  1. 2016的ChinaJoy沦为ChinaVR?

    China Joy已沦为ChinaVR,厂商烧钱参加? 在上海超过40度的高温天下,游戏爱好者们汗流满面地排起长队,拥挤地通过安检进入场馆,但是很快感受到了一丝凉意. ShowGirl少了 &quot ...

  2. ASP&period;NET MVC异步上传文件

    自己做的一个小dome.贴出来分享一下: 前端: <form id="formfile" method="post" enctype="mult ...

  3. 2016 ACM&sol;ICPC Asia Regional Dalian ICPC大连现场赛

    讲道理我挺想去大连的…… 毕竟风景不错…… 而且这次能去北京也是靠大连网络赛这一场拉开的优势…… 一道补图最短路一道数学推论简直爽爆…… 当然 除了这一场 其他场都非常划水…… 上次看到别人的博客用这 ...

  4. 201521123009 《Java程序设计》第9周学习总结

    1. 本周学习总结 2. 书面作业 本次PTA作业题集异常 Q1:常用异常 题目5-1 1.1 截图你的提交结果(出现学号) 1.2 自己以前编写的代码中经常出现什么异常.需要捕获吗(为什么)?应如何 ...

  5. Jenkins &plus; Ansible &plus; Gitlab之gitlab篇

    前言 持续交付 版本控制器:Gitlab.GitHub 持续集成工具:jenkins 部署工具:ansible  课程安排 Gitlab搭建与流程使用 Ansible环境配置与Playbook编写规范 ...

  6. Google第三方网站登录(JavaScript SDK)

    官网:https://developers.google.com/identity/sign-in/web/ 一.创建应用 a.去谷歌控制台创建应用     网址:https://accounts.g ...

  7. Centos7 update dotnet 无法识别

    使用了yum update 后 原来好好的dotnet 用不了了 /usr/bin/dotnet 找不到 卸载重装都没法了.... 解决方法: 把dotnet 拷贝到 /usr/bin 下面去就好了 ...

  8. 洛谷P3567 KUR-Couriers &lbrack;POI2014&rsqb; 主席树&sol;莫队

    正解:主席树/莫队 解题报告: 传送门! 这题好像就是个主席树板子题的样子,,,? 毕竟,主席树的最基本的功能就是,维护一段区间内某个数字的个数 但是毕竟是刚get到主席树,然后之前做的一直是第k大, ...

  9. 2019-04-04-day026-模块和包的导入

    课前 估分 重新做题 思考为什么 积累问题 提前了解你的情况 40分以下 选课系统 按照反射那个版本 把反射的逻辑看明白 接着把逻辑填完整 用上pickle logging写日志 进阶 : 用软件开发 ...

  10. Linux字符集的查看及修改【转】

    一·查看字符集字符集在系统中体现形式是一个环境变量,以CentOS6.5为例,其查看当前终端使用字符集的方式可以有以下几种方式: 1.[root@david ~]# echo $LANGzh_CN.G ...