leetcode47,leetcode491,leetcode40,leetcode90,系列问题包你懂!!!Trie树对于排列问题、组合等结果集去重的应用

本文主要是介绍leetcode47,leetcode491,leetcode40,leetcode90,系列问题包你懂!!!Trie树对于排列问题、组合等结果集去重的应用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

leetcode47. 全排列 II,leetcode491. 非递减子序列,leetcode40. 组合总和 II,leetcode90. 子集 II

题目

不过多赘述

思路

实际上这几题是相同的,都是求一个结果集合,然后集合中的元素不能重复。并且每一个元素都满足一定要求。
那么实际上,我们想一想,trie树就是一种很好的具有去重功能的数据结构,那么这些问题就都可以归结为建立一个trie树,然后从头读到叶子节点,然后对于具体问题就是看他的要求了,比如排列问题相当于没要求,非递减子序列则是要求trie树dfs过程元素一致单调不减,求和就是累计递归过程中节点的值与target值对比,子集则是递归过程中每读一个新节点我就存一次结果。
这么一看是不是通透了,那么我们举一个例子,这个例子是leetcode491. 非递减子序列,也是我思考出这一思路的题,下面将是我逐步优化代码的过程。注意第三版trie树的set还可以用map,这样甚至能记录一些其他属性比如每个节点上出现该数字的次数。

题解

// 第一版,这时候我是建立一个trie树,然后再遍历的
class Solution {// 这题用trie树解决,其实init时候可以边建立树边记录数据,不用再dfs,可以优化一下List<List<Integer>> list = new ArrayList<>();List<Integer> subList = new ArrayList<>();int len = 0;public List<List<Integer>> findSubsequences(int[] nums) {Node dummy = new Node();// 别忘了初始化默认是0dummy.val = -101;dummy.subNode = new ArrayList<>();trieInit(dummy, nums, 0);dfs(dummy);return list;}public void dfs(Node root) {if (root == null) {return;}for(int i=0;i<root.subNode.size();i++) {subList.add(root.subNode.get(i).val);len++;if (len > 1) {list.add(new ArrayList<>(subList));}dfs(root.subNode.get(i));len--;subList.remove(subList.size() - 1);}}public void trieInit(Node root, int[] nums, int idx) {if (idx == nums.length) {return;}for(int i=idx;i<nums.length;i++) {if (!root.subNode.isEmpty() && root.subNode.get(root.subNode.size() - 1).val == nums[i]) {continue;}if (root.val > nums[i]) {continue;}boolean flag = false;for (int j=0;j<root.subNode.size();j++) {if (root.subNode.get(j).val == nums[i]) {trieInit(root.subNode.get(j), nums, i + 1);flag = true;break;}}if (flag) {continue;}Node node = new Node();node.val = nums[i];node.subNode = new ArrayList<>();root.subNode.add(node);/*subList.add(nums[i]);len++;if (len > 1) {list.add(new ArrayList<>(subList));}*/trieInit(node, nums, i + 1);/*len--;subList.remove(subList.size() - 1);*/}}
}class Node {List<Node> subNode;int val;
}
// 第二版,我将建立树和遍历树合为一起,一边建立一边读
class Solution {// 这题用trie树解决,其实init时候可以边建立树边记录数据,不用再dfs,可以优化一下List<List<Integer>> list = new ArrayList<>();List<Integer> subList = new ArrayList<>();int len = 0;public List<List<Integer>> findSubsequences(int[] nums) {Node dummy = new Node();// 别忘了初始化默认是0dummy.val = -101;dummy.subNode = new ArrayList<>();trieInit(dummy, nums, 0);return list;}public void trieInit(Node root, int[] nums, int idx) {//if (idx == nums.length) {return;}for(int i=idx;i<nums.length;i++) {if (!root.subNode.isEmpty() && root.subNode.get(root.subNode.size() - 1).val == nums[i]) {continue;}if (root.val > nums[i]) {continue;}boolean flag = false;for (int j=0;j<root.subNode.size();j++) {if (root.subNode.get(j).val == nums[i]) {trieInit(root.subNode.get(j), nums, i + 1);flag = true;break;}}if (flag) {continue;}Node node = new Node();node.val = nums[i];node.subNode = new ArrayList<>();root.subNode.add(node);subList.add(nums[i]);len++;if (len > 1) {list.add(new ArrayList<>(subList));}trieInit(node, nums, i + 1);len--;subList.remove(subList.size() - 1);}}
}class Node {List<Node> subNode;int val;
}
// 第三版,我意识到不需要Node类来具体对应树的数据结构,可以在每个节点进到trieInit函数时候建立一个set,这样set就是记录trie树当前节点下的所有可能子节点了,并且是自带去重的,还不用for遍历子节点了
class Solution {// 这题用trie树解决,其实init时候可以边建立树边记录数据,不用再dfs,可以优化一下List<List<Integer>> list = new ArrayList<>();List<Integer> subList = new ArrayList<>();public List<List<Integer>> findSubsequences(int[] nums) {trieInit(nums, 0);return list;}public void trieInit(int[] nums, int idx) {// 子节点中的数字都有哪些Set<Integer> set = new HashSet<>();for(int i=idx;i<nums.length;i++) {if (!subList.isEmpty() && subList.get(subList.size() - 1) > nums[i]) {continue;}if (set.contains(nums[i])) {continue;}subList.add(nums[i]);set.add(nums[i]);if (subList.size() > 1) {list.add(new ArrayList<>(subList));}trieInit(nums, i + 1);subList.remove(subList.size() - 1);}}
}

这篇关于leetcode47,leetcode491,leetcode40,leetcode90,系列问题包你懂!!!Trie树对于排列问题、组合等结果集去重的应用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/582335

相关文章

IntelliJ IDEA 中配置 Spring MVC 环境的详细步骤及问题解决

《IntelliJIDEA中配置SpringMVC环境的详细步骤及问题解决》:本文主要介绍IntelliJIDEA中配置SpringMVC环境的详细步骤及问题解决,本文分步骤结合实例给大... 目录步骤 1:创建 Maven Web 项目步骤 2:添加 Spring MVC 依赖1、保存后执行2、将新的依赖

Spring 中的循环引用问题解决方法

《Spring中的循环引用问题解决方法》:本文主要介绍Spring中的循环引用问题解决方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录什么是循环引用?循环依赖三级缓存解决循环依赖二级缓存三级缓存本章来聊聊Spring 中的循环引用问题该如何解决。这里聊

Spring Boot中JSON数值溢出问题从报错到优雅解决办法

《SpringBoot中JSON数值溢出问题从报错到优雅解决办法》:本文主要介绍SpringBoot中JSON数值溢出问题从报错到优雅的解决办法,通过修改字段类型为Long、添加全局异常处理和... 目录一、问题背景:为什么我的接口突然报错了?二、为什么会发生这个错误?1. Java 数据类型的“容量”限制

C语言中位操作的实际应用举例

《C语言中位操作的实际应用举例》:本文主要介绍C语言中位操作的实际应用,总结了位操作的使用场景,并指出了需要注意的问题,如可读性、平台依赖性和溢出风险,文中通过代码介绍的非常详细,需要的朋友可以参... 目录1. 嵌入式系统与硬件寄存器操作2. 网络协议解析3. 图像处理与颜色编码4. 高效处理布尔标志集合

关于MongoDB图片URL存储异常问题以及解决

《关于MongoDB图片URL存储异常问题以及解决》:本文主要介绍关于MongoDB图片URL存储异常问题以及解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录MongoDB图片URL存储异常问题项目场景问题描述原因分析解决方案预防措施js总结MongoDB图

SpringBoot项目中报错The field screenShot exceeds its maximum permitted size of 1048576 bytes.的问题及解决

《SpringBoot项目中报错ThefieldscreenShotexceedsitsmaximumpermittedsizeof1048576bytes.的问题及解决》这篇文章... 目录项目场景问题描述原因分析解决方案总结项目场景javascript提示:项目相关背景:项目场景:基于Spring

解决Maven项目idea找不到本地仓库jar包问题以及使用mvn install:install-file

《解决Maven项目idea找不到本地仓库jar包问题以及使用mvninstall:install-file》:本文主要介绍解决Maven项目idea找不到本地仓库jar包问题以及使用mvnin... 目录Maven项目idea找不到本地仓库jar包以及使用mvn install:install-file基

usb接口驱动异常问题常用解决方案

《usb接口驱动异常问题常用解决方案》当遇到USB接口驱动异常时,可以通过多种方法来解决,其中主要就包括重装USB控制器、禁用USB选择性暂停设置、更新或安装新的主板驱动等... usb接口驱动异常怎么办,USB接口驱动异常是常见问题,通常由驱动损坏、系统更新冲突、硬件故障或电源管理设置导致。以下是常用解决

Java中的Lambda表达式及其应用小结

《Java中的Lambda表达式及其应用小结》Java中的Lambda表达式是一项极具创新性的特性,它使得Java代码更加简洁和高效,尤其是在集合操作和并行处理方面,:本文主要介绍Java中的La... 目录前言1. 什么是Lambda表达式?2. Lambda表达式的基本语法例子1:最简单的Lambda表

Mysql如何解决死锁问题

《Mysql如何解决死锁问题》:本文主要介绍Mysql如何解决死锁问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录【一】mysql中锁分类和加锁情况【1】按锁的粒度分类全局锁表级锁行级锁【2】按锁的模式分类【二】加锁方式的影响因素【三】Mysql的死锁情况【1