2. 树的序列

 

 

 

/*
***定义数据结构---根据问题描述 可知,需知道结点与哪些结点相邻, 因此可做下表 
结点编号               1            2             3           4           5          6          7           8
相邻结点               5         3,5,6,8          2           5           1,4       2,7         6            2
可根据此图构建  vector< set<int> > 类型 
***判断是否为叶子节点 
由图可知,当编号对应的相邻结点只有一个时,该结点为叶子结点
***如何删除最小结点
先将叶子结点加入到 一个升序队列中(升序集合也可)  ,删除最小结点,然后更新 存储的表,有新叶子结点则加入升序队列中
***读取数据  (2 (6 (7)) (3) (5 (1) (4)) (8))   递归实现 
由输入数据的特点,即 由括号决定优先级,且读入数据为树形结构,可使用递归, 当遇到'('即新数字时调用一层循环,当遇到')'结束一层循环,
由此可知,当 读入2后,递归调用函数,2作为一个结点传入,然后 读入6 将2,6 填入图中 ,此时 递归调用,6作为结点传入,然后读7,此时将6,7
填入图中,然后遇到')' ,程序退回至 2为结点。。。。 
*/
//   node 默认为0,第一个结点无上一层传入的结果,用node是否为0可判断 是否 该向图中填数据 
void read( vector< set<int> >& vec, int node = 0 ){
	int num;
	cin >> num;// 默认有 提前将数字前的 ( 给读取的操作 
	if( node ){// node 不为 0 说明当前程序 开始递归,即 可找到 相邻结点 
		vec[node].insert( num );// set 集合默认从小到大排序
		vec[num].insert( node );
	}// 完成填表 
	
	while( true ){// 开始循环找到 与 num 相邻的结点 
		char c;
		cin >> c;// 将数字前的 '(' 读取
		if( c == ')' ) {// 读到当前结点所有相邻点了 
			return ;
		}
		read( vec, num );// num 作为 下一层的相邻结点(可以说为父结点) 
	}
}



int main(){
	
	char c;
	cin >> c;// 读取数字前的括号 
	if( c ){// 如果无数据则结束 
		vector< set<int> > vec( 10000, set<int>() ) ;// 构建表,vector为动态数组,可自动扩展,默认给值防止扩展  
		read( vec );// 读取表
		priority_queue< int, vector<int>, greater<int> > leafs;// 优先队列 参数分别为 数据类型,容器,升序排列 // 采用堆排序实现,每次可出队最小值
		int cnt = 0;// 统计结点的个数 
		for( int i = 0; i < vec.size(); i++ ){// 将叶子结点加入优先队列 
			if( vec[i].size() ){
				cnt++;
				if( vec[i].size() == 1 ){
					leafs.push( i );
				}
			}
		}
		// 进行删除操作,具体思路,将编号i删除队列,然后将i相邻的结点对应的相邻点的集合 删除 i,判断该结点是否沦为叶子节点,是则加入队列
		for( int i = 1; i < cnt; i++ ) {// 删除 n-1个结点 
			int node = leafs.top();// 获取最小叶子结点的编号 
			leafs.pop();// 弹出 
			int t = *vec[node].begin();// 获取叶子结点的相邻结点的编号 
			cout << t << " ";// 打印出相邻结点编号 
			// 更新表
			vec[t].erase( node ); // 找到相邻结点,将当前边删除 
			if( vec[t].size() == 1 ){// 删除边后沦为叶子 
				leafs.push( t );
			}
		}
		cout << endl;
	}
	return 0;
}

 

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值