用二叉树作为存储结构时,取到一个节点,只能获取节点的左孩子和右孩子,不能直接得到节点的任一遍历序列的前驱或者后继。但是常常我们会想要更加直观的知道节点的前驱后继。线索二叉树显得尤为的重要。
线索二叉树的关键就是要定义一个全局变量来存放上一个访问过的结点。
Node* prev;
(一)前序线索二叉树
void PrevOrderTag() { _PrevOrderTag(_root); } void _PrevOrderTag(Node* root)//前序线索二叉树 { if (root == NULL) return; if (!root->_left) { root->_leftTag = THREAD; root->_left = prev; } if (prev && !prev->_right) { prev->_rightTag = THREAD; prev->_right = root; } prev = root; if (root->_leftTag == LINK)//只有当_leftTag为LINK时递归修改前驱后继 _PrevOrderTag(root->_left); if (root->_rightTag == LINK) _PrevOrderTag(root->_right); } void PrevOrderTagPrint()//前序线索化打印 { Node* cur = _root; //while (cur) //{ // while (cur->_leftTag == LINK) // { // cout << cur->_data << " "; // cur = cur->_left; // } // cout << cur->_data << " "; // cur = cur->_right; //} //2. while (cur) { cout << cur->_data << " "; if (cur->_leftTag == LINK) { cur = cur->_left; } else { cur = cur->_right; } } }
使用二叉树的线索打印二叉树是比较方便的,不用递归就能解决问题。只要cur不为NULL就一直寻找后继打印。
(二)中序线索二叉树
void MidOrderTag() { _MidOrderTag(_root); } void _MidOrderTag(Node* root)//中序线索二叉树 { if (root == NULL) { return; } if (root->_leftTag == LINK)//只有当_leftTag为LINK时递归修改前驱后继 _MidOrderTag(root->_left); if (!root->_left) { root->_leftTag = THREAD; root->_left = prev; } if (prev&&!prev->_right) { prev->_rightTag = THREAD; prev->_right = root; } prev = root; if (root->_rightTag == LINK) _MidOrderTag(root->_right); } void MidOrderTagPrint()//中序线索打印 { Node* cur = _root; while (cur) { while (cur->_leftTag == LINK) { cur = cur->_left; } cout << cur->_data << " "; while (cur->_rightTag == THREAD) { cur = cur->_right; cout << cur->_data << " "; } cur = cur->_right;//在打印右子树之前一定保证左子树已经打印过了 } cout << endl; }
前序的线索化打印不难懂,但是中序得知道什么时候访问右结点,在访问了左节点后才能访问右节点。
(三)后序线索二叉树
void RearOrderTag() { _RearOrderTag(_root); } void _RearOrderTag(Node* root)//后序线索二叉树 { if (root == NULL) { return; } if (root->_leftTag == LINK)//只有当_leftTag为LINK时递归修改前驱后继 _RearOrderTag(root->_left); if (root->_rightTag == LINK) _RearOrderTag(root->_right); if (!root->_left) { root->_leftTag = THREAD; root->_left = prev; } if (prev&&!prev->_right) { prev->_rightTag = THREAD; prev->_right = root; } prev = root; }
注意,线索二叉树只有当tag的类型为LINK时才修改结点的前驱和后继
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。