目录
[1. 只出现一次的数字(简单)](#1. 只出现一次的数字(简单))
[1.1. 题目描述](#1.1. 题目描述)
[1.2. 解题思路](#1.2. 解题思路)
[2. 多数元素(简单)](#2. 多数元素(简单))
[2.1. 题目描述](#2.1. 题目描述)
[2.2. 解题思路](#2.2. 解题思路)
[方法五:Boyer-Moore 投票算法](#方法五:Boyer-Moore 投票算法)
[3. 颜色分类](#3. 颜色分类)
[3.1. 题目描述](#3.1. 题目描述)
[3.2. 解题思路](#3.2. 解题思路)
[4. 下一个排列](#4. 下一个排列)
[4.1. 题目描述](#4.1. 题目描述)
[4.2. 解题思路](#4.2. 解题思路)
[5. 寻找重复数](#5. 寻找重复数)
[5.1. 题目描述](#5.1. 题目描述)
[5.2. 解题思路](#5.2. 解题思路)
1. 只出现一次的数字(简单)
1.1. 题目描述
data:image/s3,"s3://crabby-images/7633b/7633bb05bb2a898b41ef34f2c7581b09c565e793" alt=""
1.2. 解题思路
方法一:位运算
data:image/s3,"s3://crabby-images/a47f5/a47f5bbb82992bcd9192d68117fe6de77545461d" alt=""
class Solution {
public int singleNumber(int[] nums) {
int single = 0;
for (int num : nums) {
single ^= num;
}
return single;
}
}
data:image/s3,"s3://crabby-images/ca74c/ca74cd9996d57724fa881d1d8acdfaada7aac656" alt=""
2. 多数元素(简单)
2.1. 题目描述
data:image/s3,"s3://crabby-images/0349c/0349c5f2baba09a54327fc454291c69e82fc90b2" alt=""
2.2. 解题思路
方法一:哈希表
data:image/s3,"s3://crabby-images/03972/03972e2833595117270da4cef2afdf22fe049377" alt=""
class Solution {
private Map<Integer, Integer> countNums(int[] nums) {
Map<Integer, Integer> counts = new HashMap<Integer, Integer>();
for (int num : nums) {
if (!counts.containsKey(num)) {
counts.put(num, 1);
} else {
counts.put(num, counts.get(num) + 1);
}
}
return counts;
}
public int majorityElement(int[] nums) {
Map<Integer, Integer> counts = countNums(nums);
Map.Entry<Integer, Integer> majorityEntry = null;
for (Map.Entry<Integer, Integer> entry : counts.entrySet()) {
if (majorityEntry == null || entry.getValue() > majorityEntry.getValue()) {
majorityEntry = entry;
}
}
return majorityEntry.getKey();
}
}
data:image/s3,"s3://crabby-images/12935/12935bf9825c0139cc69a14df038a030e7b7a333" alt=""
方法二:排序
data:image/s3,"s3://crabby-images/91f0d/91f0d1002438f36f880e180414fde3432b4d1596" alt=""
class Solution {
public int majorityElement(int[] nums) {
Arrays.sort(nums);
return nums[nums.length / 2];
}
}
data:image/s3,"s3://crabby-images/16424/16424a01cb1bae80b65e55449df15d054256e386" alt=""
方法三:随机化
data:image/s3,"s3://crabby-images/c75b4/c75b45efd30b078eef0e58f5b9ed1ddf5958a2f2" alt=""
class Solution {
private int randRange(Random rand, int min, int max) {
return rand.nextInt(max - min) + min;
}
private int countOccurences(int[] nums, int num) {
int count = 0;
for (int i = 0; i < nums.length; i++) {
if (nums[i] == num) {
count++;
}
}
return count;
}
public int majorityElement(int[] nums) {
Random rand = new Random();
int majorityCount = nums.length / 2;
while (true) {
int candidate = nums[randRange(rand, 0, nums.length)];
if (countOccurences(nums, candidate) > majorityCount) {
return candidate;
}
}
}
}
data:image/s3,"s3://crabby-images/b15a9/b15a94ce24043776b03fa77dbc5d2ab5bcc63064" alt=""
方法四:分治
data:image/s3,"s3://crabby-images/f8066/f80667f139dd2f86d72a0fbdaceca14d9f952f72" alt=""
class Solution {
private int countInRange(int[] nums, int num, int lo, int hi) {
int count = 0;
for (int i = lo; i <= hi; i++) {
if (nums[i] == num) {
count++;
}
}
return count;
}
private int majorityElementRec(int[] nums, int lo, int hi) {
// base case; the only element in an array of size 1 is the majority
// element.
if (lo == hi) {
return nums[lo];
}
// recurse on left and right halves of this slice.
int mid = (hi - lo) / 2 + lo;
int left = majorityElementRec(nums, lo, mid);
int right = majorityElementRec(nums, mid + 1, hi);
// if the two halves agree on the majority element, return it.
if (left == right) {
return left;
}
// otherwise, count each element and return the "winner".
int leftCount = countInRange(nums, left, lo, hi);
int rightCount = countInRange(nums, right, lo, hi);
return leftCount > rightCount ? left : right;
}
public int majorityElement(int[] nums) {
return majorityElementRec(nums, 0, nums.length - 1);
}
}
data:image/s3,"s3://crabby-images/c05ae/c05aead2bb0524f4d234c35b5d9a46fb38ed31a8" alt=""
方法五:Boyer-Moore 投票算法
data:image/s3,"s3://crabby-images/93cce/93ccea141daab7143729283a3308064d94396899" alt=""
data:image/s3,"s3://crabby-images/bb9a8/bb9a87dc968e3a3ca160fa4789d45ad3d894da2f" alt=""
class Solution {
public int majorityElement(int[] nums) {
int count = 0;
Integer candidate = null;
for (int num : nums) {
if (count == 0) {
candidate = num;
}
count += (num == candidate) ? 1 : -1;
}
return candidate;
}
}
data:image/s3,"s3://crabby-images/1ea5a/1ea5a1890af6386cc101673a006b607cbc5a2f99" alt=""
data:image/s3,"s3://crabby-images/a3655/a3655ac8cddf0a62ce9da892455824998a5ff1ac" alt=""
3. 颜色分类
3.1. 题目描述
data:image/s3,"s3://crabby-images/eb808/eb808adb765c474327f952ef84b6a4dc4efcf7f3" alt=""
3.2. 解题思路
方法一:单指针
data:image/s3,"s3://crabby-images/a5ad4/a5ad47a74c7cd92e65957617a1026303897fa741" alt=""
class Solution {
public void sortColors(int[] nums) {
int n = nums.length;
int ptr = 0;
for (int i = 0; i < n; ++i) {
if (nums[i] == 0) {
int temp = nums[i];
nums[i] = nums[ptr];
nums[ptr] = temp;
++ptr;
}
}
for (int i = ptr; i < n; ++i) {
if (nums[i] == 1) {
int temp = nums[i];
nums[i] = nums[ptr];
nums[ptr] = temp;
++ptr;
}
}
}
}
data:image/s3,"s3://crabby-images/d9203/d9203a94d668ea18b9334ee35180b6c4265844a6" alt=""
方法二:双指针
data:image/s3,"s3://crabby-images/bb88a/bb88a6d9986bcd1719012057e9dbc4090022e8d7" alt=""
Java代码:
class Solution {
public void sortColors(int[] nums) {
int n = nums.length;
int p0 = 0, p1 = 0;
for (int i = 0; i < n; ++i) {
if (nums[i] == 1) {
int temp = nums[i];
nums[i] = nums[p1];
nums[p1] = temp;
++p1;
} else if (nums[i] == 0) {
int temp = nums[i];
nums[i] = nums[p0];
nums[p0] = temp;
if (p0 < p1) {
temp = nums[i];
nums[i] = nums[p1];
nums[p1] = temp;
}
++p0;
++p1;
}
}
}
}
data:image/s3,"s3://crabby-images/5f137/5f1373c66f2cd33bca408402ce03a053e514f630" alt=""
方法三:双指针
data:image/s3,"s3://crabby-images/1fa86/1fa861f2bc0decdc1e9dbdcc1c1f8b301aba0e2f" alt=""
Java代码:
class Solution {
public void sortColors(int[] nums) {
int n = nums.length;
int p0 = 0, p2 = n - 1;
for (int i = 0; i <= p2; ++i) {
while (i <= p2 && nums[i] == 2) {
int temp = nums[i];
nums[i] = nums[p2];
nums[p2] = temp;
--p2;
}
if (nums[i] == 0) {
int temp = nums[i];
nums[i] = nums[p0];
nums[p0] = temp;
++p0;
}
}
}
}
data:image/s3,"s3://crabby-images/07dfa/07dfa0b266fe4f4caf8f41102b5148758a8dd55b" alt=""
4. 下一个排列
4.1. 题目描述
data:image/s3,"s3://crabby-images/bcb51/bcb519cab7547f32499eff33b2a499204a5f4f4f" alt=""
4.2. 解题思路
data:image/s3,"s3://crabby-images/d9f36/d9f36f6ad60912f3d41204a8b3b2c44ed34b0794" alt=""
方法一:两遍扫描
data:image/s3,"s3://crabby-images/568d9/568d9c7f4a56183bd65a6664f44d0858081934ef" alt=""
data:image/s3,"s3://crabby-images/d0581/d058128fdabbc46063ae94446e2629d94d6c1aef" alt=""
data:image/s3,"s3://crabby-images/2e7ce/2e7ce8022ef8292212bcddf223612b6390c7dd8e" alt=""
class Solution {
public void nextPermutation(int[] nums) {
int i = nums.length - 2;
while (i >= 0 && nums[i] >= nums[i + 1]) {
i--;
}
if (i >= 0) {
int j = nums.length - 1;
while (j >= 0 && nums[i] >= nums[j]) {
j--;
}
swap(nums, i, j);
}
reverse(nums, i + 1);
}
public void swap(int[] nums, int i, int j) {
int temp = nums[i];
nums[i] = nums[j];
nums[j] = temp;
}
public void reverse(int[] nums, int start) {
int left = start, right = nums.length - 1;
while (left < right) {
swap(nums, left, right);
left++;
right--;
}
}
}
data:image/s3,"s3://crabby-images/0d44f/0d44f57e2962d3997a48797e327dbc26e7377e41" alt=""
5. 寻找重复数
5.1. 题目描述
data:image/s3,"s3://crabby-images/eec34/eec343c07cf8b447b66ad7801d92dc12b8b89ece" alt=""
5.2. 解题思路
方法一:二分查找
data:image/s3,"s3://crabby-images/c623e/c623ef8f9129a48f89c086ae64dd2ee6490d7758" alt=""
class Solution {
public int findDuplicate(int[] nums) {
int n = nums.length;
int l = 1, r = n - 1, ans = -1;
while (l <= r) {
int mid = (l + r) >> 1;
int cnt = 0;
for (int i = 0; i < n; ++i) {
if (nums[i] <= mid) {
cnt++;
}
}
if (cnt <= mid) {
l = mid + 1;
} else {
r = mid - 1;
ans = mid;
}
}
return ans;
}
}
data:image/s3,"s3://crabby-images/2ec0f/2ec0fb9fc8e517ded05fe5028a2344bd0377804c" alt=""
方法二:二进制
data:image/s3,"s3://crabby-images/0c7ee/0c7ee2c5bee869962e4d255ecb2bf62a313b5280" alt=""
class Solution {
public int findDuplicate(int[] nums) {
int n = nums.length, ans = 0;
int bit_max = 31;
while (((n - 1) >> bit_max) == 0) {
bit_max -= 1;
}
for (int bit = 0; bit <= bit_max; ++bit) {
int x = 0, y = 0;
for (int i = 0; i < n; ++i) {
if ((nums[i] & (1 << bit)) != 0) {
x += 1;
}
if (i >= 1 && ((i & (1 << bit)) != 0)) {
y += 1;
}
}
if (x > y) {
ans |= 1 << bit;
}
}
return ans;
}
}
data:image/s3,"s3://crabby-images/79c3d/79c3dd7f1ba520c90b14ac0cf08f57af6ee67d34" alt=""
方法三:快慢指针
data:image/s3,"s3://crabby-images/a1ef3/a1ef38b1f358d3816c399a236122c2847f397716" alt=""
class Solution {
public int findDuplicate(int[] nums) {
int slow = 0, fast = 0;
do {
slow = nums[slow];
fast = nums[nums[fast]];
} while (slow != fast);
slow = 0;
while (slow != fast) {
slow = nums[slow];
fast = nums[fast];
}
return slow;
}
}
data:image/s3,"s3://crabby-images/55092/5509259faacf5ffc50fc29ec264dc78ceaee914d" alt=""