算法通关村-----归并排序

基本原理

归并排序采用分治的思想,即分而治之,分就是将一个大问题分成一些小问题求解,治就是将分得的小问题得到的答案和在一起,得到最终的结果。体现在归并排序上,就是将大的数组分成小的序列,一直分到每个序列中只包含一个元素,此时序列内有序,然后两两合并,合并的方式即是合并两个有序数组,最终序列间有序,即整个数组有序,基本过程如下图所示

代码实现

java 复制代码
public void mergeSort(int[] array, int start, int end, int[] temp) {
	//2.直至每个序列只包含一个元素,停止划分
    if (start >= end) {
        return;
    }
    //1.从中间开始,每次划分为两个序列
    mergeSort(array, start, (start + end) / 2, temp);
    mergeSort(array, (start + end) / 2 + 1, end, temp);
    //3。进行有序数组的合并
    merge(array, start, end, temp);
}

public void merge(int[] array, int start, int end, int[] temp) {
	//找到序列中点
    int mid = (start + end) / 2;
    //left遍历左边的序列
    int left = start;
    //right遍历右边的序列
    int right = mid + 1;
    //index遍历临时数组,存储合并结果
    int index = 0;
    //两个序列均从起点到终点进行遍历
    while (left <= mid && right <= end) {
    	//将两个序列中较小的元素放入临时数组中
        if (array[left] < array[right]) {
            temp[index++] =array[left++];
        }else {
            temp[index++] =array[right++];
        }
    }
    //此时仅剩一个序列未遍历结束,直接赋值
    while (left <= mid){
        temp[index++] =array[left++];
    }
    while (right<=end){
        temp[index++] =array[right++];
    }
    //将归并的结果拷贝到原数组
    for (int i=start;i<=end;i++){
        array[i] = temp[i];
    }
}

例题分析

数组中的第K个最大元素

问题描述

给定整数数组 nums 和整数 k,请返回数组中第 k 个最大的元素。
请注意,你需要找的是数组排序后的第 k 个最大的元素,而不是第 k 个不同的元素。
你必须设计并实现时间复杂度为 O(n) 的算法解决此问题。
详见leetcode215

问题分析

使用归并排序对数组进行从大到小的排序,排序后,返回数组下标为K-1的值,即为数组中第K大元素。

代码实现

java 复制代码
public int findKthLargest(int[] nums, int k) {
    int[] temp = new int[nums.length];
    mergeSort(nums,0,nums.length-1,temp);
    return nums[k-1];
}

public void mergeSort(int[] array, int start, int end, int[] temp) {
    //2.直至每个序列只包含一个元素,停止划分
    if (start >= end) {
        return;
    }
    //1.从中间开始,每次划分为两个序列
    mergeSort(array, start, (start + end) / 2, temp);
    mergeSort(array, (start + end) / 2 + 1, end, temp);
    //3。进行有序数组的合并
    merge(array, start, end, temp);
}

public void merge(int[] array, int start, int end, int[] temp) {
    //找到序列中点
    int mid = (start + end) / 2;
    //left遍历左边的序列
    int left = start;
    //right遍历右边的序列
    int right = mid + 1;
    //index遍历临时数组,存储合并结果
    int index = left;
    //两个序列均从起点到终点进行遍历
    while (left <= mid && right <= end) {
        //将两个序列中较小的元素放入临时数组中
        if (array[left] > array[right]) {
            temp[index++] =array[left++];
        }else {
            temp[index++] =array[right++];
        }
    }
    //此时仅剩一个序列未遍历结束,直接赋值
    while (left <= mid){
        temp[index++] =array[left++];
    }
    while (right<=end){
        temp[index++] =array[right++];
    }
    //将归并的结果拷贝到原数组
    for (int i=start;i<=end;i++){
        array[i] = temp[i];
    }
}
相关推荐
llwszx1 小时前
深入理解Java锁原理(一):偏向锁的设计原理与性能优化
java·spring··偏向锁
云泽野2 小时前
【Java|集合类】list遍历的6种方式
java·python·list
二进制person2 小时前
Java SE--方法的使用
java·开发语言·算法
OneQ6662 小时前
C++讲解---创建日期类
开发语言·c++·算法
JoJo_Way3 小时前
LeetCode三数之和-js题解
javascript·算法·leetcode
小阳拱白菜3 小时前
java异常学习
java
.30-06Springfield3 小时前
人工智能概念之七:集成学习思想(Bagging、Boosting、Stacking)
人工智能·算法·机器学习·集成学习
FrankYoou4 小时前
Jenkins 与 GitLab CI/CD 的核心对比
java·docker
麦兜*5 小时前
Spring Boot启动优化7板斧(延迟初始化、组件扫描精准打击、JVM参数调优):砍掉70%启动时间的魔鬼实践
java·jvm·spring boot·后端·spring·spring cloud·系统架构
KK溜了溜了5 小时前
JAVA-springboot 整合Redis
java·spring boot·redis