JavaC++题解与拓展——leetcode606.根据二叉树创建字符串【HashSet,ArrayDeque,unordered_set学习与使用】

本文主要是介绍JavaC++题解与拓展——leetcode606.根据二叉树创建字符串【HashSet,ArrayDeque,unordered_set学习与使用】,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

每日一题做题记录,参考官方和三叶的题解

目录

  • 题目要求
  • 思路一:递归
    • Java
    • C++
  • 思路二:迭代(用栈替代递归)
    • Java
      • HashSet
      • ArrayDeque
    • C++
      • unorder_set(无序set容器)
  • 总结

题目要求

在这里插入图片描述
注:当节点仅有一个子树,右子树为空需去除冗余括号,左子树为空需添加一对括号。

思路一:递归

题目是输出树的前序遍历结果的变体形式(给每个子树加括号),所以可以使用深度优先遍历进行递归解决。
下文采用两种表达形式,Java中将DFS另外定义,C++直接调用自己,前者方法更具有“树”类题目的普适性,后者更明了不容易落下括号。(不是因为C++字符串不好修改才不一样的)

Java

class Solution {StringBuilder res = new StringBuilder();public String tree2str(TreeNode root) {DFS(root);return res.substring(1, res.length() - 1); //忽略首尾括号}void DFS(TreeNode root) {res.append("(");res.append(root.val);if (root.left != null)DFS(root.left);else if (root.right != null) //左空右不空res.append("()"); //指代空的左子树if (root.right != null)DFS(root.right);res.append(")");        }
}
  • 时间复杂度:O(m + n),m为边数,n为节点数
  • 空间复杂度:O(n)

C++

class Solution {
public:string tree2str(TreeNode *root) {if (root == nullptr)return "";if (root->left == nullptr && root->right == nullptr) //叶子return to_string(root->val);if (root->right == nullptr) //右子树空则跳过,以防止产生冗余括号return to_string(root->val) + "(" + tree2str(root->left) + ")";return to_string(root->val) + "(" + tree2str(root->left) + ")(" + tree2str(root->right) + ")";}
};
  • 时间复杂度:O(n),n为节点数
  • 空间复杂度:O(n)

思路二:迭代(用栈替代递归)

定义一个栈存,栈底到栈顶依次存根到当前节点的经过的节点,将其依次加入结果并添加括号,因此还需一个额外的集合存储已遍历(输出)过的节点。未遍历过则添加“(”和该节点并向下遍历其子树,遍历过则添加“)”。

Java

class Solution {public String tree2str(TreeNode root) {StringBuilder res = new StringBuilder();Set<TreeNode> vis = new HashSet<>(); //是否遍历过Deque<TreeNode> stack = new ArrayDeque<>(); //基于双端队列创建栈stack.addLast(root);while (!stack.isEmpty()) {TreeNode t = stack.pollLast();if (vis.contains(t)) //遍历过res.append(")");else {stack.addLast(t);res.append("(");res.append(t.val);//先进后出,所以先压入右子树内容if (t.right != null)stack.addLast(t.right);if (t.left != null)stack.addLast(t.left);else if (t.right != null)res.append("()");vis.add(t);}}return res.substring(1, res.length() - 1); //忽略首尾冗余括号}
}
  • 时间复杂度:O(m + n),m为边数,n为节点数
  • 空间复杂度:O(n)

HashSet

  • 学习参考链接
  • 简介
    • 基于HashMap实现,元素不可重复,但可有空值;
    • 不会对插入数据排序。
方法功能
contains(key)判断key是否存在于容器中
add(key)将key加入容器

ArrayDeque

  • 学习参考链接
  • 简介
    • 一个两端皆可插入/删除的队列
方法功能
addLast(key)将key加入队尾
isEmpty()队列是否为空
pollLast(key)返回并删除队尾元素key

C++

class Solution {
public:string tree2str(TreeNode* root) {string res = "";stack<TreeNode *> st;st.push(root);unordered_set<TreeNode *> vis;while(!st.empty()) {auto node = st.top();if(vis.count(node)) {if(node != root)res += ")";st.pop();}else {vis.insert(node);if(node != root)res += "(";res += to_string(node -> val);if(node -> left == nullptr && node -> right != nullptr)res += "()"; //顶替空的左子树if(node -> right != nullptr)st.push(node -> right);if(node -> left != nullptr)st.push(node -> left);}}return res;}
};
  • 时间复杂度:O(n),n为节点数
  • 空间复杂度:O(n)

unorder_set(无序set容器)

  • 学习参考链接
  • 简介
    • unorder_set容器是STL无序容器(哈希容器)之一,底层采用哈希表存储结构,用链地址法解决数据位置发生冲突的哈希表;
    • 存值不存键;
    • 各元素值互不相等且不可修改;
    • 不会对插入数据排序(与set容器的差异)。
成员方法功能
count(key)在容器中查找值为key的元素的个数
insert(key)将key加入容器

总结

本题属于简单题目,可运用“树”题目的套路化解法——递归与迭代。
其中,迭代方法中需额外定义一个无序集合存储遍历过的节点,在Java与C++中分别以Hashset和unordered_set实现,二者实质上均为哈希表结构。


欢迎指正与讨论!

这篇关于JavaC++题解与拓展——leetcode606.根据二叉树创建字符串【HashSet,ArrayDeque,unordered_set学习与使用】的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

SpringBoot 获取请求参数的常用注解及用法

《SpringBoot获取请求参数的常用注解及用法》SpringBoot通过@RequestParam、@PathVariable等注解支持从HTTP请求中获取参数,涵盖查询、路径、请求体、头、C... 目录SpringBoot 提供了多种注解来方便地从 HTTP 请求中获取参数以下是主要的注解及其用法:1

HTTP 与 SpringBoot 参数提交与接收协议方式

《HTTP与SpringBoot参数提交与接收协议方式》HTTP参数提交方式包括URL查询、表单、JSON/XML、路径变量、头部、Cookie、GraphQL、WebSocket和SSE,依据... 目录HTTP 协议支持多种参数提交方式,主要取决于请求方法(Method)和内容类型(Content-Ty

深度解析Java @Serial 注解及常见错误案例

《深度解析Java@Serial注解及常见错误案例》Java14引入@Serial注解,用于编译时校验序列化成员,替代传统方式解决运行时错误,适用于Serializable类的方法/字段,需注意签... 目录Java @Serial 注解深度解析1. 注解本质2. 核心作用(1) 主要用途(2) 适用位置3

sky-take-out项目中Redis的使用示例详解

《sky-take-out项目中Redis的使用示例详解》SpringCache是Spring的缓存抽象层,通过注解简化缓存管理,支持Redis等提供者,适用于方法结果缓存、更新和删除操作,但无法实现... 目录Spring Cache主要特性核心注解1.@Cacheable2.@CachePut3.@Ca

C#下Newtonsoft.Json的具体使用

《C#下Newtonsoft.Json的具体使用》Newtonsoft.Json是一个非常流行的C#JSON序列化和反序列化库,它可以方便地将C#对象转换为JSON格式,或者将JSON数据解析为C#对... 目录安装 Newtonsoft.json基本用法1. 序列化 C# 对象为 JSON2. 反序列化

深入浅出Spring中的@Autowired自动注入的工作原理及实践应用

《深入浅出Spring中的@Autowired自动注入的工作原理及实践应用》在Spring框架的学习旅程中,@Autowired无疑是一个高频出现却又让初学者头疼的注解,它看似简单,却蕴含着Sprin... 目录深入浅出Spring中的@Autowired:自动注入的奥秘什么是依赖注入?@Autowired

Spring 依赖注入与循环依赖总结

《Spring依赖注入与循环依赖总结》这篇文章给大家介绍Spring依赖注入与循环依赖总结篇,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1. Spring 三级缓存解决循环依赖1. 创建UserService原始对象2. 将原始对象包装成工

Java中如何正确的停掉线程

《Java中如何正确的停掉线程》Java通过interrupt()通知线程停止而非强制,确保线程自主处理中断,避免数据损坏,线程池的shutdown()等待任务完成,shutdownNow()强制中断... 目录为什么不强制停止为什么 Java 不提供强制停止线程的能力呢?如何用interrupt停止线程s

SpringBoot请求参数传递与接收示例详解

《SpringBoot请求参数传递与接收示例详解》本文给大家介绍SpringBoot请求参数传递与接收示例详解,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋... 目录I. 基础参数传递i.查询参数(Query Parameters)ii.路径参数(Path Va

SpringBoot路径映射配置的实现步骤

《SpringBoot路径映射配置的实现步骤》本文介绍了如何在SpringBoot项目中配置路径映射,使得除static目录外的资源可被访问,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一... 目录SpringBoot路径映射补:springboot 配置虚拟路径映射 @RequestMapp