POJ 2255 Tree Recovery

          发现自己懒得思考,不能一直刷一眼就看出结果的水题吧?

           思路:

           根据二叉树前中后序遍历的特性

                 

1、前序遍历的第一个字母必是 根

2、在中序遍历的字母串中找出 根字母,那么根字母左右两边的字符串就分别是它的左、右子树

3、利用递归复原二叉树(把子树看作新的二叉树)

4、后序遍历特征:后序遍历字母串 自右至左 依次为:

最外层(总树,设为第0层)右子树的根,内1层右子树的根,内2层右子树的根….内n层右子树的根,内n层左子树的根,内n-1层左子树的根……内1层左子树的根,最外层(总树,第0层)左子树的根。把总树的左子树作为新的总树,继续递归即可。   (注意:总树的叶就是作为“单叶”这棵树本身的右根)

5、输出后序遍历时,只需按4的顺序从左到右排列,再倒置输出即可


 

#include<iostream>
#include<cstring>
using namespace std;
char post[26];
int point = 0;
void right_to_left(char preo[],char inor[])
{
	post[point++] = preo[0];
	const int length = strlen(inor);
	if(length == 1)
	     return ;
    int j = 0;
    for(;j<length;j++)
        if(inor[j] == preo[0])
            break;
    const int flag = j;
    int i = ++j;
    
    char inor_temp[26],preo_temp[26];
    bool tag = false;//tag标记时代、否有左or右子树
    
    for(j=0;i<length;++i,++j)
    {
	    inor_temp[j] = inor[i];
	    tag = true;
	}
	
	if(tag)
	{
		inor_temp[j] = '\0';
		for(i = 0,j=length-j;j<length;++i,++j)
		    preo_temp[i] = preo[j];
        preo_temp[i] = '\0';//加上字符串结束标志,不然下一次调用strlen()函数将不能成功返回长度
        right_to_left(preo_temp,inor_temp);
    }
    
    tag = false;
    for(i = 0;i<flag;i++)
    {
    	inor_temp[i] = inor[i];
    	tag = true;
    }
    
	if(tag)
	{
	     inor_temp[i] = '\0'	;
	     for(i=0,j=1;i<flag;++i,++j)
	     {
	     	preo_temp[i] = preo[j];
	     }
	     preo_temp[i] = '\0';
	 
	 right_to_left(preo_temp,inor_temp);
    }
    return ;
}

int main(void)
{
	char preo[26],inor[26];
	while(cin>>preo>>inor)
	{
		right_to_left(preo,inor);
		for(int i=--point;i>=0;i--)
		    printf("%c",post[i]);
        printf("\n");
    }
    return  0;
}
    


转自: 優YoU http://user.qzone.qq.com/289065406/blog/1299063032

 


posted @ 2013-08-05 20:04  jlins  阅读(197)  评论(0编辑  收藏  举报