【Leetcode-每日一题】跳跃游戏 IV

跳跃游戏 IV
难度:困难
在这里插入图片描述
在这里插入图片描述
今天的题很容易理解,但我依然做不出来/(ㄒoㄒ)/~~
看了官方题解,又debug跑了一遍,然后自己写出来也算完成任务了吧
贴一下官方思路
方法一:广度优先搜索
思路
记数组 arr 的长度为 n。题目描述的数组可以抽象为一个无向图,数组元素为图的顶点,相邻下标的元素之间有一条无向边相连,所有值相同元素之间也有无向边相连。每条边的权重都为 1,即此图为无权图。求从第一个元素到最后一个元素的最少操作数,即求从第一个元素到最后一个元素的最短路径长度。求无权图两点间的最短路可以用广度优先搜索来解,时间复杂度为 O(V+E),其中 V 为图的顶点数,E 为图的边数。

在此题中,V = n,而 E 可达 O(n^2)数量级,按照常规方法使用广度优先搜索会超时。造成超时的主要原因是所有值相同的元素构成了一个稠密子图,普通的广度优先搜索方法会对这个稠密子图中的所有边都访问一次。但对于无权图的最短路问题,这样的访问是不必要的。在第一次访问到这个子图中的某个节点时,即会将这个子图的所有其他未在队列中的节点都放入队列。在第二次访问到这个子图中的节点时,就不需要去考虑这个子图中的其他节点了,因为所有其他节点都已经在队列中或者已经被访问过了。因此,在用广度优先搜索解决此题时,先需要找出所有的值相同的子图,用一个哈希表 idxSameValue 保存。在第一次把这个子图的所有节点放入队列后,把该子图清空,就不会重复访问该子图的其他边了。

代码如下:

	public int minJumps(int[] arr) {
        //key为数组arr中的值,value为该值出现的下标
        Map<Integer,List<Integer>> idxSameValue = new HashMap<>();
        for (int i = 0; i < arr.length; i++) {
            idxSameValue.putIfAbsent(arr[i],new ArrayList<>());
            idxSameValue.get(arr[i]).add(i);
        }
        //二维数组第一位记录下标,第二位记录到达改下标所用步数
        Deque<int[]> deque = new ArrayDeque<>();
        deque.offer(new int[]{0,0});
        //set记录访问过的下标
        Set<Integer> visitSet = new HashSet<>();
        visitSet.add(0);
        while(!deque.isEmpty()){
            int[] poll = deque.poll();
            int idx = poll[0];
            int step = poll[1];
            if (idx == arr.length-1){
                return step;
            }
            step++;
            int v = arr[idx];
            if (idxSameValue.containsKey(v)){
                for (int i : idxSameValue.get(v)){
                    if (visitSet.add(i)){
                        deque.offer(new int[]{i,step});
                    }
                }
            }
            idxSameValue.remove(v);
            if (idx+1<arr.length && visitSet.add(idx+1)){
                deque.offer(new int[]{idx+1,step});
            }
            if (idx-1>=0 && visitSet.add(idx-1)){
                deque.offer(new int[]{idx-1,step});
            }
        }
        return -1;
    }

执行结果:成功
在这里插入图片描述
困难题AC就好AC就好

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值