voidmerge_sort(vector<int> &nums, int l, int r, vector<int> &temp){ if (l + 1 >= r) return; int m = (l + r) / 2; //分解 merge_sort(nums, l, m, temp); merge_sort(nums, m, r, temp); //归并原理 int p = l, q = m, i = l; //分别从两个小数组开头开始,先挑小的放进结果数组 while (p < m || q < r) { if (q >= r || (p < m && nums[p] <= nums[q])) temp[i++] = nums[p++]; else temp[i++] = nums[q++]; } //临时的数组赋回原数组 for (i = l; i < r; i++) nums[i] = temp[i]; }
插入排序
先排好小的元素。
1 2 3 4 5 6
voidinsertion_sort(vector<int> &nums, int n){ for (int i = 0; i < n; i++) { for (int j = i; j > 0 && nums[j] < nums[j-1]; j--) swap(nums[j], nums[j-1]); } }
冒泡排序
先排好大的元素。
1 2 3 4 5 6 7 8
voidbubble_sort(vector<int> &nums, int n){ for (int i = 0; i < n-1; i++) { for (int j = 0; j < n-1-i; j++) { if (a[j] < a[j+1]) swap(nums[j], nums[j+1]); } } }
选择排序
选定第一个元素为最小元,然后向后比较找真正的最小元,与之交换。
1 2 3 4 5 6 7 8 9 10 11
voidselection_sort(vector<int> &nums, int n){ int min; for (int i = 0; i < n-1; i++) { min = i; for (int j = i+1; j < n; j++) { if (nums[j] < nums[min]) min = j; } swap(nums[min], nums[i]); } }
intfindKthLargest(vector<int>& nums, int k){ int l = 0, r = nums.size() - 1, target = nums.size() - k; while (l < r) { int mid = quickSelection(nums, l, r); if (mid == target) return nums[mid]; if (mid < target) l = mid + 1; else r = mid - 1; } return nums[l]; } // 辅函数 - 快速选择,目标是把数组按大小切成两半,返回的是分界点 intquickSelection(vector<int>& nums, int l, int r){ int i = l + 1, j = r; while (true) { while (i < r && nums[i] <= nums[l]) i++; while (l < j && nums[j] >= nums[l]) j--; if (i >= j) break; swap(nums[i], nums[j]); } swap(nums[l], nums[j]); return j; }
紧接着,我们对桶的频次进行排序,前 k 大个桶即是前 k 个频繁的数。这里我们可以使用各种 排序算法,甚至可以再进行一次桶排序,把每个旧桶根据频次放在不同的新桶内。针对样例来说,因为目前最大的频次是 4,我们建立 [1,2,3,4] 四个新桶,它们分别放入的旧桶为 [[3,4],[2],[],[1]],表示不同数字出现的频率。最后,我们从后往前遍历,直到找到 k 个旧桶。
代码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
vector<int> topKFrequent(vector<int>& nums, int k){ unordered_map<int, int> counts; int max_count = 0; for (constint & num : nums) { counts[num]++; max_count = max(max_count, counts[num]); } vector<vector<int>> buckets(max_count + 1); for (constauto & p : counts) buckets[p.second].push_back(p.first); vector<int> ans; for (int i = max_count; i >= 0; i--) { for (constint & num : buckets[i]) ans.push_back(num); if (ans.size() == k) break; } return ans; }
给定一个包含红色、白色和蓝色共 n 个元素的数组 nums,原地对它们进行排序,使得相同颜色的元素相邻,并按照红色、白色、蓝色顺序排列。我们使用整数 0、 1 和 2 分别表示红色、白色和蓝色。
必须在不使用库的sort函数的情况下解决这个问题。
示例 1:
输入:nums = [2,0,2,1,1,0] 输出:[0,0,1,1,2,2]
示例 2:
输入:nums = [2,0,1] 输出:[0,1,2]
桶排序。
代码
1 2 3 4 5 6 7 8 9 10 11 12 13 14
voidsortColors(vector<int>& nums){ unordered_map<int, int> counts; int max_count = 0; for (constint & num : nums) counts[num]++; int i = 0; for (i = 0; i < counts[0]; i++) nums[i] = 0; for ( ; i < counts[1] + counts[0]; i++) nums[i] = 1; for ( ; i < counts[2] + counts[1] + counts[0]; i++) nums[i] = 2; return; }