zoukankan      html  css  js  c++  java
  • Codeforces482B【线段树构造】

    题意:
    有M个限制,每个限制有l,r,q,表示从a[l]~a[r]取且后的数一定为q,问是否有满足的数列。
    思路:
    看到大牛说是线段树,线段树对于区间操作,印象中乘啊,+啊,-啊都不错,但是并没有就是对于这个位运算就不懂了;
    这题的题意就是构造,大致思路是
    每条限制是对于每个区间处理就是或上q(可以保证相应的二进制一定是1),然后用线段树处理完m个限制,最后还要询问一下m个限制是否满足;
    所以具体操作就是利用线段树进行区间或操作,区间查询且。
    PS:
    在最后取n个数的时候撒比了,直接取了树上的n个节点位置的数,= =、q[num].x和当left==right的num完全不一样的啊。。哎真踏马蠢了。。
    code…

    #include <bits/stdc++.h>
    using namespace std;
    typedef long long LL;
    
    const int N=1e5+10;
    struct asd{
        int left,right;
        int x;
    };
    asd q[N*4];
    
    struct nod{
        int a,b,c;
    };
    nod temp[N];
    
    void build(int num,int L,int R)
    {
        q[num].left=L;
        q[num].right=R;
        if(L==R)
        {
            q[num].x=0;
            return;
        }
        build(2*num,L,(L+R)/2);
        build(2*num+1,(L+R)/2+1,R);
        q[num].x=q[2*num].x^q[2*num+1].x;
    }
    
    void update(int num,int s,int t,int c)
    {
        if(s<=q[num].left&&t>=q[num].right)
        {
            q[num].x|=c;
            return;
        }
        int mid=(q[num].left+q[num].right)/2;
        if(mid>=t)
            update(2*num,s,t,c);
        else if(mid<s)
            update(2*num+1,s,t,c);
        else
        {
            update(2*num,s,mid,c);
            update(2*num+1,mid+1,t,c);
        }
    }
    int query(int num,int s,int t)
    {
        if(s<=q[num].left&&t>=q[num].right)
            return q[num].x;
        int mid=(q[num].left+q[num].right)/2;
        if(mid>=t)
            return query(2*num,s,t);
        else if(mid<s)
            return query(2*num+1,s,t);
        else
        {
            return query(2*num,s,mid)&query(2*num+1,mid+1,t);
        }
    }
    
    vector<int>ans;
    void solve(int num)
    {
        if(num!=1)
        {
            q[num].x|=q[num/2].x;
        }
        if(q[num].left==q[num].right)
        {
            ans.push_back(q[num].x);
            return;
        }
        solve(2*num);
        solve(2*num+1);
    }
    
    int main()
    {
        int a,b,c;
        int n,m;
        scanf("%d%d",&n,&m);
        build(1,1,n);
        for(int i=1;i<=m;i++)
        {
            scanf("%d%d%d",&a,&b,&c);
            update(1,a,b,c);
            temp[i].a=a;
            temp[i].b=b;
            temp[i].c=c;
        }
        bool flag=true;
        for(int i=1;i<=m;i++)
        {
            if(query(1,temp[i].a,temp[i].b)!=temp[i].c)
            {
                flag=false;
                break;
            }
        }
        solve(1);
        if(flag)
        {
            puts("YES");
            printf("%d",q[1].x);//我直接for一下输出和下面的在vector里面输出为啥顺序不一样。。废话。。踏马这个num是最终子节点么,卧槽。。
            for(int i=2;i<=n;i++)
            {
                printf(" %d",q[i].x);
            }
    
            printf("%d",ans[0]);
            for(int i=1;i<ans.size();i++)
            {
                printf(" %d",ans[i]);
            }
        }
        else
            puts("NO");
        return 0;
    }
  • 相关阅读:
    express 的安全中间件 helmet 简介
    一个设置过期时间的方案
    vscode 中的 vue 格式化
    linux 中部署不同版本 node.js 并同时使用的方案
    webpack 多页面模式配置
    1.assert
    我是一个线程(转)
    Android FragmentTransactionExtended:使Fragment以多种样式动画切换
    Android ORM应用开发框架KJFrameForAndroid使用详解
    Android Studio插件推荐(PreIOC,GsonFormat)
  • 原文地址:https://www.cnblogs.com/keyboarder-zsq/p/5934796.html
Copyright © 2011-2022 走看看