题目链接

Bzoj崩了之后在洛谷偶然找到的点分好题!


在暴力的角度来说,如果我们$O(n)$枚举根节点,有没有办法在$O(n)$的时间内找到答案呢?

此时如果用树形$dp$的想法,发现是可做的,因为可以推得以下的结论:

设$x$为根节点,$d[i]$为$i$节点到$x$的距离(即深度),$g[i]$为$i$节点到最近的出入口(即叶子节点)的距离,$ans_{x}$为以$x$为根节点时的答案。

如果$d[i] \geq g[i]$,则我们可以确定,以$i$为子树,对于$x$为根时的答案贡献为$1$。

如下图:

在对于以$i$为根的子树,会对$ans_{x}$产生$1$的贡献,可以理解为一个人从$i$为根的子树的任意叶子节点出发,可以比贝茜更先到达$i$

而这种解法只需要先$dfs$两次得到$g[i]$和$d[i]$,然后一次$dfs$得到答案,复杂度为$O(n^{2})$。

但是这种做法不够理想,我们还想更快的实现。

如果我们对树的性质较为熟悉,我们知道:

$1$.对于树的某棵子树,子树有m个节点,有:$\sum du[i]=2*m-1$

$2$.对于某棵树,树有n个节点,有:$\sum du[i]=2*n-2$

$PS$:$du[i]$为$i$节点的度。

将性质$1$变形为:$1=\sum (2-du[i])$

在本题中,贡献为1的子树有一个性质,即:$d[i] \geq g[i]\& \&d[fa[i]]<g[fa[i]]$。可以理解为他的父亲贡献为子节点个数,即上图中的$i$的父亲。

所以$ans_{x}$=贡献为1的子树数量之和。这不是废话吗......

所以根据性质$1$,有:$ans_{x}=\sum_{i=1}^{n}[d[i] \geq g[i]](2-du[i])$,稍微解释一下式子的来由:

因为子树的$\sum (2-du[i])=1$,而$1$刚好是一颗子树的贡献,所以满足$d[i] \geq g[i]$的点集,可以组成$ans_{x}$那么多棵贡献为1的子树。如下图:

所以满足$g[i] \geq d[i]$的点集为上图圈出来的点,而答案为贡献为1的子树数量:$3$。


此时我们可以用点分治的想法,将:

$ans_{x}=\sum_{i=1}^{n}[d[i] \geq g[i]](2-du[i])$

求解问题变化成求解点对问题:

$ans_{x}=\sum_{i=1}^{n}[dis(x,i) \geq g[i]](2-du[i])$,$dis(x,i)$为$x$到$i$的距离。

所以设$w$为当前子树的重心,$p[i]$为$i$到重心的距离。

则$dis(x,i) \geq g[i]\rightarrow p[x]+p[i] \geq g[i]\rightarrow p[x] \geq g[i]-p[i]$

而在每次求出$p[i]$后,可以使用树状数组维护$g[i]-p[i]$,不过注意$g[i]-p[i]$会小于0,所以维护时向右移$n$的数量。

细节问题可以看代码,其余的问题欢迎提问。

 #include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn = 1e5 + 5e4;
const int inf = 2e9 + ;
struct node {
int s, e, w, next;
}edge[maxn];
int head[maxn], len;
void init() {
memset(head, -, sizeof(head));
len = ;
}
void add(int s, int e) {
edge[len].s = s;
edge[len].e = e;
edge[len].next = head[s];
head[s] = len++;
}
int root, lens, sum;
int d[maxn], du[maxn], o[maxn], vis[maxn], g[maxn], son[maxn], siz[maxn], ans[maxn];
int rt[maxn], n;
int lowbit(int x) {
return x & -x;
}
void Add(int x, int val) {
for (int i = x; i <= * n; i += lowbit(i))
rt[i] += val;
}
int query(int x) {
int ans = ;
for (int i = x; i > ; i -= lowbit(i))
ans += rt[i];
return ans;
}
void getroot(int x, int fa) {
siz[x] = , son[x] = ;
for (int i = head[x]; i != -; i = edge[i].next) {
int y = edge[i].e;
if (y == fa || vis[y])continue;
getroot(y, x);
siz[x] += siz[y];
son[x] = max(son[x], siz[y]);
}
son[x] = max(son[x], sum - siz[x]);
if (son[x] < son[root])root = x;
}
void getd(int x, int fa) {
o[++lens] = x;
for (int i = head[x]; i != -; i = edge[i].next) {
int y = edge[i].e;
if (y == fa || vis[y])continue;
d[y] = d[x] + ;
getd(y, x);
}
}
void cal(int x, int val, int add) {
lens = , d[x] = val;
getd(x, );
for (int i = ; i <= lens; i++)
Add(g[o[i]] - d[o[i]] + n, - du[o[i]]);
for (int i = ; i <= lens; i++)
ans[o[i]] += add * query(d[o[i]] + n);
for (int i = ; i <= lens; i++)
Add(g[o[i]] - d[o[i]] + n, du[o[i]] - );
}
void solve(int x) {
cal(x, , );
vis[x] = ;
for (int i = head[x]; i != -; i = edge[i].next) {
int y = edge[i].e;
if (vis[y])continue;
cal(y, , -);
sum = siz[y];
root = ;
getroot(y, );
solve(root);
}
}
void dfs1(int x, int fa, int dep) { g[x] = inf;
if (du[x] == )g[x] = ;
for (int i = head[x]; i != -; i = edge[i].next) {
int y = edge[i].e;
if (y == fa)continue;
dfs1(y, x, dep + );
g[x] = min(g[x], g[y] + );
}
}
void dfs2(int x, int fa) {
for (int i = head[x]; i != -; i = edge[i].next) {
int y = edge[i].e;
if (y == fa)continue;
g[y] = min(g[y], g[x] + );
dfs2(y, x);
}
}
int main() {
scanf("%d", &n);
init();
for (int i = , x, y; i < n; i++) {
scanf("%d%d", &x, &y);
add(x, y);
add(y, x);
du[x]++, du[y]++;
}
dfs1(, , );
dfs2(, );
son[] = n, root = , sum = n, getroot(, );
solve(root);
for (int i = ; i <= n; i++) {
if (du[i] == )printf("1\n");
else printf("%d\n", ans[i]);
}
}

[洛谷P4183][USACO18JAN]Cow at Large P的更多相关文章

  1. 洛谷 P4183 - [USACO18JAN]Cow at Large P(点分治)

    洛谷题面传送门 点分治 hot tea. 首先考虑什么样的点能够对以 \(u\) 为根的答案产生 \(1\) 的贡献.我们考虑以 \(u\) 为根对整棵树进行一遍 DFS.那么对于一个点 \(v\), ...

  2. luogu P4183 [USACO18JAN]Cow at Large P

    传送门 首先考虑N^2做法,每次从一个点出发,如果到达一个点,然后到达这个点的时间\(\le\)离这个点最近的叶子距离\(di_x\),那么答案+1,否则继续找点 这个暴力很不好优化.可以这样认为,如 ...

  3. [USACO18JAN]Cow at Large G(树形DP)

    P4186 [USACO18JAN]Cow at Large G(树形DP) Luogu4186 设dp[i]表示i点需要放多少个农民.则有 \(if(near[i]-dep[i]<=dep[i ...

  4. 洛谷 P4181 [USACO18JAN]Rental Service

    P4181 [USACO18JAN]Rental Service 题意翻译 farmer john有N(1≤N≤100,000)头牛,他想赚跟多的钱,所以他准备买牛奶和出租牛.有M(1≤M≤100,0 ...

  5. 洛谷P3611 [USACO17JAN]Cow Dance Show奶牛舞蹈

    题目描述 After several months of rehearsal, the cows are just about ready to put on their annual dance p ...

  6. 洛谷P3120 [USACO15FEB]Cow Hopscotch

    题目描述 Just like humans enjoy playing the game of Hopscotch, Farmer John&apos;s cows have invented ...

  7. 【洛谷P3014】Cow Line

    题目大意:康托展开和逆康托展开模板题. 题解: 注:20!约为 2e18. 代码如下 #include <bits/stdc++.h> using namespace std; const ...

  8. 【洛谷P2966】Cow Toll Paths

    题目大意:给定 N 个节点,M 条边的无向图,边有边权,点有点权,现给出 Q 个询问,每个询问查询两个节点之间的最短路径,这里最短路径的定义是两个节点之间的最短路径与这条路径中经过的节点点权的最大值之 ...

  9. [USACO18JAN]Cow at Large P

    Description: 贝茜被农民们逼进了一个偏僻的农场.农场可视为一棵有 \(N\) 个结点的树,结点分别编号为 \(1,2,\ldots, N\) .每个叶子结点都是出入口.开始时,每个出入口都 ...

随机推荐

  1. python解析字体反爬

    爬取一些网站的信息时,偶尔会碰到这样一种情况:网页浏览显示是正常的,用python爬取下来是乱码,F12用开发者模式查看网页源代码也是乱码.这种一般是网站设置了字体反爬 一.58同城 用谷歌浏览器打开 ...

  2. BZOJ3875--骑士游戏(SPFA处理带后效性的动态规划)

    3875: [Ahoi2014]骑士游戏 Time Limit: 30 Sec  Memory Limit: 256 MBSubmit: 181  Solved: 91[Submit][Status] ...

  3. Unity3D_(游戏)甜品消消乐03_游戏UI设计

    甜品消消乐01_游戏基础界面 传送门 甜品消消乐02_游戏核心算法 传送门 甜品消消乐03_游戏UI设计    传送门 (源码在文章最下面~) 实现过程 游戏界面UI 分数与时间的UI显示 有关游戏U ...

  4. vue2.0中watch总结:普通监听和深度监听

    watch:{} 是一个对象,一定要当成对象来用,可监听数据,是vue中数据发生变化进行处理的函数, 它有三个选项 第一个handler:其值是一个回调函数.即监听到变化时应该执行的函数.第二个是de ...

  5. 前端开发——让算法"动"起来

    正文 当然在我们不清楚具体操作细节前我们可以先假设一下,我们能够用什么来实现.按照以前看过的排序动画我将其分为 1.Js操作Dom,再搭配简单的css 2.Canvas动画 之后在查资料的时候发现还有 ...

  6. winscp连接后目录名称乱码

    1.点击[编辑](点后变为保存按钮)按钮,高级按钮变为可用,点击[高级] 2.文件名utf-8编码,默认为“自动”,勾选为“开启”

  7. 多个swiper使用样式出了问题

    observer:true,//修改swiper自己或子元素时,自动初始化swiper observeParents:true,//修改swiper的父元素时,自动初始化swiper 不行直接设  w ...

  8. SparseArray II

    SparseArray: SparseArray是android里为<Interger,Object>这样的Hashmap而专门写的类,目的是提高内存效率,其核心是折半查找函数(binar ...

  9. 新建一个浏览器APP

    安卓开发环境准备好了,试试新建一个浏览器程序吧 1.Start a new Android Studio Project 2.选这个像微信一样的样式 3.选择语言和版本 4.等待创建完成,拖一个Web ...

  10. IDEA项目追踪快捷键

    1.查看某个方法在哪里被调用: 在方法上右键选择FindUsages: 快捷键,在方法上Ctrl+G 2.从Ctroller方法直接跳过接口找到实现类方法: 在方法上右键:选择GoTo>Impl ...