http://codeforces.com/contest/348/problem/B

题意:给一棵树,每个叶子结点有w[i]个苹果,每个子树的苹果数量为该子树所有叶子结点苹果数量之和,要使得每个结点的各个子树苹果数量相等,求至少需要拿走的苹果数量。

思路:一开始以为只要使得所有子树之和相同就行了。

 void dfs(int u, int fa) {
int num = , mi = INF;
for(int i = head[u]; ~i; i = edge[i].nxt) {
int v = edge[i].v; if(v == fa) continue;
dfs(v, u);
sz[u] += sz[v]; num++; mi = mi > sz[v] ? sz[v] : mi;
}
//printf("%d : %lld - %d - %d\n", u, sz[u], mi, num);
ans += sz[u] - mi * num;
sz[u] = mi * num + w[u];
}

后来看错误的样例,是没理解好题意。

例如这组样例:

10
0 9 5 0 0 0 0 0 9 7
7 5
8 1
1 5
4 3
2 4
4 7
7 9
10 6
6 8

红色的为正确的,蓝色为之前想错的。

题解:“对于一棵以u为根的子树,如果要减少若干个苹果,那么需要从u的每棵子树中取走等量的苹果,这个过程会递归下去直到叶子,
考虑对每个节点维护两个信息,mx[u]表示u子树中最多的苹果数,cnt[u]表示u子树中苹果数必须是cnt[u]的倍数,
如果u是叶子,那么有mx[u]=a[u],cnt[u]=1,否则有cnt[u]=lcm(cnt[v]),这里v是u的儿子,mx[u]则是不超过min(mx[v])的最大的cnt[u]的倍数,
最后结果就是mx[1],复杂度O(nlogA),这个logA是gcd的复杂度。”

真的好厉害 = =

 #include <bits/stdc++.h>
using namespace std;
typedef long long LL;
#define N 100010
#define INF 0x3f3f3f3f
struct Edge {
int v, nxt;
} edge[N*];
LL w[N], ans, mx[N], cnt[N];
int head[N], tot;
void Add(int u, int v) {
edge[tot] = (Edge) { v, head[u] }; head[u] = tot++;
edge[tot] = (Edge) { u, head[v] }; head[v] = tot++;
} LL gcd(LL a, LL b) { return b ? gcd(b, a % b) : a; } LL lcm(LL a, LL b) { return a / gcd(a, b) * b; } void dfs(int u, int fa) {
int num = ;
for(int i = head[u]; ~i; i = edge[i].nxt) {
int v = edge[i].v; if(v == fa) continue;
dfs(v, u);
if(!num) mx[u] = mx[v], cnt[u] = cnt[v];
else {
if(cnt[u] < 1e14) cnt[u] = lcm(cnt[u], cnt[v]);
mx[u] = min(mx[u], mx[v]) / cnt[u] * cnt[u]; // mx 必须是cnt[u]的倍数
}
num++;
}
if(!num) mx[u] = w[u], cnt[u] = ;
else {
mx[u] *= num;
if(cnt[u] < 1e14) cnt[u] *= num;
}
// printf("%d : %lld - %lld\n", u, mx[u], cnt[u]);
} int main() {
int n; scanf("%d", &n); LL ans = ;
for(int i = ; i <= n; i++) scanf("%lld", &w[i]), ans += w[i];
memset(head, -, sizeof(head)); tot = ;
for(int i = ; i < n; i++) {
int u, v; scanf("%d%d", &u, &v); Add(u, v);
}
dfs(, -);
cout << ans - mx[] << endl;
return ;
}

Codeforces 348B:Apple Tree(DFS+LCM+思维)的更多相关文章

  1. Codeforces 348B - Apple Tree

    348B - Apple Tree 我们设最后答案为 x , 我们我们就能用x表示出所有节点下面的苹果个数, 然后用叶子节点求lcm, 取最大的可行解. #include<bits/stdc++ ...

  2. POJ.3321 Apple Tree ( DFS序 线段树 单点更新 区间求和)

    POJ.3321 Apple Tree ( DFS序 线段树 单点更新 区间求和) 题意分析 卡卡屋前有一株苹果树,每年秋天,树上长了许多苹果.卡卡很喜欢苹果.树上有N个节点,卡卡给他们编号1到N,根 ...

  3. poj 3321 Apple Tree dfs序+线段树

    Apple Tree Time Limit: 2000MS   Memory Limit: 65536K       Description There is an apple tree outsid ...

  4. [poj3321]Apple Tree(dfs序+树状数组)

    Apple Tree Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 26762   Accepted: 7947 Descr ...

  5. POJ3321 - Apple Tree DFS序 + 线段树或树状数组

    Apple Tree:http://poj.org/problem?id=3321 题意: 告诉你一棵树,每棵树开始每个点上都有一个苹果,有两种操作,一种是计算以x为根的树上有几个苹果,一种是转换x这 ...

  6. POJ 3321 Apple Tree dfs+二叉索引树

    题目:http://poj.org/problem?id=3321 动态更新某个元素,并且求和,显然是二叉索引树,但是节点的标号不连续,二叉索引树必须是连续的,所以需要转化成连续的,多叉树的形状已经建 ...

  7. POJ 3321 Apple Tree (DFS + 树状数组)

    题意: 一棵苹果树有N个分叉,编号1---N(根的编号为1),每个分叉只能有一颗苹果或者没有苹果. 现在有两种操作: 1.某个分叉上的苹果从有变无或者从无边有. 2.需要统计以某个分叉为根节点时,它的 ...

  8. POJ 3321 Apple Tree DFS序+fenwick

    题目大意:有一颗长满苹果的苹果树,有两个操作. 1.询问以一个点为根的子树中有多少个苹果. 2.看看一个点有没有苹果,假设没有苹果.那么那里就立即长出一个苹果(= =!):否则就把那个苹果摘下来. 思 ...

  9. POJ 3321 Apple Tree DFS序 + 树状数组

    多次修改一棵树节点的值,或者询问当前这个节点的子树所有节点权值总和. 首先预处理出DFS序L[i]和R[i] 把问题转化为区间查询总和问题.单点修改,区间查询,树状数组即可. 注意修改的时候也要按照d ...

随机推荐

  1. [AC自己主动机+可能性dp] hdu 3689 Infinite monkey theorem

    意甲冠军: 给n快报,和m频率. 然后进入n字母出现的概率 然后给目标字符串str 然后问m概率倍的目标字符串是敲数量. 思维: AC自己主动机+可能性dp简单的问题. 首先建立trie图,然后就是状 ...

  2. windows常用cmd指令

    打开命令行 1.在菜单栏中搜索命令行 2.在文件管理器的Path栏输入cmd,则在当前目录打开命令行 3.Windows+R,输入cmd,回车 ping(网络诊断工具) ping是Windows.Un ...

  3. js的一些写法问题

    尽量不要拼接字符,用自定义标签来完成     用winform的形式更佳  

  4. sql Left right join 多表 注意表的连接顺序

    多表左/右连接,表的连接顺序也可以影响查询速度 左连接时,应该把小表放在前面连接例子:A.B.C三表左连接情况1:A先和B连接,得到100条记录100条记录再和C左连接情况2:A先和C连接,得到50条 ...

  5. PySide——Python图形化界面入门教程(一)

    PySide——Python图形化界面入门教程(一) ——基本部件和HelloWorld 翻译自:http://pythoncentral.io/intro-to-pysidepyqt-basic-w ...

  6. qml实现对SSL的支持(使用msys2,同时支持32和64位)超详细 good

    首先准备环境.两种方法,使用mingw64 或者VS 直接放上下载地址https://sourceforge.net/projects/msys2/我下载的是msys2-x86_64-20161025 ...

  7. shell中select、case的使用

    case和select结构在技术上说并不是循环, 因为它们并不对可执行代码块进行迭代. 但是和循环相似的是, 它们也依靠在代码块顶部或底部的条件判断来决定程序的分支. select   select结 ...

  8. Dec Working Note

    01 新的一个月,也是16年最后一个月,意义非凡. 那么第一天就要来点非凡的意义:提出离职. 纠结了好久,最后还是离职了,感觉是好他妈的爽,纠结什么呢. 不过今天状态不好,最近状态一直不好,上火,也没 ...

  9. Android多线程(一)

    在Android应用的开发过程中,我们不可避免的要使用多线程,获取服务器数据.下载网络数据.遍历文件目录查找特定文件等等耗时的工作都离不开线程的知识.Android继承了Java的多线程体系,同时又实 ...

  10. nginx 简单应用

    从源代码编译 Nginx 把源码解压缩之后,在终端里运行如下命令: $ ./configure $ make $ sudo make install 默认情况下,Nginx 会被安装在 /usr/lo ...