排序算法简单问题(Java)

问题:以尽量快的效率求出一个乱序数组中按数值顺序的第k小的元素。

解法一:先用快排进行排序,然后在找到第k+1个元素,时间复杂度为nlg(n)

解法二:用快排划分为俩个子序列,找到主元素的下标,该下标+1=x即为主元素在该数组的第x小元素,判断k与x的大小缩小范围,时间复杂度期望O(n),最差O(n^2)

下面只说了解法二:

java 复制代码
public class Test2 {
    public static void main(String[] args) {
        int[] arr = new int[]{2,4,5,6,7,1,9,8,3};
        System.out.println(f2(arr,6,0,arr.length-1));
    }

    static int f2(int[] arr,int k,int p,int r){
        int q = partition(arr,p,r);  //此时q前面的数都比q小,后面的数都比q大,所以该数为第q+1大的数
        if(q-p < k-1){
            return f2(arr,k-1-q+p,q+1,r);
        }else if(q-p > k-1){
            return f2(arr,k,p,q-1);
        }else{
            return arr[q];
        }
    }
    //双向扫描,快排
    static int partition(int[] arr,int p,int r){
        int pivot=arr[p];
        int left=p+1;
        int right=r;
        while(left<=right){
            while(left<=right && arr[left]<=pivot) left++;
            while(left<=right && arr[right]>pivot) right--;
            if(left<right)
                swap(arr,left,right);
        }
        swap(arr,p,right);
        return right;
    }
    static void swap(int[] arr,int l,int r) {
        int t = arr[l];
        arr[l] = arr[r];
        arr[r] = t;
    }
}

问题:找出无序数组中出现次数超过一半的元素

方法一: 排序后,中间下标元素就是我们要找的,时间复杂度nlg(n)

方法二: 顺序统计,数组长度为n,求第n/2大的元素,时间复杂度:n

方法三: 数不相同count就减一,count为0就改变cur值,最后一定只有出现次数最多的元素的count>0

但如果不允许挪动数组中元素的位置呢?

方法三:

java 复制代码
public class Test2 {
    public static void main(String[] args) {
        int[] arr = new int[]{2,4,5,6,7,1,9,8,3};
        System.out.println(f2(arr,6,0,arr.length-1));
    }

    static int f3(int[] arr){
        //方法三 数不相同count就减一,count为0就改变cur值,最后一定只有出现次数最多的元素的count>0
        int cur=arr[0],count=1;
        for(int i=1;i<arr.length;i++){
            if(cur != arr[i]){
                count--;
            }else{
                count++;
            }
            if(count==0){
                cur=arr[i];
                count=1;
            }
        }
        return cur;
    }
}

那此题目,在不允许挪动数组中元素的位置的情况下,缩小次数,出现次数最多的元素的出现次数恰好为数组长度的一半呢?

解决:在方法三的基础上加点条件即可:新增无序数组中最后一个元素的计数器

如果最后一位为所求元素,遍历完后,最后一个元素的计数器的值一定为数组长度的一半。

如果最后一位不为所求元素,当要遍历最后一个元素时,

此时方法三中的count值一定为1,因为如果不算最后一个元素的话,满足所求元素出现次数超过数组长度一半,所以此时cur值一定为所求元素。

然后因为最后一个元素的计数器的值不为数组长度的一半,所以cur值即为所求

java 复制代码
public class Test2 {
    public static void main(String[] args) {
        int[] arr = new int[]{2,4,5,6,7,1,9,8,3};
        System.out.println(f2(arr,6,0,arr.length-1));
    }

    static int f4(int[] arr) {
        int cur=arr[0],count=0;
        int countLast=0,lastNum=arr[arr.length-1];
        for(int i=0;i<arr.length;i++) {
            if (cur != arr[i]) {
                count--;
            } else {
                count++;
            }
            if (arr[i] == lastNum) {
                countLast++;
            }
            if (i == arr.length - 1) {
                if (countLast == arr.length / 2) {
                    return lastNum;
                } else {
                    return cur;
                }
            }
            if (count == 0) {
                cur = arr[i];
                count=1;
            }
        }
        return cur;
    }
}

问题:最小可用ID:在非负数组(乱序)中找到最小可分配的id(从1开始编号),数据量1000000。

就是在乱序数组(数组大小不超过1000000)中找出最小的从未出现过的数(1~1000000)。

解法一:开辟一个等大+1的数组helper,遍历一遍原数组,在helper相应位置打上标记,然后遍历一遍helper,时间复杂度2n;

java 复制代码
public class Test2 {
    public static void main(String[] args) {
        int[] arr = new int[]{3,4,5,1,6,7,8,13,9,27,40,20,2};
        System.out.println(f1(arr));
    }

    static int f1(int[] arr){
        int[] helper = new int[arr.length+1];
        for(int i=0;i<arr.length;i++){
            if(arr[i]<=arr.length){ //如果有元素大于数组长度,说明1~arr.length之间一定不连续,所求的数在该范围内。
                helper[arr[i]]++;
            }
        }
        for(int i=1;i<helper.length;i++){
            if(helper[i]!=1)
                return i;
        }
        return helper.length;
    }
}

解法二:不采用辅助空间,采用求第k小元素的方法

选取一个元素作为主元素,判断主元素的值是否等于其下标值,若等于,则所求在右侧,若主元素值大于其下标则所求在左侧,主元素值不可能小于下标值

java 复制代码
public class Test2 {
    public static void main(String[] args) {
        int[] arr = new int[]{3,4,5,1,6,7,8,13,9,27,40,20,2};
        System.out.println(f2(arr,0,arr.length-1));
    }

    static int f2(int[] arr,int p,int r){
        if(p>=r)return p+2; //p为小于所求数的最大数的下标,p+1为所求数的下标,p+2为所求数
        int partition = partition(arr, p, r);
        if(partition +1< arr[partition]){
            return f2(arr,p,partition-1);
        }else {
            return f2(arr,partition+1,r);
        }
    }

    static int partition(int[] arr,int p,int r){
        int pivot=arr[p];
        int left=p+1;
        int right=r;
        while(left<=right){
            while(left<=right && arr[left]<=pivot)left++;
            while(left<=right && arr[right]>pivot)right--;
            if(left<right)
                swap(arr,left,right);
        }
        swap(arr,p,right);
        return right;
    }
    static void swap(int[] arr,int l,int r) {
        int t = arr[l];
        arr[l] = arr[r];
        arr[r] = t;
    }
}

问题:一个数列,如果左边数大,右边数小,则称这俩个数位为一个逆序对,求一个数列中有多少个逆序对

比如:2 5 1 3 4中逆序对有 2、1 ; 5,1 ;5,3 ;5,4

解法:类似于归并数组,把数组前后分为俩部分,然后不断细分,直到剩俩个元素,然后升序排序,然后在对俩部分数组a(前半部分),b(后半部分)中第一个数进行比较,如果b小,那么b数列中的这个元素拥有a数组中元素个数 个逆序对 ,每次比较取走最小的。

java 复制代码
public class Test2 {
    public static void main(String[] args) {
        int[] arr = new int[]{2,6,7,8,9,3,1,4,5};
        helper = new int[arr.length];
        System.out.println(f4(arr));
    }

    static int f4(int[] arr){
        sort1(arr,0,arr.length-1);
        return niXuCount;
    }
    static void sort1(int[] arr,int low,int high){
        if(low<high){
            int mid = low+((high-low)>>1);
            sort1(arr,low,mid);
            sort1(arr,mid+1,high);
            merge1(arr,low,mid,high);
        }
    }
    static int[] helper;
    static int niXuCount=0;
    static void merge1(int[] arr,int low,int mid,int high){
        //把arr中数据拷贝到helper中
        for(int i=low;i<=high;i++){
            helper[i]=arr[i];
        }
        int left =low; //左侧队伍的头指针,指向待比较的元素
        int right =mid+1; //右侧队伍的头指针,指向待比较的元素
        int current=low; //原数组指针,指向待填入元素的位置
        while(left<=mid && right<=high){ //俩个队伍都没走完
            if(helper[left]<=helper[right]){
                arr[current]=helper[left];
                left++;
            }else{
                arr[current]=helper[right];
                right++;

                niXuCount+=mid+1-left;  //此时前半部分数组的大小
            }
            current++;
        }
        while(left<=mid){
            arr[current]=helper[left];
            left++;
            current++;
        }
        while(right<=mid){
            arr[current]=helper[right];
            right++;
            current++;
        }
    }
}
相关推荐
小冯的编程学习之路16 分钟前
【LeetCode】:解数独【困难】
算法·leetcode·职场和发展
DC_BLOG23 分钟前
数据结构排序
java·数据结构·算法·排序算法
G_qingxin33 分钟前
前端排序算法
前端·算法·排序算法
大丈夫立于天地间36 分钟前
OSPF - LSA对照表
网络·网络协议·学习·算法·信息与通信
Vec[95]2 小时前
如何将光源视角的深度贴图应用于摄像机视角的渲染
c++·算法·3d·贴图
23级二本计科2 小时前
排序算法的实现(插入,希尔,选择,冒泡,堆排,快排)
算法·排序算法
8Qi83 小时前
多目标优化算法——基于聚类的不规则Pareto前沿多目标优化自适应进化算法(CA-MOEA)
人工智能·算法·多目标优化·进化计算·群体智能·ca-moea
Swift社区3 小时前
LeetCode - #180 Swift 实现连续数字查询
算法·leetcode·swift
黑客K-ing4 小时前
如何安全保存用户密码及哈希算法
算法·哈希算法
最好Tony4 小时前
深度学习blog-Meanshift均值漂移算法-最大熵模型
深度学习·算法·均值算法