首页 > 其他分享 >力扣---剑指 Offer 36. 二叉搜索树与双向链表

力扣---剑指 Offer 36. 二叉搜索树与双向链表

时间:2023-04-02 22:00:11浏览次数:42  
标签:Node --- val get Offer list 链表 root 节点

输入一棵二叉搜索树,将该二叉搜索树转换成一个排序的循环双向链表。要求不能创建任何新的节点,只能调整树中节点指针的指向。

 

为了让您更好地理解问题,以下面的二叉搜索树为例:

 

 

 

我们希望将这个二叉搜索树转化为双向循环链表。链表中的每个节点都有一个前驱和后继指针。对于双向循环链表,第一个节点的前驱是最后一个节点,最后一个节点的后继是第一个节点。

下图展示了上面的二叉搜索树转化成的链表。“head” 表示指向链表中有最小元素的节点。

特别地,我们希望可以就地完成转换操作。当转化完成以后,树中节点的左指针需要指向前驱,树中节点的右指针需要指向后继。还需要返回链表中的第一个节点的指针。

 

注意:本题与主站 426 题相同:https://leetcode-cn.com/problems/convert-binary-search-tree-to-sorted-doubly-linked-list/

注意:此题对比原题有改动。

来源:力扣(LeetCode)
链接:https://leetcode.cn/problems/er-cha-sou-suo-shu-yu-shuang-xiang-lian-biao-lcof
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。


 

两种方法。

1. 由于题目只要求了不能新建节点,所以我们可以直接利用一个list数组来存储所有节点,然后进行排,之后再遍历 list 中的所有元素,并对其进行连接操作。

2. 利用题目给的二叉搜索树的信息(中序遍历时,元素是从小到大),根据二叉搜索树的特点,可以直接将所有元素存入 list 中(此时list中的元素必定是非递减),然后和第一个方法相同操作。对比第一个方法,省略了排序的步骤。

1:

// 题目要求不能新建节点,但没说不能用额外空间存储。
// 因此,思路就来了,即:利用 list 数组来存储所有的节点,然后对list进行排序,排序规则是val从小到大。
// 接着,再遍历 list,将第i 个元素的左节点指向第 i - 1 个元素,右节点指向第 i+ 1 个元素。
// 最后再对开头和结尾进行连接操作,然后返回list第一个元素即可。
/*
// Definition for a Node.
class Node {
    public int val;
    public Node left;
    public Node right;

    public Node() {}

    public Node(int _val) {
        val = _val;
    }

    public Node(int _val,Node _left,Node _right) {
        val = _val;
        left = _left;
        right = _right;
    }
};
*/
class Solution {
    public Node treeToDoublyList(Node root) {
        // 节点为 null
        if (root == null) {
            return root;
        }
        // 节点个数为 1
        if (root.left == null && root.right == null) {
            root.left = root;
            root.right = root;
            return root;
        }
        List<Node> list = new ArrayList<>();
        // 深度优先遍历,将所有节点存入 list 中。
        dfs(root, list);
        // 排序
        list.sort((a, b) -> (a.val - b.val));
        // 对list中所有元素进行连接操作。
        int len = list.size() - 1;
        for (int i = 1; i < len; i ++) {
            list.get(i).left = list.get(i - 1);
            list.get(i).right = list.get(i + 1);
        }
        list.get(0).left = list.get(len);
        list.get(0).right = list.get(1);
        list.get(len).left = list.get(list.size() - 2);
        list.get(len).right = list.get(0);
        return list.get(0);
    }
    // 深度优先遍历,将所有的非 null 节点存入 list 中。
    private void dfs(Node root, List<Node> list) {
        if (root == null) {
            return;
        }
        list.add(root);
        dfs(root.left, list);
        dfs(root.right, list);
    }
}

 

 2:

// 题目要求不能新建节点,但没说不能用额外空间存储。
// 因此,思路就来了,即:利用 list 数组来存储所有的节点,然后对list进行排序,排序规则是val从小到大。
// 接着,再遍历 list,将第i 个元素的左节点指向第 i - 1 个元素,右节点指向第 i+ 1 个元素。
// 最后再对开头和结尾进行连接操作,然后返回list第一个元素即可。
/*
// Definition for a Node.
class Node {
    public int val;
    public Node left;
    public Node right;

    public Node() {}

    public Node(int _val) {
        val = _val;
    }

    public Node(int _val,Node _left,Node _right) {
        val = _val;
        left = _left;
        right = _right;
    }
};
*/
class Solution {
    public Node treeToDoublyList(Node root) {
        // 节点为 null
        if (root == null) {
            return root;
        }
        // 节点个数为 1
        if (root.left == null && root.right == null) {
            root.left = root;
            root.right = root;
            return root;
        }
        List<Node> list = new ArrayList<>();
        // 深度优先遍历,将所有节点存入 list 中。
        dfs(root, list);
        // 排序
        // list.sort((a, b) -> (a.val - b.val));
        // 对list中所有元素进行连接操作。
        int len = list.size() - 1;
        for (int i = 1; i < len; i ++) {
            list.get(i).left = list.get(i - 1);
            list.get(i).right = list.get(i + 1);
        }
        list.get(0).left = list.get(len);
        list.get(0).right = list.get(1);
        list.get(len).left = list.get(list.size() - 2);
        list.get(len).right = list.get(0);
        return list.get(0);
    }
    // 深度优先遍历,将所有的非 null 节点存入 list 中。
    // 中序遍历。
    private void dfs(Node root, List<Node> list) {
        if (root == null) {
            return;
        }
        dfs(root.left, list);
        list.add(root);
        dfs(root.right, list);
    }
}

 

标签:Node,---,val,get,Offer,list,链表,root,节点
From: https://www.cnblogs.com/allWu/p/17281480.html

相关文章

  • 今日报告-42
    今日打卡所花时间(包括上课):4h代码量(行):200发表博客:2篇(不包括本篇)学习进度和了解到的知识点:今天基本完成了前端的页面,后端的识别逻辑也初步能够使用,现在要解决的就是将其链接起来。目前遇到了一些问题,实例化Servlet类出现异常。之前做测试的时候能够正常运行,导入maven项......
  • 每日总结2023-04-02
    今天完成了厂商端 1.登录界面  2.查看数据界面,在此界面商家收到通知增加待完成选项,并选择是否准备好以及完成。   3.个人信息界面,可以保存个人信息 ......
  • JAVA - IO 流
    FileInputStreamimportjava.io.FileOutputStream;importjava.io.IOException;publicclassFileOutPutStreamDemo{/*FileOutputStream使用细节:1.write方法写出的数字:是ASCCI表中字符对应的数字,因此a=>972.创建FileOutputStream对象时“E:\java......
  • ACM预备队-大一下学期week(3)集训
    1.饿饿,饭饭2题目链接:饿饿饭饭2-Problem-DaimayuanOnlineJudge1#include<iostream>2usingnamespacestd;34intmain(){5intT;6cin>>T;7while(T--){8intn;9cin>>n;10inta[n];1......
  • [论文阅读] Diff-Font: Diffusion Model for Robust One-Shot Font Generation
    pretitle:Diff-Font:DiffusionModelforRobustOne-ShotFontGenerationaccepted:arxiv2022paper:https://arxiv.org/abs/2212.05895code:noneref:https://www.zhihu.com/question/545764550关键词:one-shot,字体生成,扩散模型阅读理由:扩散模型在字体这边的第一次应......
  • linux内核数据结构 --- list_head
    以structkobject为例,讲解如何使用链表structlist_headstructkobject{constchar*name;structlist_headentry;structkobject*parent;...};structlist_head类型变量作为structkobject的成员(从面向对象的角度,也可以看成str......
  • SpringBoot集成Activiti7-单独配置数据源
    框架:SpringBoot+Mybatis+Activiti7思路:单独给mybatis和activiti配置datasourceMybati配置单数据源方法单数据源只需要在yml中配置url:jdbc:mysql://localhost:3306/localtest?serverTimezone=GMT%2B8&useUnicode=true&characterEncoding=utf8&rewriteBatchedStatemen......
  • nginx keepalive_timeout 300; 504 Gateway Time-out
    实践:1、http{includemime.types;#includeluawaf.conf;includeproxy.conf;default_typeapplication/octet-stream;server_names_hash_bucket_size512;client_header_buffer_size32k;large_client_header_buffers432k;client_max_b......
  • 力扣---剑指 Offer 34. 二叉树中和为某一值的路径
    给你二叉树的根节点root和一个整数目标和targetSum,找出所有从根节点到叶子节点路径总和等于给定目标和的路径。叶子节点是指没有子节点的节点。 示例1:  输入:root=[5,4,8,11,null,13,4,7,2,null,null,5,1],targetSum=22输出:[[5,4,11,2],[5,8,4,5]]示例2:......
  • odoo 开发入门教程系列-准备一些操作(Action)?
    准备一些操作(Action)?到目前为止,我们主要通过声明字段和视图来构建模块。在任何真实的业务场景中,我们都希望将一些业务逻辑链接到操作按钮。在我们的房地产示例中,我们希望能够:取消或将房产设置为已售出接受或拒绝报价有人可能会说,我们已经可以通过手动更改状态来完成这些事......