0
点赞
收藏
分享

微信扫一扫

LeetCode 540. 有序数组中的单一元素(二分+异或技巧) / 1380. 矩阵中的幸运数 / 1719. 重构一棵树的方案数

大自然在召唤 2022-02-16 阅读 25
leetcodejava

540. 有序数组中的单一元素

2022.2.14 每日一题

题目描述

给你一个仅由整数组成的有序数组,其中每个元素都会出现两次,唯有一个数只会出现一次。

请你找出并返回只出现一次的那个数。

你设计的解决方案必须满足 O(log n) 时间复杂度和 O(1) 空间复杂度。

示例 1:

示例 2:

提示:

思路

一个二分思路,

class Solution {
    public int singleNonDuplicate(int[] nums) {
        //异或是On
        //对数复杂度的话二分,怎么二分了
        //有奇数个数,如果当前查找的数是第偶数个,且前面一个数和当前数相同,那么说明前面都是成对的
        //如果前面一个数和当前数不同,那么就说明前面的数有单独的
        //如果当前个数是奇数个,那么也看前面一个数,如果和前面的相同,那么说明前面也不对了

        int n = nums.length;
        int left = 0;
        int right = n - 1;
        while(left < right){
            int mid = (right - left) / 2 + left;
            //如果当前个数是偶数
            if((mid + 1) % 2 == 0){
                //如果和前面的相同,那么说明前面没有单独的数
                if(mid > 0 && (nums[mid] == nums[mid - 1])){
                    left = mid + 1;
                }else{
                    right = mid - 1;
                }
            }else{
                if(mid > 0 && (nums[mid] == nums[mid - 1])){
                    right = mid - 2;
                }else{
                    left = mid;
                }
            }
        }
        return nums[left];
    }
}

那么这道题主要学习的是什么呢,就是官解的这个异或
如果是奇数,那么异或就找的是前一个数;如果是偶数,异或找的就是后一个数
所以可以把两个逻辑合并到一个逻辑里

如果当前下标是偶数,那么和后一个数比,如果相同,说明前面没问题
如果当前下标是奇数,那么和前一个数比,如果相同,也说没前面没问题
可以写到一起

class Solution {
    public int singleNonDuplicate(int[] nums) {
        int n = nums.length;
        int l = 0, r = n - 1;
        while (l < r) {
            int mid = l + r >> 1;
            if (nums[mid] == nums[mid ^ 1]) l = mid + 1;
            else r = mid;
        }
        return nums[r];
    }
}


作者:AC_OIer
链接:https://leetcode-cn.com/problems/single-element-in-a-sorted-array/solution/gong-shui-san-xie-er-duan-xing-fen-xi-yu-17nv/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

1380. 矩阵中的幸运数

2022.2.15 每日一题

题目描述

给你一个 m * n 的矩阵,矩阵中的数字 各不相同 。请你按 任意 顺序返回矩阵中的所有幸运数。

幸运数是指矩阵中满足同时下列两个条件的元素:

示例 1:

示例 2:

示例 3:

提示:

思路

模拟

class Solution {
    public List<Integer> luckyNumbers (int[][] matrix) {
        int m = matrix.length;
        int n = matrix[0].length;

        int[] hang = new int[m];

        int[] lie = new int[n];
        for(int i = 0; i < m; i++){
            for(int j = 0; j < n; j++){
                if(matrix[i][j] < matrix[i][hang[i]]){
                    hang[i] = j;
                }
                if(matrix[i][j] > matrix[lie[j]][j])
                    lie[j] = i;
            }
        }

        List<Integer> res = new ArrayList<>();
        for(int i = 0; i < m; i++){
            if(lie[hang[i]] == i)
                res.add(matrix[i][hang[i]]);
        }
        return res;
    }
}

1719. 重构一棵树的方案数

2022.2.16 每日一题

题目描述

给你一个数组 pairs ,其中 pairs[i] = [xi, yi] ,并且满足:

pairs 中没有重复元素
xi < yi

令 ways 为满足下面条件的有根树的方案数:

树所包含的所有节点值都在 pairs 中。
一个数对 [xi, yi] 出现在 pairs 中 当且仅当 xi 是 yi 的祖先或者 yi 是 xi 的祖先。
注意:构造出来的树不一定是二叉树。

两棵树被视为不同的方案当存在至少一个节点在两棵树中有不同的父节点。

请你返回:

如果 ways == 0 ,返回 0 。
如果 ways == 1 ,返回 1 。
如果 ways > 1 ,返回 2 。

一棵 有根树 指的是只有一个根节点的树,所有边都是从根往外的方向。

我们称从根到一个节点路径上的任意一个节点(除去节点本身)都是该节点的 祖先 。根节点没有祖先。

示例 1:

示例 2:

示例 3:

提示:

思路

说实话,真的难
看了一下题解,写的很详细,仔细看还是可以看懂的,但是在家也不想仔细研究了,cv了

class Solution {
    public int checkWays(int[][] pairs) {
        //根据第三个例子可以理解到一个题意,就是所有的关系都会在pairs数组中给出
        //在第三个例子中,因为所构造的树中的关系,pairs中并没有全部给出,所以不能形成一棵树
        //换句话也就是说,pairs中的关系不能形成一棵树

        Map<Integer, Set<Integer>> adj = new HashMap<Integer, Set<Integer>>();
        for (int[] p : pairs) {
            adj.putIfAbsent(p[0], new HashSet<Integer>());
            adj.putIfAbsent(p[1], new HashSet<Integer>());
            adj.get(p[0]).add(p[1]);
            adj.get(p[1]).add(p[0]);
        }
        /* 检测是否存在根节点*/
        int root = -1;
        Set<Map.Entry<Integer, Set<Integer>>> entries = adj.entrySet();
        for (Map.Entry<Integer, Set<Integer>> entry : entries) {
            int node = entry.getKey();
            Set<Integer> neighbours = entry.getValue();
            if (neighbours.size() == adj.size() - 1) {
                root = node;
            }
        }
        if (root == -1) {
            return 0;
        }

        int res = 1;
        for (Map.Entry<Integer, Set<Integer>> entry : entries) {
            int node = entry.getKey();
            Set<Integer> neighbours = entry.getValue();
            if (node == root) {
                continue;
            }
            int currDegree = neighbours.size();
            int parent = -1;
            int parentDegree = Integer.MAX_VALUE;

            /* 根据 degree 的大小找到 node 的父节点 parent */
            for (int neighbour : neighbours) {
                if (adj.get(neighbour).size() < parentDegree && adj.get(neighbour).size() >= currDegree) {
                    parent = neighbour;
                    parentDegree = adj.get(neighbour).size();
                }
            }
            if (parent == -1) {
                return 0;
            }

            /* 检测 neighbours 是否是 adj[parent] 的子集 */
            for (int neighbour : neighbours) {
                if (neighbour == parent) {
                    continue;
                }
                if (!adj.get(parent).contains(neighbour)) {
                    return 0;
                }
            }
            if (parentDegree == currDegree) {
                res = 2;
            }
        }
        return res;

    }
}
举报

相关推荐

0 条评论