[FJOI2020]染色图的联通性问题 题解
FJOI2020 D1T2
题目大意
给出一个由 $n$ 个点 $m$ 条边构成的染色无向图,求删去每一个点及与其相连的边后图中不连通的同色点对数量。$n,m\leq 10^5$。
思路分析
可以想到先统计原图的答案,然后对删去每个点后的多出的答案进行计算,输出时加上即可。
原图的答案很容易统计,遍历一遍同时计算即可。
如何统计删去每个点后多出的答案?模拟过后很容易发现,多出的答案就是删去这个点后断开的连通块之间形成的同色点对,只需要知道断开后每个连通块的各色点的数量即可。暴力统计显然复杂度太高,连最低档的分都拿不到。毒瘤FJOI
可以想到用 tarjan 找删去后的各个连通块,统计用启发式合并或线段树合并。线段树合并实现简单但是空间较大,但是还是有神犇卡过去了。这里用的是线段树合并。
合并的时候怎么计算呢?
设当前已合并的连通块该颜色的点数和为 $x$ ,原连通块该颜色的点数为 $y$ ,那么遍历一个新的连通块时,设该连通块该颜色的点数为 $z$ ,则将该连通块合并后与其它部分断开后的贡献为为 $z*(y-x-z)+x*(y-x-z)=(x+z)(y-x-z)$ 。注意,若该连通块与原连通块之间会被断掉产生多出的答案,则需要加上 $x*z$ 。
注意,上面的原连通块断开后即为当前节点的父节点所在的连通块。
这样这道题就很好解了。对于原图中的每个连通块:
1. 先遍历一遍,计算出连通块中每个颜色的点数
2. 跑一遍 tarjan ,同时合并数据,计算断开连通块中的每个点后多出的答案
3. 再遍历一遍,计算连通块与原图的其它连通块贡献的答案,然后将当前连通块的数据清空
#include<iostream>
#include<cstdio>
#include<cstring>
#define ll long long
using namespace std;
const int N=5e5+100;
struct Seg
{
int lson,rson;
ll val,sumv;
#define lson(i) t[i].lson
#define rson(i) t[i].rson
#define val(i) t[i].val
#define sumv(i) t[i].sumv
}t[N*21];
int n,m,tot,cnt,D;
ll now,sum;
int head[N],ver[2*N],Next[2*N];
int rt[N],c[N],nowc[N],sumc[N],dfn[N],low[N];
ll ans[N];
bool vp[N],vq[N];
void add(int x,int y)
{
ver[++tot]=y,Next[tot]=head[x],head[x]=tot;
ver[++tot]=x,Next[tot]=head[y],head[y]=tot;
}
void change(int &p,int l,int r,int k)
{
if(!p)
p=++cnt;
if(l==r)
{
val(p)++,sumv(p)+=nowc[l]-1;
return ;
}
int mid=l+r>>1;
if(k<=mid)
change(lson(p),l,mid,k);
else
change(rson(p),mid+1,r,k);
sumv(p)=sumv(lson(p))+sumv(rson(p));
}
int merge(int x,int y,int l,int r)
{
if(!x)
return y;
if(!y)
return x;
if(l==r)
{
sumv(x)=(val(x)+val(y))*(nowc[l]-val(x)-val(y));//断开后的答案
now+=val(x)*val(y);//当前合并的两个连通块断开的贡献
val(x)+=val(y);
return x;
}
int mid=l+r>>1;
lson(x)=merge(lson(x),lson(y),l,mid);
rson(x)=merge(rson(x),rson(y),mid+1,r);
sumv(x)=sumv(lson(x))+sumv(rson(x));
return x;
}
void pre(int x)
{
vp[x]=1;nowc[c[x]]++;
for(int i=head[x];i;i=Next[i])
{
int y=ver[i];
if(!vp[y])
pre(y);
}
}//先遍历一遍,计算出连通块中每个颜色的点数
void tarjan(int x)
{
int nowr=0;//临时根
low[x]=dfn[x]=++cnt;
change(rt[x],1,D,c[x]);
for(int i=head[x];i;i=Next[i])
{
int y=ver[i];
if(!dfn[y])
{
tarjan(y);
low[x]=min(low[x],low[y]);
if(low[y]>=dfn[x])//断开会使连通块断开,类似割点
{
now=0;
rt[x]=merge(rt[x],rt[y],1,D);
ans[x]+=now;//多出的答案
}
else
nowr=merge(nowr,rt[y],1,D);//不会断开,合并到临时根上,避免多统计答案
}
else
low[x]=min(low[x],dfn[y]);
}
ans[x]+=sumv(rt[x]);
rt[x]=merge(rt[x],nowr,1,D);//合并临时根
}//跑一遍 tarjan ,同时合并数据,计算断开连通块中的每个点后多出的答案
void query(int x)
{
vq[x]=1;
sum+=nowc[c[x]]*sumc[c[x]];//计算当前连通块与其它连通块的贡献
sumc[c[x]]+=nowc[c[x]],nowc[c[x]]=0;
for(int i=head[x];i;i=Next[i])
{
int y=ver[i];
if(!vq[y])
query(y);
}
}//再遍历一遍,计算当前连通块与原图的其它连通块贡献的答案,然后将当前连通块的数据清空
int main()
{
//freopen("pair.in","r",stdin);
//freopen("pair.out","w",stdout);
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++)
scanf("%d",&c[i]),D=max(D,c[i]);
for(int i=1,x,y;i<=m;i++)
{
scanf("%d%d",&x,&y);
add(x,y);
}
for(int i=1;i<=n;i++)
if(!dfn[i])
pre(i),tarjan(i),query(i);
for(int i=1;i<=n;i++)
printf("%lld\n",sum+ans[i]);
return 0;
}
[FJOI2020]染色图的联通性问题 题解的更多相关文章
- tarjan算法,一个关于 图的联通性的神奇算法
一.算法简介 Tarjan 算法一种由Robert Tarjan提出的求解有向图强连通分量的算法,它能做到线性时间的复杂度. 我们定义: 如果两个顶点可以相互通达,则称两个顶点强连通(strongly ...
- FJOI2020 游记
Day -1 啥都不会,药丸 看了看统考题,好难,爆零的节奏 文化课OI双爆炸 尽力吧 Day 0 花三个多小时才到考场 福州真的好热 签到 在小礼堂待了一会,顺便给手机充了电 四点试机,今年用了新系 ...
- 洛谷P1155 双栈排序题解(图论模型转换+二分图染色+栈)
洛谷P1155 双栈排序题解(图论模型转换+二分图染色+栈) 标签:题解 阅读体验:https://zybuluo.com/Junlier/note/1311990 原题地址:洛谷P1155 双栈排序 ...
- 题解 P1682 【过家家】
P1682 过家家 题目描述 有2n个小学生来玩过家家游戏,其中有n个男生,编号为1到n,另外n个女生,编号也是1到n.每一个女生可以先选择一个和她不吵嘴的男生来玩,除此之外,如果编号为X的女生的朋友 ...
- PAT甲题题解-1126. Eulerian Path (25)-欧拉回路+并查集判断图的连通性
题目已经告诉如何判断欧拉回路了,剩下的有一点要注意,可能图本身并不连通. 所以这里用并查集来判断图的联通性. #include <iostream> #include <cstdio ...
- bzoj2958: 序列染色(DP)
2958: 序列染色 题目:传送门 题解: 大难题啊(还是我太菜了) %一发大佬QTT 代码: #include<cstdio> #include<cstring> #incl ...
- [专题总结]矩阵树定理Matrix_Tree及题目&题解
专题做完了还是要说两句留下什么东西的. 矩阵树定理通俗点讲就是: 建立矩阵A[i][j]=edge(i,j),(i!=j).即矩阵这一项的系数是两点间直接相连的边数. 而A[i][i]=deg(i). ...
- tarjan算法讲解。
tarjan算法讲解. 全网最详细tarjan算法讲解,我不敢说别的.反正其他tarjan算法讲解,我看了半天才看懂.我写的这个,读完一遍,发现原来tarjan这么简单! tarjan算法,一个关 ...
- Tarjan的缩点&&割点概述
What is Tarjan? Tarjan,是一种用来解决图的联通性的一种有效途径,它的一般俗称叫做:缩点.我们首先来设想一下: 如果我们有一个图,其中A,B,C构成一个环,那么我们在某种条件下,如 ...
随机推荐
- PHP gd_info - 取得当前安装的 GD 库的信息
gd_info — 取得当前安装的 GD 库的信息. 语法 array gd_info ( void )高佣联盟 www.cgewang.com 返回一个关联数组描述了安装的 GD 库的版本和性能. ...
- PHP xpath() 函数
定义和用法 xpath() 函数运行对 XML 文档的 XPath 查询.高佣联盟 www.cgewang.com 如果成功,该函数返回 SimpleXMLElements 对象的一个数组.如果失败, ...
- Xposed原理分析
目录 安卓系统启动 什么zygote? 安卓应用运行? Xposed介绍 Xposed构成 Xposed初始化大体工作流程 源码分析 初始化 app_main#main app_main#initia ...
- 【NOI2017】游戏 题解(2-SAT+缩点)
题目链接 题目大意:有四种场地$a,b,c,x$和三种赛车$A,B,C$,$a$不能跑$A$,$b$不能跑$B$,$c$不能跑$C$,$x$都可以跑.给定$n$个场地和$m$个四元组$(i,h_i,j ...
- 035_go语言中的速率限制
代码演示 package main import "fmt" import "time" func main() { requests := make(chan ...
- 用了Dapper之后通篇还是SqlConnection,真的看不下去了
一:背景 1. 讲故事 前几天看公司一个新项目的底层使用了dapper,大家都知道dapper是一个非常强大的半自动化orm,帮程序员解决了繁琐的mapping问题,用起来非常爽,但我还是遇到了一件非 ...
- 语言模型 Language Model (LM)
定义 什么是语言模型,通俗的讲就是从语法上判断一句话是否通顺.即判断如下的概率成立: \[p(\text{今天是周末})>p(\text{周末是今天}) \] 链式法则(chain rule) ...
- eclipse中启动tomcat出现错误的解决方法
前段时间跟着老师做课设,各方面调试都没有问题.近段时间想起来,看看之前写过的代码,翻着翻着就发现启动tomcat出现了错误 错误如下: 错误原因:tomcat路径配置有问题,之前可能配置好了然后由于种 ...
- 剑指Offer顺时针打印矩阵
题目描述 输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字,例如,如果输入如下4 X 4矩阵: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 则依次打印出数 ...
- Vuex mapState的基本使用
mapState把Store中的state映射到组件中的计算属性 Store文件 import Vue from 'vue' import Vuex from 'vuex' Vue.use(Vuex) ...