给定一个长度为 n
的整数数组,你的任务是判断在最多改变 1
个元素的情况下,该数组能否变成一个非递减数列。
我们是这样定义一个非递减数列的: 对于数组中所有的 i
(1 <= i < n),满足 array[i] <= array[i + 1]
。
示例 1:
输入: [4,2,3] 输出: True 解释: 你可以通过把第一个4变成1来使得它成为一个非递减数列。
示例 2:
输入: [4,2,1] 输出: False 解释: 你不能在只改变一个元素的情况下将其变为非递减数列。
说明: n
的范围为 [1, 10,000]。
思路:
首先数组只能由一组或二组非递减数列组成,只有一组则直接返回true,不止一组则会出现断点。
找到断点后首先分析一下断点处能否修复,如果断点处已经无法修复则直接返回false;如果能修复则从断点处往后遍历判断后面是否是一个延续到结尾的非递减数列,是则true,否则false。难点在于分析断点处能否修复。
分三种情况:第一个断点左边出现“平台”;右边出现“平台”,及两边都不出现“平台”。
具体见代码:
class Solution {
public:
bool checkPossibility(vector<int>& nums) {
int i,j,combo,n=nums.size();
for(i=1,combo=0;nums[i]>=nums[i-1]&&i<n;i++){
if(nums[i]==nums[i-1])combo++;
else combo=0;
}
if(i>=n-1)return 1;
else if(combo){//左平台
if(nums[i+1]<nums[i-1])return 0;
else{
for(j=i+2;nums[j]>=nums[j-1]&&j<n;j++);
if(j<n)return 0;
else return 1;
}
}
else if(nums[i]==nums[i+1]){//右平台
if(i>1&&nums[i-2]>nums[i])return 0;
else{
for(j=i+2;nums[j]>=nums[j-1]&&j<n;j++);
if(j<n)return 0;
else return 1;
}
}
else{
if((nums[i+1]<nums[i-1])&&(nums[i-2]>nums[i]))return 0;
else{
for(j=i+1;nums[j]>=nums[j-1]&&j<n;j++);
if(j<n)return 0;
else return 1;
}
}
}
};
其实可以变得简洁一些,这个只是照着一开始的思路写出来的,没有再修改。
结果:
显然,我的解法是不好的,最快的那个真的厉害了,还没看懂他的代码,以后好好研究一下。