数据结构 - Codeforces Round #353 (Div. 2) D. Tree Construction

简介: link mean 给定n个数,按照构造Binary Search Tree的方式来构造BST树,按顺序输出每一个非root结点的父节点的值。 analyse 构造BST树最坏情况下时间复杂度为O(n),肯定会超时。

link

mean

给定n个数,按照构造Binary Search Tree的方式来构造BST树,按顺序输出每一个非root结点的父节点的值。

analyse

构造BST树最坏情况下时间复杂度为O(n),肯定会超时。

注意到只需要输出结点的父节点的值,不需要真的构造BST树。

插到第i个结点时,我们在前i-1个结点中找两个数,一个是比Ai大的最小数,一个是比Ai小的最大数。

那么Ai的父节点肯定是这两个中的一个,到底是哪一个呢,根据画图分析,可以得出是后来插入的那个。

如下图所示:

做法:用java里面的容器TreeMap<Integer,Integer>来存储<value,index>对,根据value值来查找,然后比较这两个数的index大小来判断谁是父节点。

time complexity

O(n*logn)

code

1. java容器实现 
import java.io.BufferedInputStream;
import java.util.HashMap;
import java.util.Scanner;
import java.util.TreeSet;

public class Main{
   public static void main(String[]argv){
       Scanner in=new Scanner(new BufferedInputStream(System.in));
       while(in.hasNext()){
           int n=in.nextInt();
           TreeSet<Integer> vals=new TreeSet<>();
           HashMap<Integer,Integer> idx=new HashMap<>();
           int val=in.nextInt();
           vals.add(val);
           idx.put(val,0);
           for(int i=1;i<n;++i){
               val=in.nextInt();
               Integer hi=vals.higher(val), lo=vals.lower(val);
               if(hi==null)
                   System.out.print(lo+" ");
               else if(lo==null)
                   System.out.print(hi+" ");
               else {
                   Integer hiIdx=idx.get(hi);
                   Integer loIdx=idx.get(lo);
                   if(hiIdx>loIdx)
                       System.out.print(hi+" ");
                   else
                       System.out.print(lo+" ");
               }
               vals.add(val);
               idx.put(val,i);
           }
           System.out.println();
       }
   }
}
2. avl实现 
#include<iostream> #include<cstdio> #include<set> #include<map> #include<vector> using namespace std; ; //AVL树节点信息 class TreeNode { public: TreeNode() :lson(NULL), rson(NULL), freq(), hgt() {} int data;// int hgt;//高度 unsigned int freq;//频率 TreeNode* lson;//指向左儿子的地址 TreeNode* rson;//指向右儿子的地址 }; //AVL树类的属性和方法声明 class AVLTree { private: TreeNode* root;//根节点 void insertpri(TreeNode* &node,int x,int &pre,int &aft);//插入 int height(TreeNode* node);//求树的高度 void SingRotateLeft(TreeNode* &k2);//左左情况下的旋转 void SingRotateRight(TreeNode* &k2);//右右情况下的旋转 void DoubleRotateLR(TreeNode* &k3);//左右情况下的旋转 void DoubleRotateRL(TreeNode* &k3);//右左情况下的旋转 int Max(int cmpa, int cmpb);//求最大值 public: AVLTree() :root(NULL) {} void insert(int x,int &pre,int &aft);//插入接口 }; //计算节点的高度 int AVLTree::height(TreeNode* node) { if (node != NULL) return node->hgt; ; } //求最大值 int AVLTree::Max(int cmpa, int cmpb) { return cmpa>cmpb ? cmpa : cmpb; } //左左情况下的旋转 void AVLTree::SingRotateLeft(TreeNode* &k2) { TreeNode* k1; k1 = k2->lson; k2->lson = k1->rson; k1->rson = k2; k2->hgt = Max(height(k2->lson), height(k2->rson)) + ; k1->hgt = Max(height(k1->lson), k2->hgt) + ; k2 = k1; } //右右情况下的旋转 void AVLTree::SingRotateRight(TreeNode* &k2) { TreeNode* k1; k1 = k2->rson; k2->rson = k1->lson; k1->lson = k2; k2->hgt = Max(height(k2->lson), height(k2->rson)) + ; k1->hgt = Max(height(k1->rson), k2->hgt) + ; k2 = k1; } //左右情况的旋转 void AVLTree::DoubleRotateLR(TreeNode* &k3) { SingRotateRight(k3->lson); SingRotateLeft(k3); } //右左情况的旋转 void AVLTree::DoubleRotateRL(TreeNode* &k3) { SingRotateLeft(k3->rson); SingRotateRight(k3); } //插入 void AVLTree::insertpri(TreeNode* &node, int x,int &pre,int &aft) { if (node == NULL)//如果节点为空,就在此节点处加入x信息 { node = new TreeNode(); node->data = x; return; } if (node->data>x)//如果x小于节点的值,就继续在节点的左子树中插入x { aft = node->data; insertpri(node->lson, x,pre,aft); == height(node->lson) - height(node->rson)) if (x<node->lson->data) SingRotateLeft(node); else DoubleRotateLR(node); } else if (node->data<x)//如果x大于节点的值,就继续在节点的右子树中插入x { pre = node->data; insertpri(node->rson, x,pre,aft); == height(node->rson) - height(node->lson))//如果高度之差为2的话就失去了平衡,需要旋转 if (x>node->rson->data) SingRotateRight(node); else DoubleRotateRL(node); } else ++(node->freq);//如果相等,就把频率加1 node->hgt = Max(height(node->lson), height(node->rson)) + ; } //插入接口 void AVLTree::insert(int x,int &pre,int &aft) { insertpri(root, x,pre,aft); } map<int, int> lef, rig; int n; void init() { lef.clear(); rig.clear(); } int main() { int v,pre,aft; && n) { AVLTree tree; init(); scanf("%d", &v); tree.insert(v, pre, aft); vector<int> ans; ; i < n - ; i++) { scanf("%d", &v); pre = , aft = INF; tree.insert(v, pre, aft); //printf("pre:%d,aft:%d\n", pre, aft); ) { ans.push_back(aft); lef[aft] = ; } else { ans.push_back(pre); rig[pre] = ; } } ; i < ans.size() - ; i++) printf("%d ", ans[i]); printf(]); } ; }

 

目录
相关文章
|
存储 算法 数据库
Python高级数据结构——树(Tree)
Python高级数据结构——树(Tree)
477 1
|
存储 算法 Linux
打破常规,Linux内核新的数据结构上场maple tree(下)
打破常规,Linux内核新的数据结构上场maple tree
|
存储 分布式数据库 C语言
【初阶数据结构】树(tree)的基本概念——C语言
【初阶数据结构】树(tree)的基本概念——C语言
|
存储
数据结构之二叉查找树(Binary Search Tree)和红黑树(Red Black Tree)
二叉查找树又可以称之为 : 二叉搜索树 , 二叉排序树 , 它或者是一棵空树,或者是具有下列性质的二叉树:若它的左子树不空,则左子树上所有结点的值均小于它的根节点的值;若它的右子树不空,则右子树上所有结点的值均大于它的根结点的值;它的左、右子树也分别为二叉排序树。二叉搜索树作为一种经典的数据结构,它既有链表的快速插入与删除操作的特点,又有数组快速查找的优势 , 下图中这棵树,就是一棵典型的二叉查找树
153 1
|
5月前
|
存储 算法 Python
Python数据结构新视角:Trie树与Suffix Tree的相爱相杀,你站哪边?
【7月更文挑战第20天】在编程领域,Trie树(前缀树)与Suffix Tree(后缀树)犹如双星,各有专长。Trie树高效检索字符串集合,擅长前缀匹配,适用于自动补全和拼写检查;Suffix Tree则管理字符串所有后缀,加速子串查询,解最长公共前缀和重复子串难题。两者在不同场景发光发热,Trie树于快速响应的自动完成胜出,Suffix Tree则在基因序列分析和文本模式识别中独领风骚。抉择之间,应用场景与需求成关键,恰如剑客选剑,唯理解本质方能制胜。
41 1
|
5月前
|
存储 数据处理 开发者
告别繁琐查找!Python高级数据结构Trie树与Suffix Tree,让数据处理更轻松!
【7月更文挑战第19天】Python的Trie树优化字符串搜索,利用前缀减少无效操作,提升效率;Suffix Tree则高效处理后缀问题,尤其适用于文本搜索与生物信息学。虽构建复杂,但能加速后缀查询。掌握这两种数据结构,能有效应对大规模数据挑战,简化处理流程,提升开发效率。
113 0
|
6月前
|
存储 算法 测试技术
数据结构学习记录——树习题-Complete Binary Search Tree(题目描述、输入输出示例、数据结构的选择、核心算法、计算左子树的规模)
数据结构学习记录——树习题-Complete Binary Search Tree(题目描述、输入输出示例、数据结构的选择、核心算法、计算左子树的规模)
79 1
|
6月前
|
测试技术 C语言
数据结构学习记录——树习题—Tree Traversals Again(题目描述、输入输出示例、解题思路、解题方法C语言、解析)
数据结构学习记录——树习题—Tree Traversals Again(题目描述、输入输出示例、解题思路、解题方法C语言、解析)
48 1
|
7月前
|
存储 关系型数据库 MySQL
数据结构---B+Tree
数据结构---B+Tree
49 1
|
7月前
|
算法 索引 Python
Python高级数据结构——线段树(Segment Tree)
Python高级数据结构——线段树(Segment Tree)
294 2