zoukankan      html  css  js  c++  java
  • POJ 2777

    传送门

    题目要求支持两个操作

    C x  y  v   将区间[x,y]内木块全染成v色。

    P x  y   回答区间[x,y]中不同颜色的个数。 

    思路:

    下放标记,然后进行状态压缩。

    对,就是这样。

    压缩的状态中表示当前段存在哪几种颜色。

    最后询问操作返回的是一个压缩的值。

    这个时候我们就要统计返回值的二进制中1的个数。

    可以用以下代码快速解决。

    while(x){

      count++;

      x = x&(x-1);

    }

    至于为什么?

    不赘述。

    下面帖代码,有问题留言。

    #include<cstdio>
    #include<algorithm>
    #define lson l,m,rt<<1
    #define rson m+1,r,rt<<1|1
    #define N 1000009
    using namespace std;
    
    int w[N<<2|1],add[N<<2|1];
    
    void update(int rt){            //左子树与右子树合并
        w[rt] = (w[rt<<1]|w[rt<<1|1]);
    }
    
    void build(int l,int r,int rt){
        if(l == r){
            w[rt] |= 1;         //初始化为1
            add[rt] = 0;
            return;
        }
        int m = (l+r)>>1;
        build(lson);
        build(rson);
        update(rt);
    }
    
    void color(int l,int r,int rt,int v){
        w[rt] &= 0;
        w[rt] |= (1<<(v-1));
        add[rt] = v;
    }
    
    void push_col(int l,int r,int rt){
        if(add[rt]){                    //0代表没有标记,否则是标记了,那么下放标记
            int m = (l+r)>>1;
            color(lson,add[rt]);
            color(rson,add[rt]);
            add[rt] = 0;
        }
    }
    
    void modify(int l,int r,int rt,int nowl,int nowr,int v){
        if(nowl <= l && r <= nowr){
            color(l,r,rt,v);
            return;
        }
        push_col(l,r,rt);
        int m = (l+r)>>1;
        if(nowl <= m)modify(lson,nowl,nowr,v);
        if(m < nowr)modify(rson,nowl,nowr,v);
        update(rt);
    }
    
    int query(int l,int r,int rt,int nowl,int nowr){
        if(nowl <= l && r <= nowr)return w[rt];
        push_col(l,r,rt);
        int m = (l+r)>>1;
        long long  ans = 0;
        if(nowl <= m) ans |= query(lson,nowl,nowr);     //左边与右边要合并
        if(m < nowr)ans |= query(rson,nowl,nowr);
        return ans;
    }
    
    int read(){                             //读入优化
        int x = 0;
        char ch = getchar();
        while(ch < '0' || ch > '9')ch = getchar();
        while(ch >= '0' && ch <= '9'){
            x =x * 10 + ch -'0';
            ch = getchar();
        }
        return x;
    }
    
    int main(){
        int n = read(),t = read(),m = read();
        build(1,n,1);
        while(m--){
            char cmd[2];
            scanf("%s",cmd);
            if(cmd[0] == 'C'){
                int x = read(),y = read(),v =read();
                if(x > y)swap(x,y);
                modify(1,n,1,x,y,v);
            }
            else {
                int x = read(),y =read();
                if(x > y)swap(x,y);
                int ans = query(1,n,1,x,y);
                int count = 0;
                while(ans){             //计算个数
                    count++;
                    ans = ans&(ans-1);
                }
                printf("%d
    ",count);
            }
        }
        return 0;
    }
  • 相关阅读:
    超实用的PHP代码片段
    推荐五款优秀的PHP代码重构工具
    PHP开发搜索引擎技术全解析
    怎样成为一名PHP专家?
    PHP中该怎样防止SQL注入?
    有关PHP 10条有用的建议
    fir.im Weekly
    可能是一场很 IN 的技术分享
    fir.im Weekly
    更新日志
  • 原文地址:https://www.cnblogs.com/bingdada/p/7768169.html
Copyright © 2011-2022 走看看