Description

link

题意:给一个全\(0\)矩阵,每次支持一个修改,修改不还原(这要是还原了不就成\(A\)题了)

然后询问每一次修改完了当前矩阵的连通块个数

每一个修改的值单调不降

修改次数 \(\leq 10^6\)

Solution

这个是一道并查集题(感觉我原来从来没有写过任何并查集维护信息的题目)

具体就是我们对于每一个修改要考虑的是这个修改带来的贡献,就是和相邻颜色的对比

能合并的就合并一下,然后统计答案

这时,我们把这个题转化成了对每一种颜色考虑,然后看这个颜色的改变(最后开个桶就成了)

对于每种颜色,会有两种:添加一个颜色(正序处理),颜色被覆盖(逆序处理)

逆序的原因是被覆盖的时候原先有的这个连通块可能被整成多个连通块

(如果我们逆序处理覆盖,就等同于正序处理添加,贡献相减即可)

真是一道并查集应用的不错题目

Code

#include<bits/stdc++.h>
using namespace std;
#define int long long
namespace yspm{
inline int read()
{
int res=0,f=1; char k;
while(!isdigit(k=getchar())) if(k=='-') f=-1;
while(isdigit(k)) res=res*10+k-'0',k=getchar();
return res*f;
}
const int N=310,Q=2e6+10;
int fx[4]={0,-1,1,0},fy[4]={-1,0,0,1};
int n,m,q,maxx,ans[Q],fa[N*N],now[N][N];
struct query{int id,x,y;}; vector<query>q1[Q],q2[Q];
inline int num(int x,int y){return (x-1)*m+y;}
inline int rt(int x){return fa[x]==x?x:fa[x]=rt(fa[x]);}
inline bool in(int x,int y){return x>0&&x<=n&&y>0&&y<=m;}
inline void clear(int n){for(int i=1;i<=n;++i) fa[i]=i; return ;}
inline bool merge(int x,int y){x=rt(x),y=rt(y); if(x==y) return 0; return fa[x]=y,1;}
signed main()
{
n=read(); m=read(); q=read();
for(int i=1,x,y,c;i<=q;++i)
{
x=read(),y=read(),c=read(); maxx=c;
q2[now[x][y]].push_back((query){i,x,y});
q1[now[x][y]=c].push_back((query{i,x,y}));
}
for(int i=1;i<=n;++i)
{
for(int j=1;j<=m;++j) q2[now[i][j]].push_back((query){0,i,j});
}memset(now,-1,sizeof(now));
for(int i=0,sz,id,x,y;i<=maxx;++i)
{
sz=q1[i].size(); if(!sz) continue; clear(n*m);
for(int j=0;j<sz;++j)
{
id=q1[i][j].id; x=q1[i][j].x; y=q1[i][j].y;
now[x][y]=i; ++ans[id];
for(int k=0;k<4;++k)
{
int tx=x+fx[k],ty=y+fy[k];
if(in(tx,ty)&&now[tx][ty]==i) ans[id]-=merge(num(x,y),num(tx,ty));
}
}
}memset(now,-1,sizeof(now));
for(int i=0,sz,id,x,y;i<=maxx;++i)
{
sz=q2[i].size(); if(!sz) continue; clear(n*m);
for(int j=sz-1;j>=0;--j)
{
id=q2[i][j].id; x=q2[i][j].x; y=q2[i][j].y;
now[x][y]=i; --ans[id];
for(int k=0;k<4;++k)
{
int tx=x+fx[k],ty=y+fy[k];
if(in(tx,ty)&&now[tx][ty]==i) ans[id]+=merge(num(x,y),num(tx,ty));
}
}
}
ans[0]=1; for(int i=1;i<=q;++i) printf("%lld\n",ans[i]+=ans[i-1]);
return 0;
}
}
signed main(){return yspm::main();}

Codeforces1303F Number of Components的更多相关文章

  1. 【CF1151E】Number of Components

    [CF1151E]Number of Components 题面 CF 题解 联通块个数=点数-边数. 然后把边全部挂在较小的权值上. 考虑从小往大枚举左端点,等价于每次删掉一个元素,那么删去点数,加 ...

  2. CodeForces 1151E Number of Components

    题目链接:http://codeforces.com/problemset/problem/1151/E 题目大意: n个人排成一个序列,标号为 1~n,第 i 个人的学习成绩为 ai,现在要选出学习 ...

  3. Codefores 1151E Number of Components

    大意:给定n元素序列$a$, $1\le a_i \le n$, 定义函数$f(l,r)$表示范围在$[l,r]$以内的数构成的连通块个数, 求$\sum\limits_{i=1}^{n}\sum\l ...

  4. cf1151e number of components

    很常见的思想:将整体求改为统计每个部分的贡献 本题中统计[l, r]时, 每个连通块有一个重要特征, 最右端的数在[l,r]中而下一个数不在(好像是句废话 那么我们分别考虑每个点对连通块的贡献, 即它 ...

  5. [CF1303F] Number of Components - 并查集,时间倒流

    有一个 \(n \times m\) 矩阵,初态下全是 \(0\). 如果两个相邻元素(四连通)相等,我们就说它们是连通的,且这种关系可以传递. 有 \(q\) 次操作,每次指定一个位置 \((x_i ...

  6. Codeforces 1270H - Number of Components(线段树)

    Codeforces 题目传送门 & 洛谷题目传送门 首先需发现一个性质,那就是每一个连通块所对应的是一个区间.换句话说 \(\forall l<r\),若 \(l,r\) 在同一连通块 ...

  7. [翻译]Writing Custom Report Components 编写自定义报表组件

    摘要:简单介绍了如何编写一个FastReport的组件,并且注册到FastReport中使用.   Writing Custom Report Components 编写自定义报表组件 FastRep ...

  8. OpenCV人脸识别Eigen算法源码分析

    1 理论基础 学习Eigen人脸识别算法需要了解一下它用到的几个理论基础,现总结如下: 1.1 协方差矩阵 首先需要了解一下公式: 共公式可以看出:均值描述的是样本集合的平均值,而标准差描述的则是样本 ...

  9. SDWebImage源码解读_之SDWebImageDecoder

    第四篇 前言 首先,我们要弄明白一个问题? 为什么要对UIImage进行解码呢?难道不能直接使用吗? 其实不解码也是可以使用的,假如说我们通过imageNamed:来加载image,系统默认会在主线程 ...

随机推荐

  1. DFS+BFS(广度优先搜索弥补深度优先搜索遍历漏洞求合格条件总数)--09--DFS+BFS--蓝桥杯剪邮票

    题目描述 如下图, 有12张连在一起的12生肖的邮票.现在你要从中剪下5张来,要求必须是连着的.(仅仅连接一个角不算相连)  比如,下面两张图中,粉红色所示部分就是合格的剪取.  请你计算,一共有多少 ...

  2. Python中的常用内置对象之range对象

    range(start, stop[, step])  可生成满足条件的数.具体来说是返回一个从start开始到小于stop的相邻数的差step的等差数列列表.结果中包含start一直到小于stop的 ...

  3. struts2模型驱动传值问题

    控制台错误提示: 2020-01-08 18:34:40,292 [http-nio-8080-exec-3] [org.apache.struts2.dispatcher.Dispatcher]-[ ...

  4. 2020/2/2 PHP代码审计之反序列化

    0x00 序列化与反序列化 序列化: serialize()把对象转换为字节序列的过程称为对象的序列化 反序列化: unserialize()把字节序列恢复为对象的过程称为对象的反序列化 0x01 序 ...

  5. react 16 性能提升 总结

    1. 减少子组件渲染 当 父组件 state 内的某个值(eg:value) 不变时 子组件菜 render shouldComponentUpdate(nextProps, nextState){ ...

  6. Q4:Median of Two Sorted Arrays

    4. Median of Two Sorted Arrays 官方的链接:4. Median of Two Sorted Arrays Description : There are two sort ...

  7. python基础1--基本数据类型+流程控制

      一.基本数据类型 1.整型 int 就是整数   2.浮点型 float 就是小数     3.字符串 3.1.加了单引号.双引号.多引号的字符就认为是字符串 单引号和双引号没有什么区别,多引号用 ...

  8. php curl模拟post请求提交数据例子总结

    php curl模拟post请求提交数据例子总结 [导读] 在php中要模拟post请求数据提交我们会使用到curl函数,下面我来给大家举几个curl模拟post请求提交数据例子有需要的朋友可参考参考 ...

  9. UVALive 3983 捡垃圾的机器人 DP

    这个题目我最初的做法沿用树形DP的做法,设置一个 dp[i][0]表示机器人在i点不回去的最短路径,dp[i][1]表示机器人在i点回去的最短路径,规划方向为i-1向i转移,结果发现这个不能用树形的结 ...

  10. 对PHP-GC(垃圾回收)的一点理解

    一直对php的垃圾回收机制不明不白故自己开贴记录下. 首先,说到垃圾回收,得先知道什么情况下才能产生垃圾. 如果一个变量refcount在增加,说明在被使用,不是垃圾. 如果一个变量的refcount ...