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

相关文章

springboot循环依赖问题案例代码及解决办法

《springboot循环依赖问题案例代码及解决办法》在SpringBoot中,如果两个或多个Bean之间存在循环依赖(即BeanA依赖BeanB,而BeanB又依赖BeanA),会导致Spring的... 目录1. 什么是循环依赖?2. 循环依赖的场景案例3. 解决循环依赖的常见方法方法 1:使用 @La

Python中随机休眠技术原理与应用详解

《Python中随机休眠技术原理与应用详解》在编程中,让程序暂停执行特定时间是常见需求,当需要引入不确定性时,随机休眠就成为关键技巧,下面我们就来看看Python中随机休眠技术的具体实现与应用吧... 目录引言一、实现原理与基础方法1.1 核心函数解析1.2 基础实现模板1.3 整数版实现二、典型应用场景2

SpringBoot启动报错的11个高频问题排查与解决终极指南

《SpringBoot启动报错的11个高频问题排查与解决终极指南》这篇文章主要为大家详细介绍了SpringBoot启动报错的11个高频问题的排查与解决,文中的示例代码讲解详细,感兴趣的小伙伴可以了解一... 目录1. 依赖冲突:NoSuchMethodError 的终极解法2. Bean注入失败:No qu

Python Dash框架在数据可视化仪表板中的应用与实践记录

《PythonDash框架在数据可视化仪表板中的应用与实践记录》Python的PlotlyDash库提供了一种简便且强大的方式来构建和展示互动式数据仪表板,本篇文章将深入探讨如何使用Dash设计一... 目录python Dash框架在数据可视化仪表板中的应用与实践1. 什么是Plotly Dash?1.1

MySQL新增字段后Java实体未更新的潜在问题与解决方案

《MySQL新增字段后Java实体未更新的潜在问题与解决方案》在Java+MySQL的开发中,我们通常使用ORM框架来映射数据库表与Java对象,但有时候,数据库表结构变更(如新增字段)后,开发人员可... 目录引言1. 问题背景:数据库与 Java 实体不同步1.1 常见场景1.2 示例代码2. 不同操作

Android Kotlin 高阶函数详解及其在协程中的应用小结

《AndroidKotlin高阶函数详解及其在协程中的应用小结》高阶函数是Kotlin中的一个重要特性,它能够将函数作为一等公民(First-ClassCitizen),使得代码更加简洁、灵活和可... 目录1. 引言2. 什么是高阶函数?3. 高阶函数的基础用法3.1 传递函数作为参数3.2 Lambda

如何解决mysql出现Incorrect string value for column ‘表项‘ at row 1错误问题

《如何解决mysql出现Incorrectstringvalueforcolumn‘表项‘atrow1错误问题》:本文主要介绍如何解决mysql出现Incorrectstringv... 目录mysql出现Incorrect string value for column ‘表项‘ at row 1错误报错

如何解决Spring MVC中响应乱码问题

《如何解决SpringMVC中响应乱码问题》:本文主要介绍如何解决SpringMVC中响应乱码问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Spring MVC最新响应中乱码解决方式以前的解决办法这是比较通用的一种方法总结Spring MVC最新响应中乱码解

Java中&和&&以及|和||的区别、应用场景和代码示例

《Java中&和&&以及|和||的区别、应用场景和代码示例》:本文主要介绍Java中的逻辑运算符&、&&、|和||的区别,包括它们在布尔和整数类型上的应用,文中通过代码介绍的非常详细,需要的朋友可... 目录前言1. & 和 &&代码示例2. | 和 ||代码示例3. 为什么要使用 & 和 | 而不是总是使

pip无法安装osgeo失败的问题解决

《pip无法安装osgeo失败的问题解决》本文主要介绍了pip无法安装osgeo失败的问题解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 进入官方提供的扩展包下载网站寻找版本适配的whl文件注意:要选择cp(python版本)和你py