• LeetCode算法题-Relative Ranks(Java实现)


    这是悦乐书的第248次更新,第261篇原创

    01 看题和准备

    今天介绍的是LeetCode算法题中Easy级别的第115题(顺位题号是506)。根据N名运动员的得分,找到他们的相对等级和得分最高的三个人,他们将获得奖牌:“金牌”,“银牌”和“铜牌”。例如:

    输入:[5,4,3,2,1]

    输出:["Gold Medal","Silver Medal","Bronze Medal","4","5"]

    说明:前三名运动员获得前三名最高分,因此获得“Gold Medal”,“Silver Medal”和“Bronze Medal”。对于剩下的两名运动员,你只需要根据他们的分数输出他们的相对等级。

    注意:

    • N是正整数,不超过10,000。

    • 所有运动员的得分都保证是独一无二的。

    本次解题使用的开发工具是eclipse,jdk使用的版本是1.8,环境是win7 64位系统,使用Java语言编写和测试。

    02 第一种解法

    这道题除了要排名外,还需要保持原来元素所在的位置不变,也就要求我们要把原数组中各元素的位置先保存起来。对此,我们可以使用HashMap来保存,key为元素值,value为其坐标。然后将数组排序,因为排序后是升序,所以需要我们反过来,从后往前开始排名,而新数组中的索引即为当前元素在map中的value值。

    此解法的空间复杂度是O(n),时间复杂度是O(n log(n))。

    public String[] findRelativeRanks(int[] nums) {
        String[] arr = new String[nums.length];
        Map<Integer,Integer> map = new HashMap<Integer,Integer>();
        for (int i=0; i<nums.length; i++) {
            map.put(nums[i], i);
        }
        Arrays.sort(nums);
        for (int i=nums.length-1; i >= 0; i--) {
            if (i == nums.length-1) {
                arr[map.get(nums[i])] = "Gold Medal";
            } else if (i == nums.length-2) {
                arr[map.get(nums[i])] = "Silver Medal";
            } else if (i == nums.length-3) {
                arr[map.get(nums[i])] = "Bronze Medal";
            } else {
                arr[map.get(nums[i])] = (nums.length-i)+"";
            }
        }
        return arr;
    }
    

    03 第二种解法

    对于上面的解法,我们可以使用一个新的数组来代替HashMap,将原来的数组复制一份出来,然后对复制出来的数组进行排序,然后遍历原数组的元素,使用二分法查找当前元素在复制数组中的索引,再使用数组长度减1后再减去该索引,得到当前元素的排名,然后按照金银铜以及名次的顺序依次写入字符串结果数组中去,并返回。

    此解法的时间复杂度是O(n log2n),空间复杂度是O(n)。

    public String[] findRelativeRanks2(int[] nums) {
        String[] arr = new String[nums.length];
        int[] copy = nums.clone();
        Arrays.sort(copy);
        for (int i=0; i< nums.length; i++) {
            int rank = nums.length - 1 - Arrays.binarySearch(copy, nums[i]);
            if (rank == 0) {
                arr[i] = "Gold Medal";
            } else if (rank == 1) {
                arr[i] = "Silver Medal";
            } else if (rank == 2) {
                arr[i] = "Bronze Medal";
            } else {
                arr[i] = (rank+1)+"";
            }
        }
        return arr;
    }
    

    04 第三种解法

    我们也可以不对原数组进行排序,而使用一个新数组来保存原数组中各元素的索引值。

    先遍历原数组找到最大元素的值,然后创建一个整型包装类数组,长度为最大值加1,然后再遍历一次原数组,新数组使用原数组的元素值作为索引,值为从0开始递增的整数。然后遍历新数组,因为新数组的长度是最大值加1,所以第一名肯定是新数组的最后一位元素。

    因此我们从后往前开始遍历,同时还要借助一个新索引变量,作为结果数组的递增索引,如果当前元素不为null,并且新索引等于0,1,2,那么就按照金银铜的顺序写入结果数组,剩下的将新索引加1然后转为字符串写入结果数组即可。最后返回结果数组。

    至于当中使用包装类来做数组(其原始值为null),是因为可以避免索引0的冲突,当然使用int(其原始值为0)也可以,但是需要在原索引的基础上加1,然后循环判断的时候就需要判断是否不等于0,在找到值后还需要再减1,还原成原来的索引值,而使用包装类就没这么麻烦。

    此解法的时间复杂度是O(n),其中n为数组中最大元素的值,而非元素个数,因为使用了新数组,新数组的长度是数组中最大元素加1, 空间复杂度是O(n)。

    public String[] findRelativeRanks3(int[] nums) {
        String[] arr = new String[nums.length];
        int max = -1;
        for (int num : nums) {
            if (num > max) {
                max = num;
            }
        }
        Integer[] numIndex = new Integer[max+1];
        for (int i=0; i<nums.length; i++) {
            numIndex[nums[i]] = i;
        }
        int index = 0;
        for (int i=numIndex.length-1; i>=0; i--) {
            if (numIndex[i] != null) {
                if (index == 0) {
                    arr[numIndex[i]] = "Gold Medal";
                } else if (index == 1) {
                    arr[numIndex[i]] = "Silver Medal";
                } else if (index == 2) {
                    arr[numIndex[i]] = "Bronze Medal";
                } else {
                    arr[numIndex[i]] = (index+1)+"";
                }
                index++;
            }
        }
        return arr;
    }
    

    05 小结

    算法专题目前已日更超过三个月,算法题文章115+篇,公众号对话框回复【数据结构与算法】、【算法】、【数据结构】中的任一关键词,获取系列文章合集。

    以上就是全部内容,如果大家有什么好的解法思路、建议或者其他问题,可以下方留言交流,点赞、留言、转发就是对我最大的回报和支持!

  • 相关阅读:
    grpc源码分析之域名解析
    使用swig工具为go语言与c++进行交互
    使用VS2015编译grpc_1.3.1
    在linux中编译grpc
    win32网络模型之重叠I/O
    滚动动画animate-scroll扩展
    Android Studio Gradle更新的解决办法
    MUI——页面的创建、显示、关闭
    VS2013中使用QT插件后每次重新编译问题
    Oracle Rman恢复
  • 原文地址:https://www.cnblogs.com/xiaochuan94/p/10367976.html
Copyright © 2020-2023  润新知