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

相关文章

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

中文分词jieba库的使用与实景应用(一)

知识星球:https://articles.zsxq.com/id_fxvgc803qmr2.html 目录 一.定义: 精确模式(默认模式): 全模式: 搜索引擎模式: paddle 模式(基于深度学习的分词模式): 二 自定义词典 三.文本解析   调整词出现的频率 四. 关键词提取 A. 基于TF-IDF算法的关键词提取 B. 基于TextRank算法的关键词提取

水位雨量在线监测系统概述及应用介绍

在当今社会,随着科技的飞速发展,各种智能监测系统已成为保障公共安全、促进资源管理和环境保护的重要工具。其中,水位雨量在线监测系统作为自然灾害预警、水资源管理及水利工程运行的关键技术,其重要性不言而喻。 一、水位雨量在线监测系统的基本原理 水位雨量在线监测系统主要由数据采集单元、数据传输网络、数据处理中心及用户终端四大部分构成,形成了一个完整的闭环系统。 数据采集单元:这是系统的“眼睛”,

好题——hdu2522(小数问题:求1/n的第一个循环节)

好喜欢这题,第一次做小数问题,一开始真心没思路,然后参考了网上的一些资料。 知识点***********************************无限不循环小数即无理数,不能写作两整数之比*****************************(一开始没想到,小学没学好) 此题1/n肯定是一个有限循环小数,了解这些后就能做此题了。 按照除法的机制,用一个函数表示出来就可以了,代码如下

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

hdu4869(逆元+求组合数)

//输入n,m,n表示翻牌的次数,m表示牌的数目,求经过n次操作后共有几种状态#include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<queue>#include<set>#include<map>#include<stdio.h>#include<stdlib.h>#includ

hdu1394(线段树点更新的应用)

题意:求一个序列经过一定的操作得到的序列的最小逆序数 这题会用到逆序数的一个性质,在0到n-1这些数字组成的乱序排列,将第一个数字A移到最后一位,得到的逆序数为res-a+(n-a-1) 知道上面的知识点后,可以用暴力来解 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#in

zoj3820(树的直径的应用)

题意:在一颗树上找两个点,使得所有点到选择与其更近的一个点的距离的最大值最小。 思路:如果是选择一个点的话,那么点就是直径的中点。现在考虑两个点的情况,先求树的直径,再把直径最中间的边去掉,再求剩下的两个子树中直径的中点。 代码如下: #include <stdio.h>#include <string.h>#include <algorithm>#include <map>#

购买磨轮平衡机时应该注意什么问题和技巧

在购买磨轮平衡机时,您应该注意以下几个关键点: 平衡精度 平衡精度是衡量平衡机性能的核心指标,直接影响到不平衡量的检测与校准的准确性,从而决定磨轮的振动和噪声水平。高精度的平衡机能显著减少振动和噪声,提高磨削加工的精度。 转速范围 宽广的转速范围意味着平衡机能够处理更多种类的磨轮,适应不同的工作条件和规格要求。 振动监测能力 振动监测能力是评估平衡机性能的重要因素。通过传感器实时监