CF707D Persistent Bookcase

洛谷评测传送门

题目描述

Recently in school Alina has learned what are the persistent data structures: they are data structures that always preserves the previous version of itself and access to it when it is modified.

After reaching home Alina decided to invent her own persistent data structure. Inventing didn't take long: there is a bookcase right behind her bed. Alina thinks that the bookcase is a good choice for a persistent data structure. Initially the bookcase is empty, thus there is no book at any position at any shelf.

The bookcase consists of nn shelves, and each shelf has exactly mm positions for books at it. Alina enumerates shelves by integers from 11 to nn and positions at shelves — from 11 to mm . Initially the bookcase is empty, thus there is no book at any position at any shelf in it.

Alina wrote down qq operations, which will be consecutively applied to the bookcase. Each of the operations has one of four types:

  • 11 ii jj — Place a book at position jj at shelf ii if there is no book at it.
  • 22 ii jj — Remove the book from position jj at shelf ii if there is a book at it.
  • 33 ii — Invert book placing at shelf ii . This means that from every position at shelf ii which has a book at it, the book should be removed, and at every position at shelf ii which has not book at it, a book should be placed.
  • 44 kk — Return the books in the bookcase in a state they were after applying kk -th operation. In particular, k=0k=0 means that the bookcase should be in initial state, thus every book in the bookcase should be removed from its position.

After applying each of operation Alina is interested in the number of books in the bookcase. Alina got 'A' in the school and had no problem finding this values. Will you do so?

输入格式

The first line of the input contains three integers nn , mm and qq ( 1<=n,m<=10^{3}1<=n,m<=103 , 1<=q<=10^{5}1<=q<=105 ) — the bookcase dimensions and the number of operations respectively.

The next qq lines describes operations in chronological order — ii -th of them describes ii -th operation in one of the four formats described in the statement.

It is guaranteed that shelf indices and position indices are correct, and in each of fourth-type operation the number kk corresponds to some operation before it or equals to 00 .

输出格式

For each operation, print the number of books in the bookcase after applying it in a separate line. The answers should be printed in chronological order.

题意翻译

题目大意

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

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

输入输出样例

输入 #1复制

输出 #1复制

输入 #2复制

输出 #2复制

输入 #3复制

输出 #3复制

说明/提示

This image illustrates the second sample case.

题解:

2019.10.22模拟赛T2 50分场感谢出题人@zcs0724

PS:洛谷题目翻译有锅,操作1是将\((i,j)\)置\(1\),操作2是将\((i,j)\)置\(0\)。如果不看\(CF\)原题的小伙伴肯定会在这里挂掉,请求管理修改。

考场上一看,操作\(1,2,3\)都会,但是操作\(4\)毫无思路。所以这题挂了,还好出题人良心,有\(50\%\)的数据只有操作\(1,2,3\),所以让蒟蒻侥幸骗得\(50pts\)。

这道题的难点就是操作4的处理。对于操作4,我们发现是对其历史的一个回溯,那么就想到了做法1:可持久化数据结构。

但是蒟蒻并不会可持久化数据结构,所以只学了解法2:离线+操作树

一开始操作树这么高端的名词把蒟蒻搞蒙了。(难道,这是比可持久化树套树更难的知识点?瑟瑟发抖)但是后来仔细地揣摩一下思路,发现就是把先离线,输入进来所有询问,然后把操作建成一棵树,在树上进行处理(本题是深搜)的过程。

为什么这么做是对的呢?因为我们的操作4回溯到的是一个操作和其对应的状态。那么,用一个树状的数据结构来存操作次数显然是满足要求的,就拿样例1来举例:

当我们好好地进行1、2操作的时候,第三次操作4回到了操作0(即没有任何操作的情况) ,所以我们依据题意建立了这样一个操作树。

为什么我们在树上进行深搜对答案没有影响、能维护答案的正确性呢?这是深搜的性质决定的。学过深搜的小伙伴知道深搜有"搜索"和“还原现场”两大核心操作。搜索能帮我们得出每一个操作的答案,而还原现场就能保证这些答案互不影响。那么我们离线处理询问构建了这个操作树,这个树上的每个操作只会是操作1、操作2、操作3(操作4就是建这棵树)。那么我们就可以依据题意进行模拟了。

当然,在这里还为大家介绍简洁地解决这个题模拟部分的一个技巧:\(bitset\)容器。这是一种能够方便地处理01串操作类型题的\(STL\)大法。本蒟蒻也是用这个东西完成的对操作1、2、3的模拟。如果对\(bitset\)不太熟悉的小伙伴请移步本蒟蒻的这篇博客:

浅谈C++bitset容器

代码如下:

#include<cstdio>
#include<bitset>
using namespace std;
const int maxn=1010;
const int maxq=1e5+10;
int n,m,q;
int ans[maxq],opt[maxq],x[maxq],y[maxq],cnt;
bitset<maxn> map[maxn],all_one;
int tot,head[maxq],nxt[maxq],to[maxq];
void add(int x,int y)
{
to[++tot]=y;
nxt[tot]=head[x];
head[x]=tot;
}
void dfs(int pos)
{
bool flag=0;
if(opt[pos]==1 && map[x[pos]][y[pos]]==0)
{
flag=1;
map[x[pos]][y[pos]]=1;
cnt++;
}
if(opt[pos]==2 && map[x[pos]][y[pos]]==1)
{
flag=1;
map[x[pos]][y[pos]]=0;
cnt--;
}
if(opt[pos]==3)
{
flag=1;
cnt-=map[x[pos]].count();
map[x[pos]]^=all_one;
cnt+=map[x[pos]].count();
}
ans[pos]=cnt;
for(int i=head[pos];i;i=nxt[i])
dfs(to[i]);
if(flag)//flag就是还原现场的操作,如果某一个节点是操作4,那么直接继续就可以
{
if(opt[pos]==1)
cnt--,map[x[pos]][y[pos]]=0;
if(opt[pos]==2)
cnt++,map[x[pos]][y[pos]]=1;
if(opt[pos]==3)
{
cnt-=map[x[pos]].count();
map[x[pos]]^=all_one;
cnt+=map[x[pos]].count();//数1,取反,统计答案
}
}
}
int main()
{
scanf("%d%d%d",&n,&m,&q);
for(int i=1;i<=m;i++)
all_one.set(i);//构造一个全1的数列,为操作三所用
for(int i=1;i<=q;i++)
{
scanf("%d%d",&opt[i],&x[i]);
if(opt[i]==4)
add(x[i],i);
else
if(opt[i]<3)
scanf("%d",&y[i]);
if(opt[i]<4)
add(i-1,i);
}
dfs(0);
for(int i=1;i<=q;i++)
printf("%d\n",ans[i]);
return 0;
}

CF707D Persistent Bookcase的更多相关文章

  1. cf707D. Persistent Bookcase(离线+dfs)

    题目链接:http://codeforces.com/problemset/problem/707/D  有一个n*m的书架,有K个操作,求每个操作后一共有多少本书:有4种操作: 1:x y 如果 x ...

  2. CF707D Persistent Bookcase 可持久化线段树

    维护一个二维零一矩阵(n,m<=1000),支持四种操作(不超过10^5次): 将(i,j)置一 将(i,j)置零 将第i行零一反转yu 回到第K次操作前的状态 每次操作后输出全局一共有多少个一 ...

  3. CodeForces #368 div2 D Persistent Bookcase DFS

    题目链接:D Persistent Bookcase 题意:有一个n*m的书架,开始是空的,现在有k种操作: 1 x y 这个位置如果没书,放书. 2 x y 这个位置如果有书,拿走. 3 x 反转这 ...

  4. 【Codeforces-707D】Persistent Bookcase DFS + 线段树

    D. Persistent Bookcase Recently in school Alina has learned what are the persistent data structures: ...

  5. Codeforces Round #368 (Div. 2) D. Persistent Bookcase

    Persistent Bookcase Problem Description: Recently in school Alina has learned what are the persisten ...

  6. Persistent Bookcase

    Persistent Bookcase time limit per test 2 seconds memory limit per test 512 megabytes input standard ...

  7. Codeforces Round #368 (Div. 2) D. Persistent Bookcase 离线 暴力

    D. Persistent Bookcase 题目连接: http://www.codeforces.com/contest/707/problem/D Description Recently in ...

  8. codeforces 707D D. Persistent Bookcase(dfs)

    题目链接: D. Persistent Bookcase time limit per test 2 seconds memory limit per test 512 megabytes input ...

  9. [CF707D]Persistent Bookcase_主席树_bitset

    Persistent Bookcase 题目链接:http://codeforces.com/contest/707/problem/D 注释:略. 题解: 发现虽然$q\le 10^5$但是网格是$ ...

随机推荐

  1. Leetcode 216. 组合总和 III

    地址 https://leetcode-cn.com/problems/combination-sum-iii/ 找出所有相加之和为 n 的 k 个数的组合.组合中只允许含有 1 - 9 的正整数,并 ...

  2. NOIP 2012 文化之旅

    洛谷 P1078 文化之旅 洛谷传送门 JDOJ 1788: [NOIP2012]文化之旅 T4 JDOJ传送门 Description Input Output Sample Input Input ...

  3. 2019.08.06模拟赛T2

    题目大意: 已知三个$n$位二进制数$A$,$B$,$C$. 满足: $A+B=C$ 它们二进制位中$1$的个数分别为$a$,$b$,$c$. 求满足条件的最小的$C$. Solution 唉,又是一 ...

  4. React中引用CSS样式的方法

    相对于html中引用css的三种方法,react中也有三种方法,一一相对: 1. 行内样式:直接在组件内部定义 <div style={{width:'20px',height:'30px'}} ...

  5. 关闭Chrome浏览器的广告

    生活没有绝对的对与错:代码就不一样了,错了就编译不过,也正是因为这样,编程的人思维有时也会陷入一种狭隘中,这就是把工作和生活没有分开.Win10 右下角的广告就像程序调试中的"警告" ...

  6. csrf攻击与csrf防御

    CSRF(Cross-site request forgery)跨站请求伪造,也被称为“One Click Attack”或者Session Riding,通常缩写为CSRF或者XSRF,是一种对网站 ...

  7. angularjs 一篇文章看懂自定义指令directive

     壹 ❀ 引 在angularjs开发中,指令的使用是无处无在的,我们习惯使用指令来拓展HTML:那么如何理解指令呢,你可以把它理解成在DOM元素上运行的函数,它可以帮助我们拓展DOM元素的功能.比如 ...

  8. 【TCP/IP网络编程】:02套接字类型与协议设置

    本篇文章主要介绍创建套接字函数相关的3个输入参数的含义,它们最终确定了套接字通信所采用的协议.同时,也简单对比了TCP和UDP传输方式的区别. 什么是协议?协议是对话中使用的通信规则,而在计算机领域则 ...

  9. java.lang.ClassNotFoundException: XXX (no security manager: RMI class loader disabled)

    在搞RMI远程发布,consumer去获取rmi远程服务的代理对象的时候出现了如下的错误 问题发现: 由于我发布的对象的包路径和获取的对象的包路径不一致,导致了这样的问题 解决方案: 包路径改为一致就 ...

  10. Z从壹开始前后端分离【 .NET Core2.2/3.0 +Vue2.0 】框架之八 || API项目整体搭建 6.3 异步泛型仓储+依赖注入初探

    本文梯子 本文3.0版本文章 回顾 1.Sqlsugar 的使用 2.修改数据连接字符串 今天要完成的浅紫色部分 一.设计仓储基类接口——IBaseRepository.cs 二.将其他的仓储接口,继 ...