Time Limit: 1000 ms   Memory Limit: 256 MB

Description

  给定一张N个点、M条边的无向图 $G$ 。每个点有个权值Wi。

  我们定义 $G_i$ 为图 $G$ 中删除第 $i$ 号顶点后的图。我们想计算 $G_1, G_2, ..., G_n$ 这N张图的权值。

  对于任意一张图 $G$ ,它的权值是这样定义的:

  1. 如果 $G$ 是联通图,那么 $G$ 的权值为 $G$ 中所有顶点权值的乘积。

  2. 如果 $G$ 是非联通图,那么 $G$ 的权值为 $G$ 中所有联通块的权值之和。

  $G$ 中的一个联通块指的是 $G$ 的一个子图,并且这个子图中的点两两相连(包括直接连接或者间接连接),并且不存在子图外的点使得子图内的点能与子图外的点相连。

Input

  第一行包含两个整数 $n$ 和 $m$ $(2 \le n \le 10^5, 1 \le m \le 2 \times 10^5)$ ,分别表示点数和边数。

  第二行包含 $n$ 个整数 $w_1, w_2, ..., w_n$ $(1 \le w_i \le 10^9)$, 表示每个顶点的权值。

  接下来 m 行,每行两个整数 $x_i$ 和 $y_i$ $(1 \le x_i, y_i \le n, x_i \ne y_i)$, 表示一条无向边。

  输出只有一个整数: $S = (\sum\limits_{i=1}^{n}i\cdot z_i) \text{ mod } (10^9 + 7)$, 其中 $z_i$ 是图 $G_i$ 的权值。

Sample Input

Sample Output

10 3
3 3 3
2 3 3
2 3 1
3 1 1
3 1 2
1 3 1
1 1 2
1 2 2
1 3 2
1 2 1
3
1
3
0
1
0
1
0
0
1

Hint

  【数据范围及约定】

  子任务1(5分): $n \leq 10, m \leq 20$

  子任务2(10分): $n \leq 1000, m \leq 2000$

  子任务3(20分): 该图恰为一棵树,$m = n-1$

  子任务4(20分): 该图为一幅联通图

  子任务5(45分): 我们会拿最强的数据来评测你的程序(mmp)

  对于所有数据,$2 \le n \le 10^5, 1 \le m \le 2 \times 10^5$


题解

  没有什么能阻挡我把Tarjan打残。

  题目涉及到删点操作。

  如果删的点$u$是一个非割顶,那么它的消失貌似对这个联通块整体没有太大的影响,要处理的话仅仅是该当前联通块的权值$val$除去$u$的权值$w_u$。

  如果删的点$u$是一个割顶,那么它会将这个联通块分成若干部分,具体就是在Tarjan的缩点树上,把子树全部断开,把父亲也断开。问题来了,割顶这个东西很烦怎么处理?

 

转树

  割顶出现了!它可以同时处于多个点双内,mmp

  对于每个点双,我们暂且新建一个代表点,将点双内的所有点连向这个代表点。这样,一个割顶可以被连接到多个点双的代表点,同时整个图转成了树的形态。

  

  那么断开一个割顶$u$会影响到哪些区块,就一目了然了,即这种树上,$u$的所有子树和父亲那一头的部分。

  发现这其实同化了断开非割顶的操作,非割顶永远处于根节点或叶子节点,其实本质上处理是一样的。

  维护

  $$f_u=\prod\limits_{v\in 以u为根的树}w[v]\\g_u=\sum\limits_{v是u的子树}f[v]$$

  则删去一个点$u$,对所在联通块权值$val$的影响即为:

  $$val=\frac{val}{f_u}+g_u$$

    即父亲那一头的权值+所有子树的权值和

小细节与特判

  1.处理删去割顶的时候(即上面的最后一个公式),$\frac{val}{f_u}$希望得到的是父亲那一头的权值,但如果$u$是树的根,这玩意弄出来却是1,而不是我们希望的0(坑爹),所以记录一下我们要处理的割顶是不是一个树的根,特判一下。

    2.Tarjan深搜的起始点要记为割顶。


 #include <cstdio>
#define min(a,b) (a<b?a:b)
using namespace std;
typedef long long ll;
const ll N=,Mod=1e9+;
int n,m,h1[N],h2[N*],tot;
int col[N],colcnt,st[N],top,bcnt,head[N];
ll info[N],sumup,ans,f[N*],g[N*],w[N*];
int dfn[N],low[N],ins[N],tmcnt;
bool cut[N];
struct Edge{int v,next;}G[N*];
inline void addEdge(int u,int v,int *h){
G[++tot].v=v; G[tot].next=h[u]; h[u]=tot;
}
void tarjan(int u,int fa){
st[++top]=u;
ins[u]=;
dfn[u]=low[u]=++tmcnt;
col[u]=colcnt;
info[col[u]]=(info[col[u]]*w[u])%Mod;
for(int i=h1[u],v,ccnt=;i;i=G[i].next)
if((v=G[i].v)!=fa){
if(!ins[v]){
ccnt++;
tarjan(v,u);
low[u]=min(low[u],low[v]);
if((!fa&&ccnt>)||(fa&&dfn[u]<=low[v]))
cut[u]=;
if(dfn[u]<=low[v]){
w[(++bcnt)+n]=;
do{
addEdge(st[top],bcnt+n,h2);
addEdge(bcnt+n,st[top],h2);
top--;
}while(st[top+]!=v);
addEdge(u,bcnt+n,h2);
addEdge(bcnt+n,u,h2);
}
}
else if(ins[v]==)
low[u]=min(low[u],dfn[v]);
}
ins[u]=;
}
void dfs(int u,int fa){
f[u]=w[u]; g[u]=;
for(int i=h2[u],v;i;i=G[i].next)
if((v=G[i].v)!=fa){
dfs(v,u);
f[u]=(f[u]*f[v])%Mod;
g[u]=(g[u]+f[v])%Mod;
}
}
ll ksm(ll bas,ll tm){
if(tm==) return ;
ll ret=ksm(bas,tm/);
ret=(ret*ret)%Mod;
return ((tm&)?ret*bas:ret)%Mod;
}
ll inv(int x){return ksm(x,Mod-);}
int main(){
scanf("%d%d",&n,&m);
for(int i=;i<=n;i++) scanf("%lld",&w[i]);
for(int i=,u,v;i<=m;i++){
scanf("%d%d",&u,&v);
addEdge(u,v,h1); addEdge(v,u,h1);
}
for(int i=;i<=n;i++)
if(!dfn[i]){
info[++colcnt]=;
tarjan(i,);
cut[i]=;
sumup=(sumup+info[colcnt])%Mod;
head[colcnt]=i;
dfs(i,);
}
for(ll i=,k;i<=n;i++){
int c=col[i];
if(!cut[i])
k=(sumup+Mod*-info[c]+(info[c]*inv(w[i]))%Mod)%Mod;
else{
if(head[c]!=i) k=(sumup+Mod*-info[c]+(info[c]*inv((f[i])%Mod)%Mod)%Mod+g[i])%Mod;
else k=(sumup+Mod*-info[c]+g[i])%Mod;
}
ans=(ans+(i*k)%Mod)%Mod;
}
printf("%lld\n",ans);
return ;
}

奇妙代码

Fantasia (Tarjan+树形DP)的更多相关文章

  1. Codeforces 980F Cactus to Tree 仙人掌 Tarjan 树形dp 单调队列

    原文链接https://www.cnblogs.com/zhouzhendong/p/CF980F.html 题目传送门 - CF980F 题意 给定一个 $n$ 个节点 $m$ 条长为 $1$ 的边 ...

  2. Tarjan+树形DP【洛谷P2515】[HAOI2010]软件安装

    [洛谷P2515][HAOI2010]软件安装 题目描述 现在我们的手头有N个软件,对于一个软件i,它要占用Wi的磁盘空间,它的价值为Vi.我们希望从中选择一些软件安装到一台磁盘容量为M计算机上,使得 ...

  3. bzoj 4784: [Zjoi2017]仙人掌【tarjan+树形dp】

    其实挺简单的但是没想出来---- 首先判断无解情况,即,一开始的图就不是仙人掌,使用tarjan判断如果一个点dfs下去有超过一个点比他早,则说明存在非简单环. 然后考虑dp,显然原图中已经属于某个简 ...

  4. [BZOJ2427][HAOI2010]软件安装(tarjan+树形DP)

    如果依赖关系出现环,那么对于一个环里的点,要么都选要么都不选, 所以每个环可以当成一个点,也就是强连通分量 然后就可以构造出一颗树,然后树形背包瞎搞一下就行了 注意要搞一个虚拟节点当根节点 Code ...

  5. bzoj 2427: [HAOI2010]软件安装【tarjan+树形dp】

    一眼最大权闭合子图,然后开始构图,画了画之后发现我其实是个智障网络流满足不了m,于是发现正确的打开方式应该是一眼树上dp 然后仔细看了看性质,发现把依赖关系建成图之后是个奇环森林,这个显然不能直接dp ...

  6. 2018.10.04 NOIP模拟 航班(tarjan+树形dp)

    传送门 考场上自己yy了一个双连通只有40分. 然后换根dp求最长路就行了. 代码

  7. HDU4612(Warm up)2013多校2-图的边双连通问题(Tarjan算法+树形DP)

    /** 题目大意: 给你一个无向连通图,问加上一条边后得到的图的最少的割边数; 算法思想: 图的边双连通Tarjan算法+树形DP; 即通过Tarjan算法对边双连通缩图,构成一棵树,然后用树形DP求 ...

  8. hdu2242(树形dp+tarjan+缩点)

    hdu2242 http://acm.hdu.edu.cn/showproblem.php?pid=2242 给定n,m表示n个点,m条边 每个点有个权值 问我们删除两某条边(割边)后将图分为两个部分 ...

  9. 【bzoj2427】[HAOI2010]软件安装 Tarjan+树形背包dp

    题目描述 现在我们的手头有N个软件,对于一个软件i,它要占用Wi的磁盘空间,它的价值为Vi.我们希望从中选择一些软件安装到一台磁盘容量为M计算机上,使得这些软件的价值尽可能大(即Vi的和最大).但是现 ...

随机推荐

  1. 基于Swt、ffmpeg、jacob、vlc、SApi、h2技术编写简单的旁白生成器

    一.简介: 前一段时间尝试录制了几集3D编程方面的视频教程,我发现录制时最大的障碍是让脑中的思考.手上的操作和嘴里的解说保持同步,一旦三个"线程"中有一个出错,就必须停下来重新录制 ...

  2. 理解maven的核心概念

    原文出处:http://www.cnblogs.com/holbrook/archive/2012/12/24/2830519.html 好久没进行java方面的开发了,最近又完成了一个java相关的 ...

  3. IO (二)

    1 字符流的缓冲区 缓冲区的出现提高了对数据的读写效率. 对应的类: BufferedWriter BufferedReader 缓冲区要结合流才能使用. 在流的基础上对流的功能进行了增强. 2 Bu ...

  4. [DeeplearningAI笔记]神经网络与深度学习4.深度神经网络

    觉得有用的话,欢迎一起讨论相互学习~Follow Me 4.2 深层神经网络中的前向传播 4.3 核对矩阵的维数 经验方法论 对于神经网络想增加得到没有bug的程序的概率的方法:需要仔细的思考矩阵的维 ...

  5. 基于Controller接口的控制器及简单应用

    DispatcherServlet在Spring当中充当一个前端控制器的角色,它的核心功能是分发请求.请求会被分发给对应处理的Java类,Spring MVC中称为Handle.在Spring 2.5 ...

  6. 是否编码输出html字符

    template.config(name, value)方法用于更改引擎的默认配置. 其中字段escape,类型为boolean,默认为true. 首先,我们不修改配置信息输出一段带有html标签的字 ...

  7. 关于document.body.scrollTop与documentElement.scrollTop

    遇到document.body.scrollTop值为0的问题 今天在写一个小demo的时候,使用滚动条,我用document.body.scrollTop获取滚动条的位置,但是很奇怪的发现在谷歌上获 ...

  8. xBIM IFC 输出 Excel 报表

    目录 xBIM 应用与学习 (一) xBIM 应用与学习 (二) xBIM 基本的模型操作 xBIM 日志操作 XBIM 3D 墙壁案例 xBIM 格式之间转换 xBIM 使用Linq 来优化查询 x ...

  9. MySQL死锁[转]

    案例描述       在定时脚本运行过程中,发现当备份表格的sql语句与删除该表部分数据的sql语句同时运行时,mysql会检测出死锁,并打印出日志. 两个sql语句如下:       (1)inse ...

  10. BZOJ 2406: 矩阵 [上下界网络流 二分答案]

    2406: 矩阵 题意:自己去看吧,最小化每行每列所有元素与给定矩阵差的和的绝对值中的最大值 又带绝对值又带max不方便直接求 显然可以二分这个最大值 然后判定问题,给定矩阵每行每列的范围和每个元素的 ...