很典型的按照边考虑贡献的题。

题目描述

小A居住的城市可以认为由n个街区组成。街区从1到n依次标号街区与街区之间由街道相连,每个街区都可以通过若干条街道到达任意一个街区,共有n-1条街道。其中标号为i的街区居住了i名居民。居民会去拜访别人,但是要花费dis(u,v)的过路费,u是他所在的城市,v是他拜访的人所在的城市。你需要求出,所有人都拜访其他人一次花费的过路费之和。

输入格式

第一行一个整数nn接下来n-1行,每行2个整数n−1n−1个整数描述n-1条街道

输出格式

一个整数,表示总花费之和

样例输入

5

1 2

2 3

2 4

1 5

样例输出

184

数据规模与约定

对于30%的数据,满足n≤200n≤200

对于60%的数据,满足n≤3000n≤3000

对于100%的数据,满足n≤1000000n≤1000000


题目分析

是一道典型的按边统计答案的题。但为什么我又没想出来啊。

题目求的是∑u∑v∗dis(u,v),那么来考虑一下问题的瓶颈在哪里。

按照定义直接做

首先是按照定义直接做的想法。

那么统计枚举所有点对,是 O(n^2) 的,预处理 dis(u,v) 有 O(n^3) 的floyd;还有 O(n^2) 的做 n 次dfs。

然而这个方向的做法空间复杂度是肯定要 O(n^2) 的,而且统计枚举的复杂度也难以改进。

考试时候就是吊死在这颗树上没出来了……

分边考虑贡献

统计时候不要那么“直接”,而是把整个答案分部分来考虑。

对于每一个点,与之相关的答案是 i*∑dis[i] 。于是我们发现最后的答案是只与 sum_{dis_i} 有关的。也就是说,对于点 x ,如果预处理了以它为根的 dis[] ,那么其贡献就是可以 O(1) 求出的。因此,解题瓶颈从处理点对的 dis[u][v] 变为了转移 dis[i] 。

对此,Cptraser表示有一种神奇的“平衡移动”方法。

这里(1,2)这条边是正在枚举的边。我们现在要做的是快速将 ∑dis[](以1为根) 转为 ∑dis[](以2为根) 。

图画出来后就很显然了。有$\sum_{newDis[]} \qquad \quad =\sum_{dis[]} \quad +tot_v-2*size[v]$ 。其中 size[v] 表示以 v 为根子树大小。当然这里所谓的子树大小是要提前人为确定一个根节点的。

这般把答案分部分之后,我们就会惊喜地发现复杂度降为$O(n+m)$了。

 #include<bits/stdc++.h>
typedef long long ll;
const ll MO = 1e9+;
const int maxn = ;
const int maxm = ; int n;
int edges[maxm],nxt[maxm],head[maxn],edgeTot;
ll tmp,ans,sum,dis[maxn],size[maxn]; int read()
{
char ch = getchar();
int num = ;
bool fl = ;
for (; !isdigit(ch); ch = getchar())
if (ch=='-') fl = ;
for (; isdigit(ch); ch = getchar())
num = (num<<)+(num<<)+ch-;
if (fl) num = -num;
return num;
}
void addedge(int u, int v)
{
edges[++edgeTot] = v, nxt[edgeTot] = head[u], head[u] = edgeTot;
edges[++edgeTot] = u, nxt[edgeTot] = head[v], head[v] = edgeTot;
}
void dfs1(int x, int fa)
{
size[x] = x;
for (int i=head[x]; i!=-; i=nxt[i])
if (edges[i]!=fa)
dis[edges[i]] = dis[x]+, dfs1(edges[i], x), size[x] += size[edges[i]];
}
void dfs2(int x, int fa)
{
ll cnt = ;
for (int i=head[x]; i!=-; i=nxt[i])
if (edges[i]!=fa){
int v = edges[i];
cnt = (sum-2ll*size[v])%MO;
tmp = (tmp+cnt)%MO;
ans = (ans+tmp*v%MO)%MO;
dfs2(v, x);
tmp = (tmp-cnt+MO)%MO;
}
}
ll qmi(ll a, ll b)
{
ll ret = ;
while (b)
{
if (b&) ret = ret*a%MO;
a = a*a%MO, b >>= ;
}
return ret;
}
int main()
{
memset(head, -, sizeof head);
// freopen("city.in","r",stdin);
// freopen("city.out","w",stdout);
n = read();
for (int i=; i<n; i++) addedge(read(), read());
dfs1(, );
for (int i=; i<=n; i++) tmp += 1ll*i*dis[i];
ans = tmp, sum = 1ll*(n+)*n/;
dfs2(, );
printf("%lld\n",ans*qmi(, MO-)%MO);
return ;
}

END

【树形dp】7.14城市的更多相关文章

  1. CF 219D Choosing Capital for Treeland 树形DP 好题

    一个国家,有n座城市,编号为1~n,有n-1条有向边 如果不考虑边的有向性,这n个城市刚好构成一棵树 现在国王要在这n个城市中选择一个作为首都 要求:从首都可以到达这个国家的任何一个城市(边是有向的) ...

  2. Codeforces Beta Round #14 (Div. 2) D. Two Paths 树形dp

    D. Two Paths 题目连接: http://codeforces.com/contest/14/problem/D Description As you know, Bob's brother ...

  3. 洛谷 P1453 城市环路 ( 基环树树形dp )

    题目链接 题目背景 一座城市,往往会被人们划分为几个区域,例如住宅区.商业区.工业区等等.B市就被分为了以下的两个区域--城市中心和城市郊区.在着这两个区域的中间是一条围绕B市的环路,环路之内便是B市 ...

  4. 【牛客】乃爱与城市拥挤程度 — 树形dp,up and down

    我太难了 这题做得我要死了,来来回回写了大概八九个小时 错误的原因要么是快速幂写错(一生之敌,要么是忘取模爆\(longlong\)变负数\(QAQ\) \(update\) \(2019.11.13 ...

  5. 2018.09.14 洛谷P3931 SAC E#1 - 一道难题 Tree(树形dp)

    传送门 简单dp题. f[i]表示以i为根的子树被割掉的最小值. 那么有: f[i]=min(∑vf[v],dist(i,fa))" role="presentation" ...

  6. Codeforces Beta Round #14 (Div. 2) Two Paths (树形DP)

    Two Paths time limit per test 2 seconds memory limit per test 64 megabytes input standard input outp ...

  7. BZOJ4890 [Tjoi2017]城市 【树形dp】

    题目链接 BZOJ4890 题解 枚举断开哪一条边,然后对剩余的两棵树分别做一遍换根法树形dp 需要求出每个点到树中其它点距离的最大值\(f[i]\)和次大值\(g[i]\)[用以辅助换根计算最大值] ...

  8. BZOJ 4890: [Tjoi2017]城市 树形dp

    标签:树形dp,枚举,树的直径 一上来看到这个题就慌了,只想到了 $O(n^3)$ 的做法. 碰到这种题时要一步一步冷静地去分析,观察数据范围. 首先,$n\leqslant 5000$,所以可以先 ...

  9. 【NOIP2016提高A组集训第14场11.12】随机游走——期望+树形DP

    好久没有写过题解了--现在感觉以前的题解弱爆了,还有这么多访问量-- 没有考虑别人的感受,没有放描述.代码,题解也写得歪歪扭扭. 并且我要强烈谴责某些写题解的代码不打注释的人,像天书那样,不是写给普通 ...

  10. 树形DP小结

    树形DP1.简介:树是一种数据结构,因为树具有良好的子结构,而恰好DP是从最优子问题更新而来,那么在树上做DP操作就是从树的根节点开始深搜(也就是记忆化搜索),保存每一步的最优结果.tips:树的遍历 ...

随机推荐

  1. IT兄弟连 JavaWeb教程 Servlet会话跟踪 Cookie的优缺点

    Cookie技术存储的数据类型只能是字符串,且不支持中文 ●  保存的数据大小有限,4kb ●  太依赖用户浏览器的设置,用户可以禁用Cookie! ●  数据存储在客户端的文本文件中,不安全,不建议 ...

  2. 渗透测试之无文件渗透简单使用-windows

    无文件渗透测试工作原理:无文件恶意程序最初是由卡巴斯基在2014年发现的,一直不算是什么主流的攻击方式,直到此次事件的发生.说起来无文件恶意程序并不会为了执行而将文件或文件夹复制到硬盘上,反而是将pa ...

  3. 花花的森林(倍增,LCA

    花花的森林,嗯,这是一篇正经的题解. 模拟考的时候没有看出来要怎么求啊,暴力地树形DP.换根.合并.求直径.居然也险险地拿到了80分,不过我们要正经地想正解. 容易想到我们可以让时光倒流,让空间扭转, ...

  4. 执行gulp build报错

    问题与分析 在执行gulp build报错如下: D:\coding\Resume\Resumes>gulp build gulp build[5628]: src\node_contextif ...

  5. 测试 | 代码覆盖测试工具 | Eclemma

    安装: 打开eclipse,点击Help菜单下的Install New Software 在弹出的对话框中,点击Add 输入Name,如EclEmma 输入Location: http://updat ...

  6. django-返回客户端外网ip服务

    在服务器应用初始化的时候,比如salt-minion,需要在配置文件里说明自己的id,一般用ip. 如果都在一个内网里,从ip命令里获取就行了. 但现在的企业后台环境更加复杂,很多都是跨机房.有物理机 ...

  7. Linux (二)

    PS :显示系统进程 -a :显示所有进程(包括其他用户的进程) -u :用户以及其他详细信息 -x :显示没有控制终端的进程 -ef :显示所有 top :用于动态地监视进程活动与系统负载的信息 p ...

  8. __getitem__,__setitem__,__delitem__

    __getitem__.__setitem__.__delitem__ 总结: __getitem__,__setitem_,__delitem__ : obj[‘属性’]的方式去操作属性时触发的方法 ...

  9. CATIA 各个版本代号详解

    一. 第几代(V-"version")简介 1982—1988年,catia相继发布了第一代—V1版本.第二代—V2版本.第三代—V3版本,并于1993年发布了功能强大的第四代—V ...

  10. Spring的ioc(DI)复习概念和原理简介

    IOC的好处 ioc或者说di的概念很显然了,反转控制和依赖注入,那本来直接new就行的东西,为什么要搞这么复杂呢?? 开发维护方便,高层设计不用依赖底层的,不然底层一个类改下构造器,高层就全要改,因 ...