leetcode 1806. 还原排列的最少操作步数

题目链接:leetcode 1806

1.题目

给你一个偶数 n​​​​​​ ,已知存在一个长度为 n 的排列 perm ,其中 perm[i] == i​(下标 从 0 开始 计数)。

一步操作中,你将创建一个新数组 arr ,对于每个 i :

如果 i % 2 == 0 ,那么 arr[i] = perm[i / 2]

如果 i % 2 == 1 ,那么 arr[i] = perm[n / 2 + (i - 1) / 2]

然后将 arr​​ 赋值​​给 perm 。

要想使 perm 回到排列初始值,至少需要执行多少步操作?返回最小的 非零 操作步数。

2.示例

1)示例 1:

输入:n = 2

输出:1

解释:最初,perm = [0,1]

第 1 步操作后,perm = [0,1]

所以,仅需执行 1 步操作

2)示例 2:

输入:n = 4

输出:2

解释:最初,perm = [0,1,2,3]

第 1 步操作后,perm = [0,2,1,3]

第 2 步操作后,perm = [0,1,2,3]

所以,仅需执行 2 步操作

3)示例 3:

输入:n = 6

输出:4

4)提示:

2 <= n <= 1000

n​​​​​​ 是一个偶数

3.分析

我们以n=6为例,可以发现无论perm数组如何变化,arr数组每个下表对应perm数组的下标是不会变化的,那么对应关系如下:

0:0

1:3

2:1

3:4

4:2

5:5

我们以下标1为例子,从最初arr[1]=perm[3]逐步推,可以发现下标的顺序分别为:3,4,2,1,到最后arr[1]=perm[1]=1,所以问题转化为求最小环的长度

4.代码

cpp 复制代码
class Solution {
public:
    map<int,int> map1;
    int ans=0;
    void get_dfs(int x){
        ans++;
        if(map1[x]==1) return;
        get_dfs(map1[x]);
    }
    int reinitializePermutation(int n) {
        for(int i=1;i<n;i++)
            if(i%2==0) map1[i]=i/2;
            else map1[i]=n/2+(i-1)/2;
        get_dfs(1);return ans;
    }
};
// 0:0
// 1:3
// 2:1
// 3:4
// 4:2
// 5:5
// 6:3
相关推荐
iAkuya1 分钟前
(leetcode)力扣100 62N皇后问题 (普通回溯(使用set存储),位运算回溯)
算法·leetcode·职场和发展
近津薪荼2 分钟前
dfs专题5——(二叉搜索树中第 K 小的元素)
c++·学习·算法·深度优先
xiaoye-duck3 分钟前
吃透 C++ STL list:从基础使用到特性对比,解锁链表容器高效用法
c++·算法·stl
松☆7 分钟前
CANN与大模型推理:在边缘端高效运行7B参数语言模型的实践指南
人工智能·算法·语言模型
java干货17 分钟前
为什么 “File 10“ 排在 “File 2“ 前面?解决文件名排序的终极算法:自然排序
开发语言·python·算法
皮皮哎哟25 分钟前
数据结构:嵌入式常用排序与查找算法精讲
数据结构·算法·排序算法·二分查找·快速排序
程序员清洒34 分钟前
CANN模型剪枝:从敏感度感知到硬件稀疏加速的全链路压缩实战
算法·机器学习·剪枝
vortex51 小时前
几种 dump hash 方式对比分析
算法·哈希算法
Wei&Yan2 小时前
数据结构——顺序表(静/动态代码实现)
数据结构·c++·算法·visual studio code
团子的二进制世界2 小时前
G1垃圾收集器是如何工作的?
java·jvm·算法