树的建立与遍历

树在计算机科学中是一种非常重要的数据结构,常被用于搜索、排序、存储等领域。

树的建立可以通过定义节点结构,不断向树中添加新的节点进行。以下是一个简单的树节点的定义:

struct TreeNode {
    int val;
    struct TreeNode *left;
    struct TreeNode *right;
};

其中,val 存放节点的值,left 和 right 分别指向左子树和右子树的根节点。在创建新节点时,可以使用以下代码:

struct TreeNode *createNode(int val) {
    struct TreeNode *node =new TreeNode;
    node->val = val;
    node->left = NULL;
    node->right = NULL;
    return node;
}

树的遍历有三种方法:前序遍历、中序遍历和后序遍历,分别按照不同的顺序访问树中的节点。以下是这三种遍历方法的 C 代码实现:

void preorder(struct TreeNode *root) {
    if (root != NULL) {
        printf("%d ", root->val); // 访问根节点
        preorder(root->left); // 前序遍历左子树
        preorder(root->right); // 前序遍历右子树
    }
}

void inorder(struct TreeNode *root) {
    if (root != NULL) {
        inorder(root->left); // 中序遍历左子树
        printf("%d ", root->val); // 访问根节点
        inorder(root->right); // 中序遍历右子树
    }
}

void postorder(struct TreeNode *root) {
    if (root != NULL) {
        postorder(root->left); // 后序遍历左子树
        postorder(root->right); // 后序遍历右子树
        printf("%d ", root->val); // 访问根节点
    }
}

其中,前序遍历先访问根节点再访问左子树和右子树,中序遍历先访问左子树再访问根节点和右子树,后序遍历先访问左子树再访问右子树和根节点。

还有一种遍历的方法是层次遍历,层次遍历与以上三种方法不一样,层次遍历需要队列来实现。

void levelOrder(struct TreeNode* root) {
    if (root == NULL) return;
    
    struct TreeNode *queue[1000];
    int front = 0, rear = 0;
    queue[rear++] = root; // 根节点入队
    
    while (front < rear) { // 队列不为空
        struct TreeNode *node = queue[front++];
        printf("%d ", node->val); // 访问队首节点
        
        if (node->left != NULL) {
            queue[rear++] = node->left; // 左子树入队
        }
        if (node->right != NULL) {
            queue[rear++] = node->right; // 右子树入队
        }
    }
}

该代码中,使用了一个队列来保存待访问的节点。首先将根节点入队,然后开始循环,每次访问队首节点并将其出队,然后把它的左右子树入队。继续进行下一轮循环,直到队列为空为止。

P1827 [USACO3.4] 美国血统 American Heritage

这个题是给出前中遍历让你输出后序遍历,我先还原树然后在输出树的后序遍历

前序遍历是先遍历根节点,再遍历根节点的左右子树。

那么,前序序列的第一个节点,一定是根节点。

找到根节点,再确定根节点在中序序列中的位置,就可以分出左右两棵子树。

这道题我们不需要建树,只要通过递归不断切割字符串就好了。

#include<stdio.h>
#include<string.h>
void build(int m,char s2[],char s1[])
{
    if(m<=0)
        return;
    int p=strchr(s1,s2[0])-s1;//找到根结点,在中序遍历的位置
    build(p,s2+1,s1);//构造左子树的后序遍历
    build(m-p-1,s2+p+1,s1+p+1);//构造右子树的后序遍历
    printf("%c",s2[0]); 
     
}
int main()
{
    int i,n,m,j,k;
    char s1[30],s2[30],t;
    while(scanf("%s%s",&s1,&s2)!=EOF)
    {
        n=strlen(s1);
        m=strlen(s2);
        t=s2[0];
        build(m,s2,s1);
        printf("\n");   
    }
    return 0;
}

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值