刷题路线参考:https://github.com/chefyuan/algorithm-base

大家好,我是拿输出博客督促自己刷题的老三,前面学习了十大排序:万字长文|十大基本排序,一次搞定!,接下来我们看看力扣上有没有什么能拿排序解决的题目吧!

排序基础

简单了解一下基本的排序——

基本排序分类:

基本排序性能:

排序方法 时间复杂度(平均) 时间复杂度(最坏) 时间复杂度(最好) 空间复杂度 稳定性
冒泡排序 O(n²) O(n²) O(n) O(1) 稳定
选择排序 O(n²) O(n²) O(n²) O(1) 不稳定
插入排序 O(n²) O(n²) O(n) O(1) 稳定
希尔排序 O(n^(1.3-2)) O(n²) O(n) O(1) 不稳定
归并排序 O(nlogn) O(nlogn) O(nlogn) O(n) 稳定
快速排序 O(nlogn) O(n²) O(nlogn) O(nlogn) 不稳定
堆排序 O(nlogn) O(nlogn) O(nlogn) O(1) 不稳定
计数排序 O(n+k) O(n+k) O(n+k) O(n) 稳定
桶排序 O(n+k) O(n²) O(n) O(n+k) 稳定
基数排序 O(n*k) O(n*k) O(n*k) O(n+k) 稳定

更具体的可以查看:万字长文|十大基本排序,一次搞定!

好了,开始我们愉快的刷题之旅吧!

刷题现场

LeetCode912. 排序数组

题目:912. 排序数组 (https://leetcode-cn.com/problems/sort-an-array/)

难度:中等

描述:给你一个整数数组 nums,请你将该数组升序排列。

示例 1:

输入:nums = [5,2,3,1]
输出:[1,2,3,5]

示例 2:

输入:nums = [5,1,1,2,0,0]
输出:[0,0,1,1,2,5]

思路:

这道题如果用api,一行就搞定了——Arrays.sort(nums),那面试官的反应多半是,门在那边,慢走不送。

所以,毫无疑问,我们要手撕排序了。

如果对排序算法不太熟,可以上一个冒泡排序,但是这个明显只能说中规中矩,所以,我们选择:

手撕快排

关于快排,就不多讲。

直接上代码:

class Solution {
public int[] sortArray(int[] nums) {
quickSort(nums,0,nums.length-1);
return nums;
} public void quickSort(int[] nums,int left, int right){
//结束条件
if(left>=right){
return;
}
//分区
int partitionIndex=partition(nums,left,right);
//递归左分区
quickSort(nums,left,partitionIndex-1);
//递归右分区
quickSort(nums,partitionIndex+1,right);
} public int partition(int[] nums,int left,int right){
//基准值
int pivot=nums[left];
//mark标记下标
int mark=left;
for(int i=left+1;i<=right;i++){
if(nums[i]<pivot){
//小于基准值,则mark后移,并交换位置
mark++;
int temp=nums[mark];
nums[mark]=nums[i];
nums[i]=temp;
}
}
//把基准值放到mark的位置
nums[left]=nums[mark];
nums[mark]=pivot;
return mark;
}
}
  • 时间复杂度:快排时间复杂度O(nlogn)

有时间的可以把十大排序都在这道题练上一练。

LeetCode347. 前 K 个高频元素

题目:347. 前 K 个高频元素(https://leetcode-cn.com/problems/top-k-frequent-elements/)

难度:中等

描述:

给你一个整数数组 nums,请你将该数组升序排列。

示例 1:

输入: nums = [1,1,1,2,2,3], k = 2
输出: [1,2]

示例 2:

输入: nums = [1], k = 1
输出: [1]

提示:

1 <= nums.length <= 105
k 的取值范围是 [1, 数组中不相同的元素的个数]
题目数据保证答案唯一,换句话说,数组中前 k 个高频元素的集合是唯一的

进阶:你所设计算法的时间复杂度 必须 优于 O(n log n) ,其中 n 是数组大小。

思路:

这道题第一思路是什么呢?

统计元素出现频率,从大到小排序,取前k个元素。

我们想挑战一下进阶要求,时间复杂度优于O(nlogn),所以熟悉的冒泡、快排之类的比较类排序都不可用,只能使用非比较类的三种排序方法:计数排序、桶排序、基数排序。

这里我们选择HashMap+桶排序的方式。

使用HashMap存储元素出现频率,使用桶排序来进行排序。

代码如下:

    public int[] topKFrequent(int[] nums, int k) {
//使用HashMap存储元素出现频率
Map<Integer, Integer> map = new HashMap<>();
for (int num : nums) {
map.put(num, map.getOrDefault(num, 0) + 1);
}
//桶
List<Integer>[] buckets = new List[nums.length + 1];
//往桶里添加元素出现次数
for (Integer key : map.keySet()) {
//根据出现频率决定元素入哪个桶
int count = map.get(key);
//初始化桶
if (buckets[count] == null) buckets[count] = new ArrayList<>();
//将元素存到桶中
buckets[count].add(key);
}
//结果列表
List<Integer> result = new ArrayList<>();
//取倒数k个非空桶中的元素
for (int i = buckets.length - 1; k > 0; i--) {
if (buckets[i] != null) {
//取出桶中的元素
for (Integer num : buckets[i]) {
result.add(num);
k--;
}
}
}
//将列表中的元素赋给数组
int[] res = new int[result.size()];
for (int i = 0; i < res.length; i++) {
res[i] = result.get(i);
}
return res;
}
  • 时间复杂度:这道题用了桶排序,时间复杂度O(n)。

剑指 Offer 45. 把数组排成最小的数

题目:剑指 Offer 45. 把数组排成最小的数 (https://leetcode-cn.com/problems/ba-shu-zu-pai-cheng-zui-xiao-de-shu-lcof/)

难度:中等

描述:

输入一个非负整数数组,把数组里所有数字拼接起来排成一个数,打印能拼接出的所有数字中最小的一个。

示例 1:

输入: [10,2]
输出: "102"

示例 2:

输入: [3,30,34,5,9]
输出: "3033459"

思路:

稍微分析一下这道题,发现这道题其实也是一道排序题。

只要我们把数组里的元素按照某种规则进行排序。

现在的问题就是这个排序规则是什么呢?

因为需要拼接字符串,以[3,30]为例,“3”+“30”=“330”,“30”+"3"="303",330>303,那么我们就可以说3大于30。

所以定义规则:

  • 若拼接字符串 x+y>y+x ,则 x 大于y ;
  • 反之,若拼接字符串 x+y<y+x ,则 x 小于 y ;

规则图如下(来源参考[2):

那么,这道题我们就知道怎么写了。

用我们自定义的排序规则从小到大排序数组。

排序方法我们选择快排,所以这道题就是自定义排序+快排

代码如下:

    public String minNumber(int[] nums) {
quickSort(nums, 0, nums.length - 1);
//结果
StringBuilder sb = new StringBuilder();
for (int num : nums) {
sb.append(String.valueOf(num));
}
return sb.toString();
} //快排
public void quickSort(int[] nums, int left, int right) {
if (left >= right) return;
int partionIndex = partion(nums, left, right);
quickSort(nums, left, partionIndex - 1);
quickSort(nums, partionIndex + 1, right);
} public int partion(int[] nums, int left, int right) {
int pivot = nums[left];
int mark = left;
for (int i = left + 1; i <= right; i++) {
if (lessThan(nums[i], pivot)) {
mark++;
int temp = nums[mark];
nums[mark] = nums[i];
nums[i] = temp;
}
}
nums[left] = nums[mark];
nums[mark] = pivot;
return mark;
} //自定义大小比较规则
public boolean lessThan(int x, int y) {
String sx = String.valueOf(x), sy = String.valueOf(y);
return (sx + sy).compareTo(sy + sx) < 0;
}

写的比较臃肿,但比较清晰。

有一种利用内置排序来实现的写法,不太建议:

    public String minNumber(int[] nums) {
String[] strs = new String[nums.length];
for(int i = 0; i < nums.length; i++){
strs[i] = String.valueOf(nums[i]);
}
Arrays.sort(strs, (x,y) -> (x+y).compareTo(y+x));
StringBuilder ans = new StringBuilder();
for(String s : strs)
ans.append(s);
return ans.toString();
}
  • 时间复杂度:O(nlogn)。

有一道题:179. 最大数 和这道题基本一样。

剑指 Offer 51. 数组中的逆序对

题目:剑指 Offer 51. 数组中的逆序对 (https://leetcode-cn.com/problems/shu-zu-zhong-de-ni-xu-dui-lcof/)

难度:困难

描述:

在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对。输入一个数组,求出这个数组中的逆序对的总数。

示例 1:

输入: [7,5,6,4]
输出: 5

思路:

这一道题是困难,有没有被吓住?

其实这道题如果用归并排序的思路去解决的话,就没有想象中的那么难。

归并排序这里就不讲了。

解决这道题,我们只需要在归并排序的基础上,加上对逆序对的统计:

归并+逆序对统计示意图(图片来源参考[3]):

现在的关键点是,归并的过程如何计算逆序对个数?

我们可以看一下,合并的时候,l指向左子数组2的位置,r指向右子数组0的位置,num[l]>nums[r],因为子数组是有序的,所以l后面几个元素也都一定大于0,所以可以得出,此时逆序对数量=mid-l+1。

代码如下:

class Solution {
//统计逆序对
int count = 0; public int reversePairs(int[] nums) {
mergeSort(nums, 0, nums.length - 1);
return count;
} //归并排序
public void mergeSort(int[] nums, int left, int right) {
//结束
if (left >= right) return;
int mid = left + (right - left) / 2;
//左半部分
mergeSort(nums, left, mid);
//右半部分
mergeSort(nums, mid + 1, right);
//合并
merge(nums, left, mid, right);
} //合并
public void merge(int[] arr, int left, int mid, int right) {
//临时数组
int[] tempArr = new int[right - left + 1];
//指向左右子数组指针
int l = left, r = mid + 1;
int index = 0;
//把左右子数组较小元素放入到临时数组
while (l <= mid && r <= right) {
if (arr[l] <= arr[r]) {
tempArr[index++] = arr[l++];
} else {
//增加一行,统计逆序对
count += (mid - l + 1);
tempArr[index++] = arr[r++];
}
}
//将左子数组剩余的元素拷贝到临时数组
while (l <= mid) {
tempArr[index++] = arr[l++];
}
//将右边子数组剩余的元素拷贝到临时数组
while (r <= right) {
tempArr[index++] = arr[r++];
}
//将临时数组的元素拷贝给原数组
for (int i = 0; i < tempArr.length; i++) {
arr[i + left] = tempArr[i];
}
}
}
  • 时间复杂度:归并排序时间复杂度O(nlogn)。

LeetCode147. 对链表进行插入排序

题目:剑指 Offer 51. 数组中的逆序对 (https://leetcode-cn.com/problems/shu-zu-zhong-de-ni-xu-dui-lcof/)

难度:困难

描述:

对链表进行插入排序。

插入排序的动画演示如上。从第一个元素开始,该链表可以被认为已经部分排序(用黑色表示)。

每次迭代时,从输入数据中移除一个元素(用红色表示),并原地将其插入到已排好序的链表中。

插入排序算法:

  1. 插入排序是迭代的,每次只移动一个元素,直到所有元素可以形成一个有序的输出列表。
  2. 每次迭代中,插入排序只从输入数据中移除一个待排序的元素,找到它在序列中适当的位置,并将其插入。
  3. 重复直到所有输入数据插入完为止。

示例 1:

输入: 4->2->1->3
输出: 1->2->3->4

示例 2:

输入: -1->5->3->4->0
输出: -1->0->3->4->5

思路:

这道题不只是插入排序,还涉及到链表的操作,关于链表,可以查看:LeetCode通关:听说链表是门槛,这就抬脚跨门而入

  • 关于插入排序:我们需要从未排序序列里将元素插入到排序序列的合适位置

  • 关于链表插入:链表插入是插入节点前驱节点改变后继的一个操作,为了头插也能统一,通常我们会加一个虚拟头节点

  • 所以,综合起来,我们需要标记有序序列和无序序列的分界点,遍历无序序列的时候,记录前驱,当需要将无序序列插入到有序序列的时候,遍历有序序列,找到插入位置,先删除该节点,再插入

代码如下:

    public ListNode insertionSortList(ListNode head) {
if (head == null && head.next == null) {
return head;
}
//虚拟头节点
ListNode dummy = new ListNode(-1);
dummy.next = head;
//记录有序序列终点
ListNode last = head;
//遍历无序序列
ListNode after = head.next;
while (after != null) {
if (last.val <= after.val) {
after = after.next;
last = last.next;
continue;
}
//遍历有序序列,查找插入位置
ListNode prev = dummy;
while (prev.next.val <= after.val) {
prev = prev.next;
}
//找到插入位置
//删除无序序列节点
last.next = after.next;
//插入有序序列
after.next = prev.next;
prev.next = after;
//继续移动
after=last.next;
}
return dummy.next;
}
  • 时间复杂度:O(n²)。

总结

熟悉的顺口溜总结:

简单的事情重复做,重复的事情认真做,认真的事情有创造性地做。

我是三分恶,一个追求实力,正在努力的程序员。

点赞关注不迷路,咱们下期见!


参考:

[1]. https://github.com/chefyuan/algorithm-base

[2]. https://leetcode-cn.com/problems/ba-shu-zu-pai-cheng-zui-xiao-de-shu-lcof/solution/mian-shi-ti-45-ba-shu-zu-pai-cheng-zui-xiao-de-s-4/

[3]. https://leetcode-cn.com/problems/shu-zu-zhong-de-ni-xu-dui-lcof/solution/jian-zhi-offer-51-shu-zu-zhong-de-ni-xu-pvn2h/

LeetCode通关:通过排序一次秒杀五道题,舒服!的更多相关文章

  1. 每天一道面试题LeetCode 80--删除排序数组中的重复项 II(python实现)

    LeetCode 80--删除排序数组中的重复项 II 给定一个排序数组,你需要在原地删除重复出现的元素,使得每个元素最多出现两次,返回移除后数组的新长度. 不要使用额外的数组空间,你必须在原地修改输 ...

  2. LeetCode 80. 删除排序数组中的重复项 II

    LeetCode 80. 删除排序数组中的重复项 II

  3. LeetCode:删除排序链表中的重复元素【83】

    LeetCode:删除排序链表中的重复元素[83] 题目描述 给定一个排序链表,删除所有重复的元素,使得每个元素只出现一次. 示例 1: 输入: 1->1->2 输出: 1->2 示 ...

  4. LeetCode初级算法--排序和搜索01:第一个错误的版本

    LeetCode初级算法--排序和搜索01:第一个错误的版本 搜索微信公众号:'AI-ming3526'或者'计算机视觉这件小事' 获取更多算法.机器学习干货 csdn:https://blog.cs ...

  5. 前端与算法 leetcode 26. 删除排序数组中的重复项

    目录 # 前端与算法 leetcode 26. 删除排序数组中的重复项 题目描述 概要 提示 解析 算法 # 前端与算法 leetcode 26. 删除排序数组中的重复项 题目描述 26. 删除排序数 ...

  6. 力扣(LeetCode)删除排序链表中的重复元素II 个人题解

    给定一个排序链表,删除所有含有重复数字的节点,只保留原始链表中 没有重复出现 的数字. 思路和上一题类似(参考 力扣(LeetCode)删除排序链表中的重复元素 个人题解)) 只不过这里需要用到一个前 ...

  7. LeetCode通关:哈希表六连,这个还真有点简单

    精品刷题路线参考: https://github.com/youngyangyang04/leetcode-master https://github.com/chefyuan/algorithm-b ...

  8. 领扣(LeetCode)删除排序数组中的重复项 个人题解

    给定一个排序数组,你需要在原地删除重复出现的元素,使得每个元素只出现一次,返回移除后数组的新长度. 不要使用额外的数组空间,你必须在原地修改输入数组并在使用 O(1) 额外空间的条件下完成. 示例 1 ...

  9. 【LeetCode】删除排序数组中的重复项&&移除特定元素【双指针,原地算法】

    给定一个排序数组,你需要在原地删除重复出现的元素,使得每个元素只出现一次,返回移除后数组的新长度. 不要使用额外的数组空间,你必须在原地修改输入数组并在使用 O(1) 额外空间的条件下完成. 示例 1 ...

随机推荐

  1. SpringMVC 参数中接收之一 List

    作者:张艳涛 time:2020-07-31 SpingMVC 一.前台传数组,SpingMVC用addusers(@RequestBody List<UserPojo> userlist ...

  2. Kubernetes的基本概念和术语

    Kubernetes中的大部分概念如Node.Pod.Replication Controller. Service等都可以被看作一种资源对象,几乎所有资源对象都可以通过 Kubernetes提供的k ...

  3. SQL语句(四)联表查询

    目录 一.关联查询的分类 按年代分 按功能分 二.sql92语法的连接 语法 1. 简单应用 2. 为表起别名 3. 加入筛选 4. 加入分组 5. 三表连接 6. 非等值连接 7. 自连接 三.sq ...

  4. fastboot刷机小脚本

    在Windows系统下,一般刷机命令是在cmd路径下执行如下命令: 1.adb reboot bootloader2.fastboot flash boot +boot路径3.fastboot fla ...

  5. Solon 1.5.22 发布

    Solon 是一个轻量的Java基础开发框架.强调,克制 + 简洁 + 开放的原则:力求,更小.更快.更自由的体验.支持:RPC.REST API.MVC.Job.Micro service.WebS ...

  6. 解决win10快速访问不能取消固定

    最近发现win10的快速访问不能取消固定,比如ftp和smb之类的都不能取消固定 最后百度了一下发现一个简易的方法: 在文件资源管理器地址栏输入:%APPDATA%\Microsoft\Windows ...

  7. STP进阶版MSTP

    一.MSTP简介 1.1.MSTP工作原理 mstp是一个公有生成树协议,在实际生产环境中得到了广泛的应用.传统的生成树只运行一个实例,且收敛速度慢,RSTP在传统的STP基础上通过改进达到了加速网络 ...

  8. 【笔记】scikit-learn中的PCA(真实数据集)

    sklearn中的PCA(真实的数据集) (在notebook中) 加载好需要的内容,手写数字数据集 import numpy as np import matplotlib.pyplot as pl ...

  9. JunAMS v1.2.1.20190403代码审计笔记

    前言 CNVD-2020-24741 过程 JunAMS是以ThinkPHP为框架的开源内容管理系统,本地搭建受影响版本JunAMS v1.2.1.20190403 前台没有上传功能,进入后台.发现在 ...

  10. hdmi 随笔

    从图片来看,每张图片开始传输的是45像素的垂直同步, 1.控制数据贯穿所有时间,没个不是控制数据的传输都被控制数据包围.控制数据还要通过控制位指示,下一个数据是数据岛还是视频信号. 2.terc4 全 ...