Codeforces1303F Number of Components
Description
题意:给一个全\(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的更多相关文章
- 【CF1151E】Number of Components
[CF1151E]Number of Components 题面 CF 题解 联通块个数=点数-边数. 然后把边全部挂在较小的权值上. 考虑从小往大枚举左端点,等价于每次删掉一个元素,那么删去点数,加 ...
- CodeForces 1151E Number of Components
题目链接:http://codeforces.com/problemset/problem/1151/E 题目大意: n个人排成一个序列,标号为 1~n,第 i 个人的学习成绩为 ai,现在要选出学习 ...
- 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 ...
- cf1151e number of components
很常见的思想:将整体求改为统计每个部分的贡献 本题中统计[l, r]时, 每个连通块有一个重要特征, 最右端的数在[l,r]中而下一个数不在(好像是句废话 那么我们分别考虑每个点对连通块的贡献, 即它 ...
- [CF1303F] Number of Components - 并查集,时间倒流
有一个 \(n \times m\) 矩阵,初态下全是 \(0\). 如果两个相邻元素(四连通)相等,我们就说它们是连通的,且这种关系可以传递. 有 \(q\) 次操作,每次指定一个位置 \((x_i ...
- Codeforces 1270H - Number of Components(线段树)
Codeforces 题目传送门 & 洛谷题目传送门 首先需发现一个性质,那就是每一个连通块所对应的是一个区间.换句话说 \(\forall l<r\),若 \(l,r\) 在同一连通块 ...
- [翻译]Writing Custom Report Components 编写自定义报表组件
摘要:简单介绍了如何编写一个FastReport的组件,并且注册到FastReport中使用. Writing Custom Report Components 编写自定义报表组件 FastRep ...
- OpenCV人脸识别Eigen算法源码分析
1 理论基础 学习Eigen人脸识别算法需要了解一下它用到的几个理论基础,现总结如下: 1.1 协方差矩阵 首先需要了解一下公式: 共公式可以看出:均值描述的是样本集合的平均值,而标准差描述的则是样本 ...
- SDWebImage源码解读_之SDWebImageDecoder
第四篇 前言 首先,我们要弄明白一个问题? 为什么要对UIImage进行解码呢?难道不能直接使用吗? 其实不解码也是可以使用的,假如说我们通过imageNamed:来加载image,系统默认会在主线程 ...
随机推荐
- 量化投资_Multicharts数组操作函数_append()追加函数(自定义)
1. Multicharts中关于数组的操作比较麻烦,而且当中所谓的动态数组的定义并不是像其他语言那种的概念.因此要对数组进行元素“”追加“”的话,需要重新更改数组的索引,然后再最后一个位置添加val ...
- select、poll和epoll机制
一.参考网址 1.select函数及fd_set介绍 2.linux select 函数和 fd_set 用法 2.select.poll和epoll的区别 3.利用select实现IO多路复用TCP ...
- 201809-2 买菜 Java
思路: 顺序读入,例如:小H装车的时间段为[1,3],小W装车的时间段为[2,4],重叠部分为[2,3],记在数组times[2]中.最后输出时判断数组times中值大于1的(其实就是2),即为重叠部 ...
- [Python函数]encode,decode
前言: 我们知道,计算机是以二进制为单位的,也就是说计算机只识别0和1,也就是我们平时在电脑上看到的文字,只有先变成0和1,计算机才会识别它的意思.这种数据和二进制的转换规则就是编码.计算机的发展中, ...
- (递归)P1025 数的划分
题解: #include<iostream>using namespace std;int ret=0,m_n;void p(int n,double k,int j){ if(k==1) ...
- Thread--CountDownLatch & CyclicBarrier
参考:http://www.importnew.com/21889.html CountDownLatch countDown() 方法执行完只是计数器减一, 并不会阻塞当前运行线程的的后续代码执行. ...
- 读书笔记 - javascript 高级程序设计 - 第二章 在Html中使用JavaScript
1 <script>的6个属性 async 立即下载当前script标签的外部脚本 但不能影响别的 charset 没用了 defer 文档显示之后再执行脚本,只对外部脚本有效 lan ...
- Python笔记_第五篇_Python数据分析基础教程_前言
1. 前言: 本部分会讲解在Python环境下进行数值运算.以NumPy为核心,并讲解其他相关库的使用,诸如Matplotlib等绘图工具等. C.C++和Forttran等变成语言各有各的优势,但是 ...
- 动态加载JS文件方法总结
1.JQuery方法 $.getScript("./test.js"); //加载js文件 $.getScript("./test.js",function() ...
- 文献阅读报告 - Pedestrian Trajectory Prediction With Learning-based Approaches A Comparative Study
概述 本文献是一篇文献综述,以自动驾驶载具对外围物体行动轨迹的预测为切入点,介绍了基于运动学(kinematics-based)和基于机器学习(learning-based)的两大类预测方法. 并选择 ...