维护一个二维零一矩阵(n,m<=1000),支持四种操作(不超过10^5次):

  • 将(i,j)置一
  • 将(i,j)置零
  • 将第i行零一反转yu
  • 回到第K次操作前的状态
  • 每次操作后输出全局一共有多少个一

你发现如果每一次操作都复制一整行的话是可以用 $bitset$ 优化的,自带/32

所以,我们对于每一个时刻维护一个线段树,其中 $i$ 节点表示第 $i$ 行对应的 $bitset$ 编号.

对于前 $3$ 个操作,每一次操作时都暴力新建一个 bitset,然后每次在可持久化线段树上最多更新一个单点.

顺便在线段树的节点上再维护一个 $1$ 的数量即可.

#include <bits/stdc++.h>
#define M 1004
#define N 100005
#define setIO(s) freopen(s".in","r",stdin) , freopen(s".out","w",stdout)
using namespace std;
bitset<M>v[N],uni,oo;
int rt[N];
namespace seg
{
#define lson t[x].ls
#define rson t[x].rs
int tot;
int newnode() { return ++ tot; }
struct Node
{
int ls,rs,id,sum;
}t[N*40];
int update(int x,int l,int r,int p,int v)
{
int now=newnode();
t[now]=t[x];
if(l==r)
{
t[now].id=v;
return now;
}
int mid=(l+r)>>1;
if(p<=mid)
{
t[now].ls=update(lson,l,mid,p,v);
}
else
{
t[now].rs=update(rson,mid+1,r,p,v);
}
t[now].sum=t[t[now].ls].sum+t[t[now].rs].sum;
return now;
}
int modify(int x,int l,int r,int p,int v)
{
int now=newnode();
t[now]=t[x];
if(l==r)
{
t[now].sum=v;
return now;
}
int mid=(l+r)>>1;
if(p<=mid) t[now].ls=modify(lson,l,mid,p,v);
else t[now].rs=modify(rson,mid+1,r,p,v);
t[now].sum=t[t[now].ls].sum+t[t[now].rs].sum;
return now;
}
int query(int x,int l,int r,int p)
{
if(!x) return 0;
if(l==r) return t[x].id;
int mid=(l+r)>>1;
if(p<=mid) return query(lson,l,mid,p);
else return query(rson,mid+1,r,p);
}
void dfs(int x,int l,int r)
{
if(!x) return;
if(l==r) printf("%d %d\n",l,t[x].sum);
int mid=(l+r)>>1;
dfs(lson,l,mid), dfs(rson,mid+1,r);
}
#undef lson
#undef rson
};
int main()
{
// setIO("now");
int n,m,q,i,j;
scanf("%d%d%d",&n,&m,&q);
for(i=1;i<=m;++i) uni[i]=1;
for(i=1;i<=q;++i)
{
int op,a,b,c;
scanf("%d",&op);
rt[i]=rt[i-1];
if(op==1)
{
scanf("%d%d",&a,&b);
int pre=seg::query(rt[i-1],1,n,a);
v[i]=v[pre];
v[i][b]=1;
rt[i]=seg::update(rt[i-1],1,n,a,i);
oo=v[i]&uni;
rt[i]=seg::modify(rt[i],1,n,a,oo.count());
}
if(op==2)
{
scanf("%d%d",&a,&b);
int pre=seg::query(rt[i-1],1,n,a);
v[i]=v[pre];
v[i][b]=0;
rt[i]=seg::update(rt[i-1],1,n,a,i);
oo=v[i]&uni;
rt[i]=seg::modify(rt[i],1,n,a,oo.count());
}
if(op==3)
{
int x;
scanf("%d",&x);
int pre=seg::query(rt[i-1],1,n,x);
v[i]=v[pre]^uni;
rt[i]=seg::update(rt[i-1],1,n,x,i);
oo=v[i]&uni;
rt[i]=seg::modify(rt[i],1,n,x,oo.count());
}
if(op==4)
{
int x;
scanf("%d",&x);
rt[i]=rt[x];
}
printf("%d\n",seg::t[rt[i]].sum);
}
return 0;
}

  

CF707D Persistent Bookcase 可持久化线段树的更多相关文章

  1. CF707D Persistent Bookcase

    CF707D Persistent Bookcase 洛谷评测传送门 题目描述 Recently in school Alina has learned what are the persistent ...

  2. PYOJ 44. 【HNSDFZ2016 #6】可持久化线段树

    #44. [HNSDFZ2016 #6]可持久化线段树 统计 描述 提交 自定义测试 题目描述 现有一序列 AA.您需要写一棵可持久化线段树,以实现如下操作: A v p x:对于版本v的序列,给 A ...

  3. 【BZOJ-3673&3674】可持久化并查集 可持久化线段树 + 并查集

    3673: 可持久化并查集 by zky Time Limit: 5 Sec  Memory Limit: 128 MBSubmit: 1878  Solved: 846[Submit][Status ...

  4. 【BZOJ-2653】middle 可持久化线段树 + 二分

    2653: middle Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 1298  Solved: 734[Submit][Status][Discu ...

  5. HDU 4866 Shooting(持久化线段树)

    view code//第二道持久化线段树,照着别人的代码慢慢敲,还是有点不理解 #include <iostream> #include <cstdio> #include & ...

  6. 【BZOJ-3653】谈笑风生 DFS序 + 可持久化线段树

    3653: 谈笑风生 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 628  Solved: 245[Submit][Status][Discuss] ...

  7. 【BZOJ3673】&&【BZOJ3674】: 可持久化并查集 by zky 可持久化线段树

    没什么好说的. 可持久化线段树,叶子节点存放父亲信息,注意可以规定编号小的为父亲. Q:不是很清楚空间开多大,每次询问父亲操作后修改的节点个数是不确定的.. #include<bits/stdc ...

  8. 【BZOJ3207】花神的嘲讽计划I 可持久化线段树/莫队

    看到题目就可以想到hash 然后很自然的联想到可持久化权值线段树 WA:base取了偶数 这道题还可以用莫队做,比线段树快一些 可持久化线段树: #include<bits/stdc++.h&g ...

  9. 【BZOJ 3674】可持久化并查集加强版&【BZOJ 3673】可持久化并查集 by zky 用可持久化线段树破之

    最后还是去掉异或顺手A了3673,,, 并查集其实就是fa数组,我们只需要维护这个fa数组,用可持久化线段树就行啦 1:判断是否属于同一集合,我加了路径压缩. 2:直接把跟的值指向root[k]的值破 ...

随机推荐

  1. Python——继承

    Python的继承是多继承机制,一个子类可以同时有多个直接父类:继承可以得到父类定义的方法,子类就可以复用父类的方法. 一.继承的语法 子类:实现继承的类. 父类(基类.超类):被继承的类. 子类继承 ...

  2. Github相关问题集锦

    问题 对于我们国内用户,有时候浏览器在进入github网站时,会出现无法加载或加载很慢的问题,针对这一问题,很是头疼,为解决这一问题,在网上搜索到的解决方法如下: 解决方案 修改hosts文件,在ho ...

  3. c++ (1) c++ 与c 的区别

    可以说c++ 语言在c基础上扩展了许多  在学习玩c语言之后  学习c++ 会发现容易一些  但是c++也有优越于c 的地方 c++ 与c 语言都属于本地编译型语言 ,直接编译成本地编译码,运行特别快 ...

  4. Java基础IO类之File类

    大三了,目前基础太差了,重新学习过!代码如下,里面都有详细的解释每一行代码代表的意思~ package IODemo; import java.io.File; import java.io.File ...

  5. 上传图片,图片过大压缩处理以及解决自拍时会出现图片横屏的bug修复、长按保存图片

    js部分:module.exports = { resize: function (file, callback, options) { //配置 options = Object.assign({ ...

  6. Java中的File操作总结

    1.创建文件 import java.io.File; import java.io.IOException; public class CreateFileExample { public stat ...

  7. Django组件之modelform

    Django的model form组件 这是一个神奇的组件,通过名字我们可以看出来,这个组件的功能就是把model和form组合起来,先来一个简单的例子来看一下这个东西怎么用:比如我们的数据库中有这样 ...

  8. kali linux Desktop Environemt types and kali linux install virualbox

    1.we know the kali linux desktop environmet can also be costomized ,Desktop environmet can use GNOME ...

  9. Linux命令——uptime

    参考:linux中uptime命令查看linux系统负载 Linux uptime command 简介 uptim告诉你系统运行了多长时间.uptime命令提供单行显示的输出,包含如下信息: 当前时 ...

  10. JS基础篇【1】

    该文讲解适用于有一定语言开发基础的朋友们,亦可当作久别重逢之回顾! 1.JS简介 JavaScript 是互联网上最流行的脚本语言,这门语言可用于 HTML 和 web,更可广泛用于服务器.PC.笔记 ...