class Solution {
// 1. 中序,递归,来自解题大佬
Node pre, head;
public Node treeToDoublyList(Node root) {
// 边界值
if(root == null) return null;
dfs(root);
// 题目要求头尾连接
head.left = pre;
pre.right = head;
// 返回头节点
return head;
}
void dfs(Node cur) {
// 递归结束条件
if(cur == null) return;
dfs(cur.left);
// 如果pre为空,就说明是第一个节点,头结点,然后用head保存头结点,用于之后的返回
if (pre == null) head = cur;
// 如果不为空,那就说明是中间的节点。并且pre保存的是上一个节点,
// 让上一个节点的右指针指向当前节点
else if (pre != null) pre.right = cur;
// 再让当前节点的左指针指向父节点,也就连成了双向链表
cur.left = pre;
// 保存当前节点,用于下层递归创建
pre = cur;
dfs(cur.right);
}
}
整体思路容易理解,先进行中序遍历(left->root->right),由于是二叉搜索树,所以中序遍历自然是由小到大进行排序。然后处理中间节点。
1.若定义的pre节点为空,说明刚遍历到第一个节点,所以把它定义为头节点,即head=cur
2.若pre节点不为空,说明已经遍历到了中间节点,则需要处理为双向链表,即让上一个节点的右指针指向当前节点,当前节点的左指针指向上一个节点,这样就构成了双向链表。
3.pre保存当前节点进行下一次递归。