还原排列的最少操作步数

原题指路

添加链接描述

题目描述

给你一个偶数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 回到排列初始值,至少需要执行多少步操作?返回最小的 非零 操作步数。

解题思路

图片1.png
首先,注意到排列中首元素和尾元素在每次变换中始终不会发生改变。
然后,根据规则可以得到其余元素的索引变化规律:

  • if i < (n / 2), i = i * 2
  • if i >= (n / 2), i = (i - (n / 2)) * 2 + 1

接着……发现在这个变换中,每当1回到自己原来的位置的时候,排列就还原了。(看评论区有说这是n元置换的恒等变换,暂时还不会证明)
悄咪咪加一句:提示里有说数据小到可以直接暴力模拟……

时间复杂度: O ( n ) O(n) O(n)(没证明,不过应该是一个关于n的线性函数)
空间复杂度: O ( 1 ) O(1) O(1)

代码

class Solution:
    def reinitializePermutation(self, n: int) -> int:
        if n == 2:
            return 1
        res, i, k = 1, 2, n / 2
        while i != 1:
            if i < k:
                i *= 2
            else:
                i = (i - k) * 2 + 1
            res += 1
        return res

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值