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. .NET 上传并解析CSV文件存库

    1.前端: 放置浏览按钮 <div class="row inner_table text-center"> <input id="fileId&quo ...

  2. ThreadPool类(线程池)

    原文:ThreadPool类(线程池) CLR线程池并不会在CLR初始化时立即建立线程,而是在应用程序要创建线程来运行任务时,线程池才初始化一个线程.线程池初始化时是没有线程的,线程池里的线程的初始化 ...

  3. WPF特效-实现3D足球效果

    原文:WPF特效-实现3D足球效果 WPF 实现 3D足球效果,效果图如下:  每个面加载不同贴图. <UserControl x:Class="MediaBalll.Model3Ds ...

  4. windows server疑难杂症

    1.某些网址.服务访问失败,可能的原因:增强的安全配置关闭增强的安全配置,并且重启电脑!!!http://jingyan.baidu.com/article/6181c3e076ac0b152ff15 ...

  5. NoSQL Manager for MongoDB 4.6.0.3 带key

    NoSQL Manager for MongoDB 4.6.0.3 是一个Windows平台的MongoDB高级管理工具.请低调使用. 博客园文件一次最大不超过10M. 官方安装包: mongodbm ...

  6. Python中实现switch分支结构

    Python不像C/C++,Java等有switch-case的语法.不过其这个功能,比如用Dictionary以及lambda匿名函数特性来替代实现. 实现方法分为两步: 首先,定义一个字典: 其次 ...

  7. ORACLE RAC+OGG配置

    实验环境主机名   IP地址rac01     192.168.56.10rac02     192.168.56.20rac-scan 192.168.56.30 目标库:ora-ogg  192. ...

  8. spring.net的简单使用(三)创建对象

    这篇主要说对象的创建方式. spring.net提供了三种创建对象的方式,分别是构造器创建,静态工厂创建,实例工厂创建. 多数的情况下,容器会根据对象定义中type属性值去直接调用相应类型的某个构造器 ...

  9. centos 7 安装 git 2.22.0

    1.安装所需软件包 yum install curl-devel expat-devel gettext-devel openssl-devel zlib-devel yum install gcc ...

  10. CPU多核控速

    初学者很多对自己开发的软件使用硬件资源的时候并不注意,造成写出的东西不是很满意. 一般有两种情况: 1.写的都是同步单线程任务,不管你电脑有多少个核都不关我事 我就用你1个核所以不管怎么样都不会把CP ...