题目来源
题目描述
题目解析
class Solution {
public:
vector<int> findErrorNums(vector<int>& nums) {
vector<int> ans(2, 0);
// 核心思想是这样的:
// 假设每个人各司其职, 那么每个岗位上就是相应的人员,并且各司其职, 但是现在混入了一个
// 闲杂人等, 他又害怕被人发现, 于是他就别人的岗位上站着, 结果等着等着, 这个岗位上应该来的人来了,
// 他一看, 自己掩饰不过去了(毕竟人家不会跑到别的岗位上,抢别人的活干, 于是他就这么一直碰运气,
// 一直走到那个没来上岗的那个人的岗位上去, 这样谁都不会发现他是的闲杂人等.而且每个岗位上都满员了.
for (int i = 0; i < nums.size(); i++)
{
while (nums[i] != nums[nums[i] - 1])
{
// for (auto i : nums) cout << i << "\t"; cout << endl;
swap(nums[i] , nums[nums[i] - 1]);
}
}
// for (auto i : nums) cout << i << ends; cout << endl;
for (int i = 0; i < nums.size(); ++i)
{
if (nums[i] != i + 1)
return {nums[i], i+1};
}
return ans;
}
};
思路
- 遍历每个数字,然后将其应该出现的位置上的数字变为其相反数,这样如果我们再变为其相反数之前已经成负数了,说明该数字是重复数,将其将入结果res中
- 然后再遍历原数组,如果某个位置上的数字为正数,说明该位置对应的数字没有出现过,加入res中即可
class Solution {
public:
vector<int> findErrorNums(vector<int>& nums) {
vector<int> res(2, -1);
for (int i : nums) {
if (nums[abs(i) - 1] < 0) res[0] = abs(i);
else nums[abs(i) - 1] *= -1;
}
for (int i = 0; i < nums.size(); ++i) {
if (nums[i] > 0) res[1] = i + 1;
}
return res;
}
};
思路
统计每个数字出现的次数了,然后再遍历次数数组,如果某个数字出现了两次就是重复数,如果出现了0次,就是缺失数
class Solution {
public:
vector<int> findErrorNums(vector<int>& nums) {
vector<int> res(2, 0), cnt(nums.size(), 0);
for (int num : nums) ++cnt[num - 1];
for (int i = 0; i < cnt.size(); ++i) {
if (res[0] != 0 && res[1] != 0) return res;
if (cnt[i] == 2) res[0] = i + 1;
else if (cnt[i] == 0) res[1] = i + 1;
}
return res;
}
};