zoukankan      html  css  js  c++  java
  • Codeforces Round #368 (Div. 2) D. Persistent Bookcase

    D - Persistent Bookcase

    题目大意:有三种操作,第一种操作,在i 个书架,j个位置放一本书,第二种操作,取下第i个书架,j 个位置的书,第三种操作,

    把第i个书架所有没有书的位置放上书,有书的位置拿掉数,第四种操作,回到第k个操作,问你每次操作后书的总数。

    两种方法:1、离线之后,建树,对于op不等于4的操作i,建一条i-1 到 i的边,对于op==4的造作i,建一条k 到 i的边,书上跑一遍dfs

    得出答案,回溯的时候恢复状态。

    2、无脑主席树,需要注意的是要用bitset优化一下空间,不然会爆炸。。。。

    离线

    #include<bits/stdc++.h>
    #define fi first
    #define se second
    #define pb push_back
    #define mk make_pair
    #define pii pair<int,int>
    #define read(x) scanf("%d",&x)
    #define lread(x) scanf("%lld",&x)
    
    using namespace std;
    
    typedef long long ll;
    const int N=2e3+7;
    const int M=1e5+7;
    const int inf=0x3f3f3f3f;
    const ll INF=0x3f3f3f3f3f3f3f3f;
    
    struct node
    {
        int op,r,c;
    }qus[M<<1];
    int n,m,q,all,ans[M],sum[N],cnt[N][N],pos[M];
    bool flag[N];
    vector<int> e[M];
    void change(int x)
    {
        int op=qus[x].op;
        int r=qus[x].r,c=qus[x].c;
        if(op==1)
        {
            if(flag[r] && cnt[r][c]==1)
                all++,sum[r]++,cnt[r][c]=0;
            else if(!flag[r] && !cnt[r][c])
                all++,sum[r]++,cnt[r][c]=1;
        }
        else if(op==2)
        {
            if(flag[r] && !cnt[r][c])
                all--,sum[r]--,cnt[r][c]=1;
            else if(!flag[r] && cnt[r][c]==1)
                all--,sum[r]--,cnt[r][c]=0;
        }
        else if(op==3)
        {
            int pre=sum[r];
            sum[r]=m-sum[r];
            all+=sum[r]-pre;
            flag[r]=!flag[r];
        }
    }
    void dfs(int u)
    {
        bool flag=true;
        int p=all;
        if(u!=0)
            change(u);
        if((qus[u].op==1 || qus[u].op==2) && all==p)
            flag=false;
        ans[u]=all;
        for(int v:e[u]) dfs(v);
        if(u!=0 && flag)
            change(u+q);
    }
    int main()
    {
        memset(pos,-1,sizeof(pos));
        read(n); read(m); read(q);
        for(int i=1;i<=q;i++)
        {
            int op,r,c=0;
            read(op); read(r);
            if(op-3<0) read(c);
            qus[i]={op,r,c};
            if(op==1)
                qus[i+q]={2,r,c};
            else if(op==2)
                qus[i+q]={1,r,c};
            else if(op==3)
                qus[i+q]={3,r,c};
            else
                qus[i+q]={4,r,c};
            if(qus[i].op!=4)
                e[i-1].push_back(i);
            else
                e[r].push_back(i);
        }
        dfs(0);
        for(int i=1;i<=q;i++)
            printf("%d
    ",ans[i]);
        return 0;
    }
    /*
    */
    View Code

    在线主席树

    #include<bits/stdc++.h>
    #define fi first
    #define se second
    #define pb push_back
    #define mk make_pair
    #define pii pair<int,int>
    #define read(x) scanf("%d",&x)
    #define lread(x) scanf("%lld",&x)
    
    using namespace std;
    
    typedef long long ll;
    const int N=1e3+1;
    const int M=1e5+1;
    const int inf=0x3f3f3f3f;
    const ll INF=0x3f3f3f3f3f3f3f3f;
    
    int n,m,q;
    struct seg_tree
    {
        int tot,root[M];
        struct node
        {
            bitset<1005> b;
            bool filp;
            int sum,l,r;
        }a[M*20];
        void up(int x,int l,int r){
            a[x].sum=a[l].sum+a[r].sum;
        }
        void updata(int l,int r,int &x,int y,int pos,int num,int op)
        {
            a[++tot]=a[y]; x=tot;
            if(l==r)
            {
                if(op==1)
                {
                    if(!a[x].filp && !a[x].b[num])
                        a[x].b[num]=true,a[x].sum++;
                    else if(a[x].filp && a[x].b[num])
                        a[x].b[num]=false,a[x].sum++;
                }
                else if(op==2)
                {
                    if(!a[x].filp && a[x].b[num])
                        a[x].b[num]=false,a[x].sum--;
                    else if(a[x].filp && !a[x].b[num])
                        a[x].b[num]=true,a[x].sum--;
                }
                else if(op==3)
                {
                    a[x].sum=m-a[x].sum;
                    a[x].filp=!a[x].filp;
                }
                return;
            }
            int mid=(l+r)>>1;
            if(pos<=mid)
                updata(l,mid,a[x].l,a[y].l,pos,num,op);
            else
                updata(mid+1,r,a[x].r,a[y].r,pos,num,op);
            up(x,a[x].l,a[x].r);
        }
    }seg;
    int main()
    {
        read(n); read(m); read(q);
        for(int i=1;i<=q;i++)
        {
            int op,r,c=0;
            read(op); read(r);
            if(op<=2) read(c);
            if(op!=4)
                seg.updata(1,n,seg.root[i],seg.root[i-1],r,c,op);
            else
                seg.updata(1,n,seg.root[i],seg.root[r],r,c,op);
            printf("%d
    ",seg.a[seg.root[i]].sum);
        }
        return 0;
    }
    /*
    */
    View Code
  • 相关阅读:
    成功的软件开发过程 迭代,进化和敏捷
    将动态aspx页面转换成为静态html页面的几种方法
    大社区型网站的架构总结之如何标示用户在使用哪个功能程序篇(草稿)
    如果你觉得能做项目经理,就不要屈就自己去做普通程序员
    软件测试人的职业发展
    软件测试工程师,应该掌握的技能
    内存分配的字节对齐
    约瑟夫问题循环链表
    Linux字符设备驱动结构
    抵制日货(中国人一年购买日本货的钱,足够维持小鬼子2年半的军费)
  • 原文地址:https://www.cnblogs.com/CJLHY/p/8461929.html
Copyright © 2011-2022 走看看