zoukankan      html  css  js  c++  java
  • 【u103】绘制二叉树

    Time Limit: 1 second
    Memory Limit: 128 MB

    【问题描述】

    二叉树是一种基本的数据结构,它要么为空,要么由根节点,左子树和右子树组成,同时左子树和右子树也分别是二叉树。 当一颗二叉树高度为m-1时,则共有m层。除m层外,其他各层的结点数都达到最大,且结点节点都在第m层时,这就是一个满二叉树。 现在,需要你用程序来绘制一棵二叉树,它由一颗满二叉树去掉若干结点而成。 对于一颗满二叉树,我们需要按照以下要求绘制: 1、结点用小写字母“o”表示,对于一个父亲结点,用“/”连接左子树,同样用“”连接右子树。 2、定义[i,j]为位于第i行,第j列的某个字符。若[i,j]为“/”,那么[i-1,j+1]与[i+1,j-1]要么为“o”,要么为“/”。若[i,j]为“”,那么[i-1,j-1]与[i+1,j+1]要么为“o”,要么为“”。同样,若[i,j]为第1-m层的某个节点(即“o”),那么[i+1,j-1]为“/”,[i+1,j+1]为“”。 3、对于第m层节点也就是叶子结点,若两个属于同一个父亲,那么它们之间由3个空格隔开,若两个结点相邻但不属于同一个父亲,那么它们之间由1个空格隔开。第m层左数第1个节点之前没有空格。 最后需要在一颗绘制好的满二叉树上删除n个结点(包括它的左右子树,以及与父亲的连接),原有的字符用空格替换(ASCII 32,请注意空格与ASCII 0的区别(若用记事本打开看起来是一样的,但是评测时会被算作错误答案!))。

    【输入格式】

    输入文件binary.in的第1行包含2个正整数m和n,为需要绘制的二叉树层数已经从m层满二叉树中删除的结点数。 接下来n行,每行两个正整数,表示第i层第j个结点需要被删除(1
    【输出格式】

    输出文件binary.out为按照题目要求绘制的二叉树。

    【数据规模】

    30%的数据满足:n=0; 50%的数据满足:2<=m<=5; 100%的数据满足:2<=m<=10,0<=n<=10。 0

    Sample Input1
    
    2 0
    
    
    
    
    
    
    Sample Output1
    
      o  
     /  
    o   o
    
    Sample Input1
    
    4 3
    3 2
    4 1
    3 4
    
    
    
    
    Sample Output1
    
               o           
              /           
             /            
            /             
           /              
          /               
         o           o     
        /           /      
       /           /       
      o           o        
                /        
        o       o   o  
    
    【样例说明】

    【题目链接】:http://noi.qz5z.com/viewtask.asp?id=u103

    【题解】

    有点小恶心的题;
    首先层与层之间的那个间距以及’/’的个数从第下往上数第二层开始有ai=2*ai-1 +1的关系;(我也不知道为什么是从第二层开始);
    然后根节点离最左边的距离为3<<(m-2);
    然后整个树占据的范围是1..3<<(m-1)-1;(宽度);
    高度的话用两层之间的距离和加m就能算出来;
    然后对于一开始输入的n个点,最后在进行绘制的时候不要理它们就好;
    绘制当然得递归绘制;

    o   o
      /
      o   o
        /
        o
    从下往上怎样判断它的左儿子是i+1层的第几个节点?
    只要看一下当前节点是第i层的第几个节点x即可,如果往左就是x*2-1,往右边就是x*2;它们分别表示了我们接下来要到第i+1层的第几个节点;


    【完整代码】

    #include <cstdio>
    #include <cstdlib>
    #include <cmath>
    #include <set>
    #include <map>
    #include <iostream>
    #include <algorithm>
    #include <cstring>
    #include <queue>
    #include <vector>
    #include <stack>
    #include <string>
    using namespace std;
    #define lson l,m,rt<<1
    #define rson m+1,r,rt<<1|1
    #define LL long long
    #define rep1(i,a,b) for (int i = a;i <= b;i++)
    #define rep2(i,a,b) for (int i = a;i >= b;i--)
    #define mp make_pair
    #define pb push_back
    #define fi first
    #define se second
    
    typedef pair<int,int> pii;
    typedef pair<LL,LL> pll;
    
    void rel(LL &r)
    {
        r = 0;
        char t = getchar();
        while (!isdigit(t) && t!='-') t = getchar();
        LL sign = 1;
        if (t == '-')sign = -1;
        while (!isdigit(t)) t = getchar();
        while (isdigit(t)) r = r * 10 + t - '0', t = getchar();
        r = r*sign;
    }
    
    void rei(int &r)
    {
        r = 0;
        char t = getchar();
        while (!isdigit(t)&&t!='-') t = getchar();
        int sign = 1;
        if (t == '-')sign = -1;
        while (!isdigit(t)) t = getchar();
        while (isdigit(t)) r = r * 10 + t - '0', t = getchar();
        r = r*sign;
    }
    
    const int MAXM = 12;
    const int dx[5] = {0,1,-1,0,0};
    const int dy[5] = {0,0,0,-1,1};
    const double pi = acos(-1.0);
    
    int m,n,ma = 0,ma2 = 0;
    bool no[MAXM][1<<MAXM];
    char ans[1<<MAXM][1<<MAXM];
    
    void pri(int tx,int ty,int x,int y,int gap)
    {
        ma = max(ma,x);ma2 = max(ma2,y);
        int xx,yy;
        ans[x][y] = 'o';
        if (tx==m) return;
        xx = x,yy = y;
        if (!no[tx+1][2*ty-1])
        {
            rep1(i,1,gap)
                {
                    xx++;yy--;
                    ans[xx][yy]='/';
                }
            pri(tx+1,2*ty-1,xx+1,yy-1,gap/2);
        }
        xx = x,yy = y;
        if (!no[tx+1][2*ty])
        {
            rep1(i,1,gap)
            {
                xx++;yy++;
                ans[xx][yy] ='\';
            }
            pri(tx+1,2*ty,xx+1,yy+1,gap/2);
        }
    }
    
    int main()
    {
        //freopen("F:\rush.txt","r",stdin);
        memset(no,false,sizeof(no));
        rei(m);rei(n);
        int k;
        if (m==1)
            k=1;
        else
            k=3 << (m-1);
        rep1(i,1,n)
        {
            int x,y;
            rei(x);rei(y);
            no[x][y] = true;
            if (m==10 && y==512)
                break;
        }
        if (no[1][1] || m==1)
        {
            if (no[1][1])
                puts("");
            else
                puts("o");
            return 0;
        }
        rep1(i,1,k)
            rep1(j,1,k)
                ans[i][j] = ' ';
        int gap=2;
        if (m==2)
            gap=1;
        else
        {
            rep1(i,4,m)
                gap=gap*2+1;
        }
        int tgap = gap;
        pri(1,1,1,3 << (m-2),gap);
        int deep = 1;
        while (tgap>0)
        {
            deep+=tgap;
            deep+=1;
            tgap/=2;
        }
        //deep--;
        //cout << deep<<endl;
        rep1(i,1,deep)
            rep1(j,1,k-1)
                {
                    printf("%c",ans[i][j]);
                    if (j==k-1) puts("");
                }
        return 0;
    }
    
  • 相关阅读:
    Delphi WinAPI GetWindowRect
    Delphi WMI[2] 响应网线断开
    打印两个升序链表中共同的数据
    判断一个链表是否是回文结构
    删除有序数组中的重复项
    三数之和、最接近目标值的三数之和
    删除链表倒数第N个节点
    判断回文数
    字符串转整数
    整数反转
  • 原文地址:https://www.cnblogs.com/AWCXV/p/7626917.html
Copyright © 2011-2022 走看看