LeetCode150--子集(L78)、单词搜索(L79)

1、子集

//给你一个整数数组 nums ,数组中的元素 互不相同 。返回该数组所有可能的子集(幂集)。 
//
// 解集 不能 包含重复的子集。你可以按 任意顺序 返回解集。 
//
// 
//
// 示例 1: 
//
// 
//输入:nums = [1,2,3]
//输出:[[],[1],[2],[1,2],[3],[1,3],[2,3],[1,2,3]]
// 
//
// 示例 2: 
//
// 
//输入:nums = [0]
//输出:[[],[0]]
// 
//
// 
//
// 提示: 
//
// 
// 1 <= nums.length <= 10 
// -10 <= nums[i] <= 10 
// nums 中的所有元素 互不相同 
// 
// Related Topics 位运算 数组 回溯算法

官方题解:

https://leetcode-cn.com/problems/subsets/solution/zi-ji-by-leetcode-solution/

class Solution {
    List<List<Integer>> lists = new ArrayList<>();
    List<Integer> list = new ArrayList<>();
    public List<List<Integer>> subsets(int[] nums) {
        /*int n = nums.length;
        //这里代表着一共有多少个子集
        for (int mask = 0; mask < (1<<n); mask++) {
            list.clear();
            for (int i = 0; i < n; i++) {
                //分别判断每个位上的数是不是1,如果是就添加上去
                if((mask & (1<<i)) != 0){
                    list.add(nums[i]);
                }
            }
            lists.add(new ArrayList<Integer>(list));
        }
        return lists;*/
        dfs(0, nums);
        return lists;
    }
    private void dfs(int cur, int[] nums){
        if(cur == nums.length){
            lists.add(new ArrayList<>(list));
            return;
        }
        //添加当前这个数
        list.add(nums[cur]);
        dfs(cur+1, nums);
        //不添加当前这个数,回溯
        list.remove(list.size()-1);
        dfs(cur+1, nums);
    }
}

2、单词搜索(L79)

//给定一个 m x n 二维字符网格 board 和一个字符串单词 word 。如果 word 存在于网格中,返回 true ;否则,返回 false 。 
//
// 单词必须按照字母顺序,通过相邻的单元格内的字母构成,其中“相邻”单元格是那些水平相邻或垂直相邻的单元格。同一个单元格内的字母不允许被重复使用。 
//
// 
//
// 示例 1: 
//
// 
//输入:board = [["A","B","C","E"],["S","F","C","S"],["A","D","E","E"]], word = "AB
//CCED"
//输出:true
// 
//
// 示例 2: 
//
// 
//输入:board = [["A","B","C","E"],["S","F","C","S"],["A","D","E","E"]], word = "SE
//E"
//输出:true
// 
//
// 示例 3: 
//
// 
//输入:board = [["A","B","C","E"],["S","F","C","S"],["A","D","E","E"]], word = "AB
//CB"
//输出:false
// 
//
// 
//
// 提示: 
//
// 
// m == board.length 
// n = board[i].length 
// 1 <= m, n <= 6 
// 1 <= word.length <= 15 
// board 和 word 仅由大小写英文字母组成 
// 
//
// 
//
// 进阶:你可以使用搜索剪枝的技术来优化解决方案,使其在 board 更大的情况下可以更快解决问题? 
// Related Topics 数组 回溯算法

官方题解:https://leetcode-cn.com/problems/word-search/solution/dan-ci-sou-suo-by-leetcode-solution/

class Solution {
    public boolean exist(char[][] board, String word) {
        int h = board.length;
        int w = board[0].length;
        boolean[][] visited = new boolean[h][w];
        for (int i = 0; i < h; i++) {
            for (int j = 0; j < w; j++) {
                boolean flag = check(board, visited, i, j, word, 0);
                if (flag) {
                    return true;
                }
            }
        }
        return false;
    }
    public boolean check(char[][] board, boolean[][] visited, int i, int j, String s, int k) {
        //如果当前目前这个位置的字母和字符串当前位置的字母不相等,那么这个“时间线”就直接抛弃
        if (board[i][j] != s.charAt(k)) {
            return false;
        //如果相等有恰恰遍历完了这个字符串,那就直接返回true
        } else if (k == s.length() - 1) {
            return true;
        }
        //如果相等,但字符串又没有遍历完,那就先把这个位置标为正确,在从这个位置出发去查询邻位,因为这个位置被
        //用过了,所以需要不能再被使用
        visited[i][j] = true;
        int[][] directions = {{0, 1}, {0, -1}, {1, 0}, {-1, 0}};
        boolean result = false;
        for (int[] dir : directions) {
            int newi = i + dir[0], newj = j + dir[1];
            if (newi >= 0 && newi < board.length && newj >= 0 && newj < board[0].length) {
                if (!visited[newi][newj]) {
                    boolean flag = check(board, visited, newi, newj, s, k + 1);
                    if (flag) {
                        result = true;
                        break;
                    }
                }
            }
        }
        //回溯
        visited[i][j] = false;
        return result;
    }
}

 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值