LeetCode 2009. 使数组连续的最少操作数

给你一个整数数组 nums 。每一次操作中,你可以将 nums 中 任意 一个元素替换成 任意 整数。

如果 nums 满足以下条件,那么它是 连续的 :

nums 中所有元素都是 互不相同 的。
nums 中 最大 元素与 最小 元素的差等于 nums.length - 1 。
比方说,nums = [4, 2, 5, 3] 是 连续的 ,但是 nums = [1, 2, 3, 5, 6] 不是连续的 。

请你返回使 nums 连续 的 最少 操作次数。

示例 1:

输入:nums = [4,2,5,3]
输出:0
解释:nums 已经是连续的了。
示例 2:

输入:nums = [1,2,3,5,6]
输出:1
解释:一个可能的解是将最后一个元素变为 4 。
结果数组为 [1,2,3,5,4] ,是连续数组。
示例 3:

输入:nums = [1,10,100,1000]
输出:3
解释:一个可能的解是:
- 将第二个元素变为 2 。
- 将第三个元素变为 3 。
- 将第四个元素变为 4 。
结果数组为 [1,2,3,4] ,是连续数组。

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/minimum-number-of-operations-to-make-array-continuous
 

解法

排序,去重,二分。先对数组排序并去重的到新的数组,对新数组中每个元素查找以它为最小值时,最大值的所在位置。则原数组大小len-(maxId-i+1)为此时操作数。


class Solution {

public:
	int minOperations(vector<int>& nums) {
		sort(nums.begin(), nums.end());
		//set<int> s;
		int len = nums.size();
		vector<int> snums;
		for (int i = 0; i < len; i++)
		{
			if (i == 0 || nums[i] != nums[i-1])
				snums.push_back(nums[i]);
		}
		int slen = snums.size();
		int minNum = 1e5 + 1;
		for (int i = 0; i<slen;i++)
		{
			int id = lower_bound(snums.begin(), snums.end(), snums[i] + (len - 1)) - snums.begin();
			if (id == slen)
				--id;
			if (snums[id]>snums[i] + (len - 1))
				--id;
			minNum = min(minNum, len - (id - i + 1));
		}
		return minNum;
	}
};

对比i与i+1对应的最大值位置posi与pos(i+1),肯定是pos(i+1)>pos(i),故可以用双指针来优化。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值