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
相关推荐
We་ct2 小时前
LeetCode 5. 最长回文子串:DP + 中心扩展
前端·javascript·算法·leetcode·typescript
王老师青少年编程6 小时前
csp信奥赛C++高频考点专项训练之贪心算法 --【哈夫曼贪心】:合并果子
c++·算法·贪心·csp·信奥赛·哈夫曼贪心·合并果子
叼烟扛炮7 小时前
C++第二讲:类和对象(上)
数据结构·c++·算法·类和对象·struct·实例化
天疆说7 小时前
【哈密顿力学】深入解读航天器交会最优控制中的Hamilton函数
人工智能·算法·机器学习
wuweijianlove8 小时前
关于算法设计中的代价函数优化与约束求解的技术7
算法
leoufung8 小时前
LeetCode 149: Max Points on a Line - 解题思路详解
算法·leetcode·职场和发展
样例过了就是过了8 小时前
LeetCode热题100 最长公共子序列
c++·算法·leetcode·动态规划
HXDGCL9 小时前
矩形环形导轨:自动化循环线的核心运动单元解析
运维·算法·自动化
谭欣辰9 小时前
C++ 排列组合完整指南
开发语言·c++·算法
代码中介商9 小时前
银行管理系统的业务血肉 —— 流程、状态机、输入校验与持久化(下篇)
c语言·算法