剑指offer:二叉树与双向链表

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

思路:
(1)、非递归
在这里插入图片描述
我们知道,二叉排序树用中序遍历的方法访问出来的序列刚好就是有序序列。因此可以采用中序遍历的非递归算法来解决这个问题。
以上图的二叉排序树为例:先一直让根节点及根节点的左子树进栈,10 5 4,此时可以修改4节点的指针指向。4节点出栈。则访问5节点出栈,修改5节点的指向,7节点进栈,修改7节点的指向,再出栈,修改10节点的指向,接着12节点进栈,修改指向,遍历完成后栈为空了。结束中序遍历。
实现:

/*
struct TreeNode {
	int val;
	struct TreeNode *left;
	struct TreeNode *right;
	TreeNode(int x) :
			val(x), left(NULL), right(NULL) {
	}
};*/
class Solution {
public:
    TreeNode* Convert(TreeNode* pRootOfTree)
    {
        if(pRootOfTree==NULL)
        {
            return NULL;
        }
        
        stack<TreeNode*>s;
        TreeNode *p=pRootOfTree;//遍历树的,初始指向根节点
        TreeNode *before=NULL;//指向上一个被遍历的节点,初始指向NULL
        TreeNode *head=NULL;//用于返回的双向链表的第一个
        bool isfrist = true;//判断是否是左子树最左的节点
        while(p!=NULL||!s.empty())
        {
            while(p!=NULL)
            {
                s.push(p);
                p = p->left;
            }
            p = s.top();//栈顶指针
            s.pop();
            if(isfrist)//是最左边的指针
            {
                head = p;
                before = head;
                isfrist = false;
            }
            else{
                before->right = p;
                p->left = before;
                before = p;
            }
            p = p->right;
        }
        return head;
        
    }

(2)、递归(发现树的好多问题都要用到递归啊。。。)
将左子树构成双向链表,并返回左子树的头结点
将左链表定位到最后一个节点。
若左链表不为空,则将当前的root追加到左链表的后面。
将右子树构成双向链表并返回右链表的头结点
右链表不为空,则将它追加到root的后面
左链表为空,返回root,不为空返回左链表头结点。

实现:

/*
struct TreeNode {
	int val;
	struct TreeNode *left;
	struct TreeNode *right;
	TreeNode(int x) :
			val(x), left(NULL), right(NULL) {
	}
};*/
class Solution {
public:
    TreeNode* Convert(TreeNode* pRootOfTree)
    {
        if(pRootOfTree==NULL)
        {
            return NULL;
        }
        if(pRootOfTree->left==NULL&&pRootOfTree->right==NULL)
        {
            return pRootOfTree;
        }
        TreeNode *left = Convert(pRootOfTree->left);//左子树的链表头结点
        if(left!=NULL)//左链表不为空则将root追加到左链表后面
        {
            TreeNode *p = left;//用于寻找左链表的最后一个节点
            while(p->right!=NULL)
            {
                p = p->right;
            }
            pRootOfTree->left = p;
            p->right = pRootOfTree;
        }
        TreeNode *right = Convert(pRootOfTree->right);//将右子树合成右链表并返回头结点
        if(right!=NULL)//右链表不空,追加到root的后面
        {
            pRootOfTree->right = right;
            right->left = pRootOfTree;
        }
        return left!=NULL?left:pRootOfTree;
    }
};

finished!!!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值