python 数学+减治、下一个排列法、DFS回溯法实现:第 k 个排列【LeetCode 题目 60】

2024-04-23 09:04

本文主要是介绍python 数学+减治、下一个排列法、DFS回溯法实现:第 k 个排列【LeetCode 题目 60】,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

作者介绍:10年大厂数据\经营分析经验,现任大厂数据部门负责人。
会一些的技术:数据分析、算法、SQL、大数据相关、python
欢迎加入社区:码上找工作
作者专栏每日更新:
LeetCode解锁1000题: 打怪升级之旅
python数据分析可视化:企业实战案例
备注说明:方便大家阅读,统一使用python,带必要注释,公众号 数据分析螺丝钉 一起打怪升级

题目描述

给出集合 [1,2,3,...,n],其所有元素共有 n! 种排列。按大小顺序列出所有排列情况,并一一标记,当 n = 3 时, 所有排列如下:

  • “123”
  • “132”
  • “213”
  • “231”
  • “312”
  • “321”

给定 nk,返回第 k 个排列。

输入格式
  • n:一个整数,表示集合的大小。
  • k:一个整数,表示所求的排列序号。
输出格式
  • 返回一个字符串,表示第 k 个排列。
示例 1
输入: n = 3, k = 3
输出: "213"
示例 2
输入: n = 4, k = 9
输出: "2314"

方法一:数学 + 减治法

解题步骤
  1. 计算阶乘:首先计算所有小于等于 n 的数字的阶乘,这有助于后续确定每位数字的位置。
  2. 确定每位数字:从最高位开始,根据阶乘数确定每一位在剩余数字中的位置。
  3. 更新 k 值:更新 kk 减去前面已确定位的组合数。
  4. 重复选择数字:直到所有位置都填满。
完整的规范代码
def getPermutation(n, k):"""使用数学方法和减治法获取第k个排列:param n: int, 集合的大小:param k: int, 排列的序号:return: str, 第k个排列"""factorial = [1] * nfor i in range(1, n):factorial[i] = factorial[i - 1] * ik -= 1  # 转换成索引answer = []numbers = list(range(1, n + 1))for i in range(1, n + 1):index = k // factorial[n - i]answer.append(str(numbers.pop(index)))k %= factorial[n - i]return ''.join(answer)# 示例调用
print(getPermutation(3, 3))  # 输出: "213"
print(getPermutation(4, 9))  # 输出: "2314"
算法分析
  • 时间复杂度:(O(n^2)),计算阶乘数组为 (O(n)),确定每一位数字为 (O(n^2))(因为每次都要从列表中删除元素)。
  • 空间复杂度:(O(n)),存储阶乘数组和数字列表。

方法二:下一个排列法

解题步骤
  1. 生成最小排列:首先生成 [1,2,...,n]
  2. 应用 next permutation:应用 k-1 次“下一个排列”算法得到第 k 个排列。
完整的规范代码
def getPermutation(n, k):"""使用next permutation方法获取第k个排列:param n: int, 集合的大小:param k: int, 排列的序号:return: str, 第k个排列"""def next_permutation(nums):i = j = len(nums) - 1while i > 0 and nums[i-1] >= nums[i]:i -= 1if i == 0:   # nums are in descending ordernums.reverse()returnk = i - 1    # find the last "ascending" positionwhile nums[j] <= nums[k]:j -= 1nums[k], nums[j] = nums[j], nums[k]  l, r = k+1, len(nums)-1  # reverse the second partwhile l < r:nums[l],nums[r] = nums[r], nums[l]l +=1; r -= 1nums = list(range(1, n + 1))for _ in range(k - 1):next_permutation(nums)return ''.join(map(str, nums))# 示例调用
print(getPermutation(3, 3))  # 输出: "213"
print(getPermutation(4, 9))  # 输出: "2314"
算法分析
  • 时间复杂度:(O(n \times k)),每次生成下一个排列需要 (O(n)) 时间。
  • 空间复杂度:(O(n)),存储数字列表。

方法三:DFS回溯法

解题步骤
  1. DFS遍历:使用深度优先搜索遍历所有可能的排列。
  2. 计数并返回:当遍历到第 k 个排列时立即返回。
完整的规范代码
def getPermutation(n, k):"""使用DFS回溯法获取第k个排列:param n: int, 集合的大小:param k: int, 排列的序号:return: str, 第k个排列"""def dfs(path):nonlocal countif len(path) == n:count += 1if count == k:return pathreturnfor number in range(1, n+1):if number in path:continueres = dfs(path + [number])if res:return rescount = 0result = dfs([])return ''.join(map(str, result)) if result else ""# 示例调用
print(getPermutation(3, 3))  # 输出: "213"
print(getPermutation(4, 9))  # 输出: "2314"
算法分析
  • 时间复杂度:(O(n!)),理论上需要遍历所有排列。
  • 空间复杂度:(O(n)),递归深度为 n

不同算法的优劣势对比

在这里插入图片描述

应用示例详解:密码生成系统

场景描述

在密码生成和密码管理软件中,经常需要生成复杂且难以预测的密码来增加安全性。使用“第 k 个排列”算法可以在预定义字符集上生成随机但确定的密码,适用于需要高安全性的应用场景,如在线银行、军事通信等。

方法:数学+减治法

技术选择
选择方法一(数学+减治法),因为它可以直接计算出第 k 个排列而无需生成所有排列,提高了生成效率和保密性。

实现步骤

  1. 选择字符集:定义一个字符集,例如包含大小写字母和数字 [1-9, a-z, A-Z]
  2. 计算阶乘:预先计算出所有小于字符集大小的阶乘,用于后续计算排列位置。
  3. 确定每位字符:根据阶乘和 k 值,快速确定每一位置上的字符,直接计算出第 k 个排列。
  4. 生成密码:将计算出的排列作为密码,提供给用户或用于加密应用。

代码实现

def getPermutation(characters, k):"""使用数学方法和减治法基于给定字符集生成密码:param characters: str, 字符集:param k: int, 指定的排列序号:return: str, 生成的密码(排列)"""n = len(characters)factorial = [1] * (n + 1)for i in range(2, n + 1):factorial[i] = factorial[i - 1] * ik -= 1  # 转换为基于0的索引answer = []numbers = list(characters)for i in range(1, n + 1):index = k // factorial[n - i]answer.append(numbers.pop(index))k %= factorial[n - i]return ''.join(answer)# 示例调用
chars = "123456789ABCDEF"
k = 9432
print(getPermutation(chars, k))  # 输出: 第9432个排列
应用优势
  • 效率高:直接计算第 k 个排列,无需枚举所有可能,适合实时密码生成需求。
  • 安全性强:密码的生成基于数学计算,没有明显的规律,安全性高。
  • 适用性广:可根据不同的字符集和需求灵活定制密码生成策略。

总结

通过在密码生成系统中应用“第 k 个排列”算法,开发者可以提供一种高效且安全的方式来生成复杂密码。此外,该算法的高计算效率和确定性也使其成为理想的选择,用于需要快速生成大量密码或密钥的场合,比如动态令牌生成、临时密码分配等。此方法不仅优化了密码生成过程,也极大提高了密码管理系统的整体安全性。

这篇关于python 数学+减治、下一个排列法、DFS回溯法实现:第 k 个排列【LeetCode 题目 60】的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/928355

相关文章

Python中的魔术方法__new__详解

《Python中的魔术方法__new__详解》:本文主要介绍Python中的魔术方法__new__的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、核心意义与机制1.1 构造过程原理1.2 与 __init__ 对比二、核心功能解析2.1 核心能力2.2

Python虚拟环境终极(含PyCharm的使用教程)

《Python虚拟环境终极(含PyCharm的使用教程)》:本文主要介绍Python虚拟环境终极(含PyCharm的使用教程),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录一、为什么需要虚拟环境?二、虚拟环境创建方式对比三、命令行创建虚拟环境(venv)3.1 基础命令3

Python Transformer 库安装配置及使用方法

《PythonTransformer库安装配置及使用方法》HuggingFaceTransformers是自然语言处理(NLP)领域最流行的开源库之一,支持基于Transformer架构的预训练模... 目录python 中的 Transformer 库及使用方法一、库的概述二、安装与配置三、基础使用:Pi

Python 中的 with open文件操作的最佳实践

《Python中的withopen文件操作的最佳实践》在Python中,withopen()提供了一个简洁而安全的方式来处理文件操作,它不仅能确保文件在操作完成后自动关闭,还能处理文件操作中的异... 目录什么是 with open()?为什么使用 with open()?使用 with open() 进行

openCV中KNN算法的实现

《openCV中KNN算法的实现》KNN算法是一种简单且常用的分类算法,本文主要介绍了openCV中KNN算法的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的... 目录KNN算法流程使用OpenCV实现KNNOpenCV 是一个开源的跨平台计算机视觉库,它提供了各

Python中使用正则表达式精准匹配IP地址的案例

《Python中使用正则表达式精准匹配IP地址的案例》Python的正则表达式(re模块)是完成这个任务的利器,但你知道怎么写才能准确匹配各种合法的IP地址吗,今天我们就来详细探讨这个问题,感兴趣的朋... 目录为什么需要IP正则表达式?IP地址的基本结构基础正则表达式写法精确匹配0-255的数字验证IP地

OpenCV图像形态学的实现

《OpenCV图像形态学的实现》本文主要介绍了OpenCV图像形态学的实现,包括腐蚀、膨胀、开运算、闭运算、梯度运算、顶帽运算和黑帽运算,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起... 目录一、图像形态学简介二、腐蚀(Erosion)1. 原理2. OpenCV 实现三、膨胀China编程(

通过Spring层面进行事务回滚的实现

《通过Spring层面进行事务回滚的实现》本文主要介绍了通过Spring层面进行事务回滚的实现,包括声明式事务和编程式事务,具有一定的参考价值,感兴趣的可以了解一下... 目录声明式事务回滚:1. 基础注解配置2. 指定回滚异常类型3. ​不回滚特殊场景编程式事务回滚:1. ​使用 TransactionT

Android实现打开本地pdf文件的两种方式

《Android实现打开本地pdf文件的两种方式》在现代应用中,PDF格式因其跨平台、稳定性好、展示内容一致等特点,在Android平台上,如何高效地打开本地PDF文件,不仅关系到用户体验,也直接影响... 目录一、项目概述二、相关知识2.1 PDF文件基本概述2.2 android 文件访问与存储权限2.

使用Python实现全能手机虚拟键盘的示例代码

《使用Python实现全能手机虚拟键盘的示例代码》在数字化办公时代,你是否遇到过这样的场景:会议室投影电脑突然键盘失灵、躺在沙发上想远程控制书房电脑、或者需要给长辈远程协助操作?今天我要分享的Pyth... 目录一、项目概述:不止于键盘的远程控制方案1.1 创新价值1.2 技术栈全景二、需求实现步骤一、需求