剑指Offer----面试题27:二叉搜索树与双向链表

题目:


输入一颗二叉搜索树,将该二叉搜索树转换成一个排序的双向链表。要求不能创建任何新的结点,只能调整树中结点指针的指向。如下图中的二叉搜索树,则输出转换之后的排序双向链表。


分析:


中序遍历算法的特点是按照从小到大的顺序遍历二叉树的每一个结点。当遍历到根节点的时候,可以把树分为三部分:值为10的结点,根节点值为6的左子树,根节点值为14的右子树。根据排序链表的定义,将根节点与左子树最大的一个结点连接起来,同时将该根节点与右子树最小的一个结点连接起来。如下图所示,再用同样的方法处理左子树和右子树(递归)。



源代码如下:
BinaryTreeNode *Covert(BinaryTreeNode *pHead)
{
	if (pHead == NULL)
		return NULL;
	BinaryTreeNode *pLastNode = NULL;
	ConvertNode(pHead, &pLastNode);

	//pLastNode指向双向链表的尾结点
	//我们需要返回头结点
	BinaryTreeNode *pHeadNode = pLastNode;
	while (pHeadNode != NULL && pHeadNode->left != NULL)
		pHeadNode = pHeadNode->left;

	return pHeadNode;
}

void ConvertNode(BinaryTreeNode *node, BinaryTreeNode **pLastNode)
{
	if (node == NULL)
		return;

	BinaryTreeNode *current = node;

	if (current->left != NULL)
		ConvertNode(current->left, pLastNode);

	current->left = *pLastNode;
	if (*pLastNode != NULL)
		(*pLastNode)->right = current;

	*pLastNode = current;

	if (current->right != NULL)
		ConvertNode(current->right, pLastNode);
}


官方源代码:

#include"BinaryTree.h"
#include<cstdlib>
#include<cstdio>

using namespace OrdinaryBinaryTreeSpace6;

void ConvertNode(BinaryTreeNode* pNode, BinaryTreeNode** pLastNodeInList);

BinaryTreeNode* Convert(BinaryTreeNode* pRootOfTree)
{
	BinaryTreeNode *pLastNodeInList = NULL;
	ConvertNode(pRootOfTree, &pLastNodeInList);

	// pLastNodeInList指向双向链表的尾结点,
	// 我们需要返回头结点
	BinaryTreeNode *pHeadOfList = pLastNodeInList;
	while (pHeadOfList != NULL && pHeadOfList->left != NULL)
		pHeadOfList = pHeadOfList->left;

	return pHeadOfList;
}

void ConvertNode(BinaryTreeNode* pNode, BinaryTreeNode** pLastNodeInList)
{
	if (pNode == NULL)
		return;

	BinaryTreeNode *pCurrent = pNode;

	if (pCurrent->left != NULL)
		ConvertNode(pCurrent->left, pLastNodeInList);

	pCurrent->left = *pLastNodeInList;
	if (*pLastNodeInList != NULL)
		(*pLastNodeInList)->right = pCurrent;

	*pLastNodeInList = pCurrent;

	if (pCurrent->right != NULL)
		ConvertNode(pCurrent->right, pLastNodeInList);
}

// ====================测试代码====================
void PrintDoubleLinkedList(BinaryTreeNode* pHeadOfList)
{
	BinaryTreeNode* pNode = pHeadOfList;

	printf("The nodes from left to right are:\n");
	while (pNode != NULL)
	{
		printf("%d\t", pNode->element);

		if (pNode->right == NULL)
			break;
		pNode = pNode->right;
	}

	printf("\nThe nodes from right to left are:\n");
	while (pNode != NULL)
	{
		printf("%d\t", pNode->element);

		if (pNode->left == NULL)
			break;
		pNode = pNode->left;
	}

	printf("\n");
}

void DestroyList(BinaryTreeNode* pHeadOfList)
{
	BinaryTreeNode* pNode = pHeadOfList;
	while (pNode != NULL)
	{
		BinaryTreeNode* pNext = pNode->right;

		delete pNode;
		pNode = pNext;
	}
}

void Test(char* testName, BinaryTreeNode* pRootOfTree)
{
	if (testName != NULL)
		printf("%s begins:\n", testName);

	PrintTreeMid(pRootOfTree);

	BinaryTreeNode* pHeadOfList = Convert(pRootOfTree);

	PrintDoubleLinkedList(pHeadOfList);
}

//            10
//         /      \
//        6        14
//       /\        /\
//      4  8     12  16
void Test1()
{
	BinaryTreeNode* pNode10 = CreateBinaryTreeNode(10);
	BinaryTreeNode* pNode6 = CreateBinaryTreeNode(6);
	BinaryTreeNode* pNode14 = CreateBinaryTreeNode(14);
	BinaryTreeNode* pNode4 = CreateBinaryTreeNode(4);
	BinaryTreeNode* pNode8 = CreateBinaryTreeNode(8);
	BinaryTreeNode* pNode12 = CreateBinaryTreeNode(12);
	BinaryTreeNode* pNode16 = CreateBinaryTreeNode(16);

	ConnectBinaryTreeNodes(pNode10, pNode6, pNode14);
	ConnectBinaryTreeNodes(pNode6, pNode4, pNode8);
	ConnectBinaryTreeNodes(pNode14, pNode12, pNode16);

	Test("Test1", pNode10);

	DestroyList(pNode4);
}

//               5
//              /
//             4
//            /
//           3
//          /
//         2
//        /
//       1
void Test2()
{
	BinaryTreeNode* pNode5 = CreateBinaryTreeNode(5);
	BinaryTreeNode* pNode4 = CreateBinaryTreeNode(4);
	BinaryTreeNode* pNode3 = CreateBinaryTreeNode(3);
	BinaryTreeNode* pNode2 = CreateBinaryTreeNode(2);
	BinaryTreeNode* pNode1 = CreateBinaryTreeNode(1);

	ConnectBinaryTreeNodes(pNode5, pNode4, NULL);
	ConnectBinaryTreeNodes(pNode4, pNode3, NULL);
	ConnectBinaryTreeNodes(pNode3, pNode2, NULL);
	ConnectBinaryTreeNodes(pNode2, pNode1, NULL);

	Test("Test2", pNode5);

	DestroyList(pNode1);
}

// 1
//  \
//   2
//    \
//     3
//      \
//       4
//        \
//         5
void Test3()
{
	BinaryTreeNode* pNode1 = CreateBinaryTreeNode(1);
	BinaryTreeNode* pNode2 = CreateBinaryTreeNode(2);
	BinaryTreeNode* pNode3 = CreateBinaryTreeNode(3);
	BinaryTreeNode* pNode4 = CreateBinaryTreeNode(4);
	BinaryTreeNode* pNode5 = CreateBinaryTreeNode(5);

	ConnectBinaryTreeNodes(pNode1, NULL, pNode2);
	ConnectBinaryTreeNodes(pNode2, NULL, pNode3);
	ConnectBinaryTreeNodes(pNode3, NULL, pNode4);
	ConnectBinaryTreeNodes(pNode4, NULL, pNode5);

	Test("Test3", pNode1);

	DestroyList(pNode1);
}

// 树中只有1个结点
void Test4()
{
	BinaryTreeNode* pNode1 = CreateBinaryTreeNode(1);
	Test("Test4", pNode1);

	DestroyList(pNode1);
}

// 树中没有结点
void Test5()
{
	Test("Test5", NULL);
}

int main()
{
	Test1();
	Test2();
	Test3();
	Test4();
	Test5();

	system("pause");
	return 0;
}

运行结果:
Test1 begins:
4  6  8  10  12  14  16  The nodes from left to right are:
4       6       8       10      12      14      16
The nodes from right to left are:
16      14      12      10      8       6       4
Test2 begins:
1  2  3  4  5  The nodes from left to right are:
1       2       3       4       5
The nodes from right to left are:
5       4       3       2       1
Test3 begins:
1  2  3  4  5  The nodes from left to right are:
1       2       3       4       5
The nodes from right to left are:
5       4       3       2       1
Test4 begins:
1  The nodes from left to right are:
1
The nodes from right to left are:
1
Test5 begins:
The tree is empty
The nodes from left to right are:

The nodes from right to left are:

请按任意键继续. . .




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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值