当前位置: 首页 > news >正文

【leetcode】排列序列

给出集合 [1,2,3,...,n],其所有元素共有 n! 种排列。

按大小顺序列出所有排列情况,并一一标记,当 n = 3 时, 所有排列如下:

  1. "123"
  2. "132"
  3. "213"
  4. "231"
  5. "312"
  6. "321"

给定 n 和 k,返回第 k 个排列。

示例 1:

输入:n = 3, k = 3
输出:"213"

示例 2:

输入:n = 4, k = 9
输出:"2314"

示例 3:

输入:n = 3, k = 1
输出:"123"

提示:

  • 1 <= n <= 9
  • 1 <= k <= n!

解题思路:

例如: n = 4, k = 9

        可以知道全排列有:4*3*2*1 = 24种,如下:红色框住的为第9个

                

        从上图可知,以1开头的全排列 (4-1)! = 6 共6种

        9 / 6 = 1······3,也就是该数位于2开头的第三个。

        同理,第一位2确定后,剩下三位 1 3 4,接下来需要确定第二位;

       

              

        由上面的图片可知, 1 3 4全排列每一列分别有两个数3-1)!,3 / 2 = 1······1,可知位于第二列开头的第一个

然后确定第三位 ,第三位则还剩下2个数:1和4;

        1和4的全排列:

       

                                 

        1 4排列,每类(2-1)! = 1,1 / 1 = 1·····0,可知位于1开头的第一个

        最后一位数,只有一种情况:  (1-1)! = 1

具体算法如下:

1、定义一个逆康托数组,记录n位数字对应每个数字开头序列的个数,例如4位数prenum[1, 1, 2, 6]

将k值-1,再进行计算,这里-1是为了  9 % 6 = 3,而数组下标从0开始,-1后方便计算。

2、定义一个valid数组[0.....n];记录还没有被使用的数字, 已经使用的需要移除

3、(k -1 ) / prenum[n - i - 1]   + 1就是分别计算每一位数字位于那一列;例如 (9 -1)/ 6  + 1= 2,说明第一个数字为2,ans累加vali[2] ,2被使用后erase掉,valid中剩余[0,1,3,4]

        然后k记录剩余未计算的数 8 % 6 = 2

        k = 2再重复计算,可得到 2/2 + 1 = 2;  再取走valid中的第2个

        同理依次取完。

完整代码如下:

class Solution {
public:string getPermutation(int n, int k) {//prenum = {0!, 1!, 2!, 3!, .....(n-1)!}vector<int> pernum(n);pernum[0] = 1;for(int i= 1; i < n; i++) {pernum[i] = pernum[i-1] * i;}//k-- 方便取余后计算k--;string ans;vector<int> valid(n+1);//生成[0,1,2....n]的数组iota(valid.begin(), valid.end(), 0);for(int i = 0; i < n; i++) {int order = k / pernum[n-i-1] + 1;ans += (valid[order] + '0');//用了就从数组中删除valid.erase(valid.begin() + order);k %= pernum[n-i-1];}return ans;}
};

拓展:康托展开,也就是在全排列中,给定一个序列,求该序列位于第几个。

#include <iostream>
#include <string>
using namespace std;class Solution {public://求阶乘int fact(int x) {int ret = 1;for(int i = 2; i <= x; i++) {ret *= i;}return ret;}int getStringId(string str) {int len = str.size();int ans = 0;for(int i = 0; i < len; i++) {int cnt = 0;//记录后面的数有几个比当前的数小;//例如2314  314中只有1个数比第一位小,说明该数位于第二列。for(int j = i + 1 ; j < len; j++) {if(str[j] < str[i]) {cnt ++;}}ans += cnt*fact(len- i -1);}return ans+1;}
};
int main()
{string s;cin >> s;Solution sol;cout << "位于序列第:" << sol.getStringId(s) << " 位" << endl;return 0;
}

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • HTML5实现好看的天气预报网站源码
  • SQL injection UNION attacks SQL注入联合查询攻击
  • 【Spark On Hive】—— 基于电商数据分析的项目实战
  • 云计算实训11——web服务器的搭建、nfs服务器的搭建、备份静态文件、基于linux和windows实现文件共享
  • Hadoop中HDFS、Hive 和 HBase三者之间的关系
  • Modbus转BACnet/IP网关快速对接Modbus协议设备与BA系统
  • SpringBoot+Session+redis实现分布式登录
  • 深度学习之DeepMind的MuZero
  • 初学51单片机之指针基础与串口通信应用
  • C#进阶-基于.NET Framework 4.x框架实现ASP.NET WebForms项目IP拦截器
  • WSL 2 Oracle Linux 9.1 安装配置
  • MySQL(1)
  • 配置RIPv2的认证
  • 详解Stable Diffusion 原理图
  • excel批量新建多个同类型的表格
  • JS中 map, filter, some, every, forEach, for in, for of 用法总结
  • 【Redis学习笔记】2018-06-28 redis命令源码学习1
  • CoolViewPager:即刻刷新,自定义边缘效果颜色,双向自动循环,内置垂直切换效果,想要的都在这里...
  • CSS相对定位
  • go append函数以及写入
  • HTTP传输编码增加了传输量,只为解决这一个问题 | 实用 HTTP
  • iOS仿今日头条、壁纸应用、筛选分类、三方微博、颜色填充等源码
  • JavaScript 无符号位移运算符 三个大于号 的使用方法
  • js
  • leetcode-27. Remove Element
  • Nacos系列:Nacos的Java SDK使用
  • SpiderData 2019年2月25日 DApp数据排行榜
  • 好的网址,关于.net 4.0 ,vs 2010
  • 基于 Babel 的 npm 包最小化设置
  • 前端代码风格自动化系列(二)之Commitlint
  • 删除表内多余的重复数据
  • 实现简单的正则表达式引擎
  • 小程序01:wepy框架整合iview webapp UI
  • 移动端解决方案学习记录
  • 用 Swift 编写面向协议的视图
  • 支付宝花15年解决的这个问题,顶得上做出十个支付宝 ...
  • ​ 全球云科技基础设施:亚马逊云科技的海外服务器网络如何演进
  • ​【已解决】npm install​卡主不动的情况
  • ​Python 3 新特性:类型注解
  • ​ubuntu下安装kvm虚拟机
  • ​什么是bug?bug的源头在哪里?
  • ​香农与信息论三大定律
  • ​用户画像从0到100的构建思路
  • # 学号 2017-2018-20172309 《程序设计与数据结构》实验三报告
  • #13 yum、编译安装与sed命令的使用
  • #HarmonyOS:基础语法
  • (12)目标检测_SSD基于pytorch搭建代码
  • (175)FPGA门控时钟技术
  • (5)STL算法之复制
  • (AngularJS)Angular 控制器之间通信初探
  • (delphi11最新学习资料) Object Pascal 学习笔记---第13章第1节 (全局数据、栈和堆)
  • (Python第六天)文件处理
  • (react踩过的坑)antd 如何同时获取一个select 的value和 label值
  • (附源码)spring boot校园拼车微信小程序 毕业设计 091617
  • (附源码)计算机毕业设计ssm电影分享网站