线段树专题 节点的含义

题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=1610
题目大意:在一条线上绘制一些彩色片段,一些先前绘制的片段可能被后续的一些片段覆盖。您的任务是计算最后可以看到的不同颜色的片段。

每个数据集的第一行恰好包含一个整数n,1 <= n <= 8000,等于彩色段的数量。以下n行中的每一行由正好由单个空格分隔的3个非负整数组成:x1 x2 c
x1和x2表示段的左端点和右端点,c表示段的颜色。

所有数字都在[0,8000]范围内,它们都是整数。输入可能包含多个数据集,处理到文件末尾。

输出:输出的每一行应包含一个颜色索引,可以从顶部看到,按照这种颜色的分段计数,它们应根据颜色索引打印。如果看不到某种颜色,则不应打印。
在每个数据集后打印一个空行。

之前遇到过数组含义的问题,这里如果对x, y区间染c色,UP(1, x, y, c)更新。
那么此时节点的含义是区间的端点。

所以到最后,知道端点的含义并不知道区间的颜色。就会认为最后只能看见两种颜色。

如果我们把节点i的含义改[i-1, i]区间的颜色,那么就可以了,UP(1, x+1, y, c)更新。

然后改完还是WA了。有个奇怪的错误,就是建树的时候,我建了8001个节点WA了。改成8000个就AC。不知道这是为什么。

思考:注意对数组和节点代表的含义。

#include<bits/stdc++.h>
#define ll long long
using namespace std;

#define MAX_NODE 800005
#define MAX_ 200005

struct NODE
{
    int l, r;

}node[MAX_NODE];
int f[MAX_];     //记录节点i的node结构体数组下标f[i]
int add[MAX_NODE];

void BT(int i, int l ,int r)   //建树
{
    node[i].l=l;
    node[i].r=r;
    if(l==r)                   //根节点
    {
        f[l]=i;
        return;
    }
    BT((i<<1), l, (l+r)/2);    //建左子树
    BT((i<<1)+1, (l+r)/2+1, r);//建右子树

}

void upadd(int i)
{
    if(add[i])
    {
        add[i<<1]=add[i];          //把左右子的add标记进行更新
        add[(i<<1)+1]=add[i];

        add[i]=0;                   //清零父节点的add标记
    }
}

void UP(int i, int l, int r, int c)//更新
{
    if(node[i].l==l&&node[i].r==r)//找到更新区间
    {
        add[i]=c;                //更新add标记

        return;
    }

    if(node[i].l==node[i].r)      //叶子节点
    {
        add[i]=c;
        return;
    }

    upadd(i);                     //把此本节点的add往下移动

    int mid=(node[i].l+node[i].r)/2;//继续寻找区间
    if(r<=mid)                    //全在左区间
    {
        UP(i<<1, l, r, c);
    }
    else if(l>mid)                //全在右区间
    {
        UP((i<<1)+1, l, r, c);
    }
    else
    {
        UP(i<<1, l, mid, c);
        UP((i<<1)+1, mid+1, r, c);
    }

}

void FD(int i, int l, int r)   //查找
{
    if(l==r)
    {
        return;
    }
    upadd(i);
    FD(i<<1, l, (l+r)/2);
    FD((i<<1)+1, (l+r)/2+1, r);

}
//建树 BT(1, 1 ,n);节点1-n
//查询 FD(1, a, b);[a, b]的信息
//更新 UP(1, a, b, c);//把区间的add+=a

int c[8001];
int main()
{
    int n;
    while(scanf("%d",&n)!=EOF)
    {
        memset(c, 0, sizeof(c));
        memset(add, 0, sizeof(add));
        BT(1, 1, 8000);
        for(int i=1;i<=n;i++)
        {
            int a, b, c;
            scanf("%d%d%d",&a,&b,&c);
            UP(1, a+1, b, c+1);
        }
        FD(1, 1, 8000);
        int ans=0;
        for(int i=1;i<=8000;i++)
        {
            if(add[f[i]]!=add[f[i-1]]&&add[f[i]]!=0)
            {
                c[add[f[i]]-1]++;
            }
        }
        for(int i=0;i<=8000;i++)
        {
            if(c[i]!=0)
            {
                printf("%d %d\n",i, c[i]);
            }
        }
        printf("\n");

    }

    return 0;
}
全部评论

相关推荐

2024-12-18 14:13
蚌埠坦克学院 golang
苏州科技大学:面试官:接个面试,对面同学是个杀软二次元
点赞 评论 收藏
分享
2024-12-26 20:46
复旦大学 C++
国棉17厂丶小王:拿了offer的那个周末晚上去网吧通宵,去网吧不知道玩什么刷了lc的每日一题,然后试着第一次打开了三角洲行动,从此少了一个已经刷了700道题的lc用户,但是烽火地带多了一只🐭🐭
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客企业服务