【LOJ】#2024. 「JLOI / SHOI2016」侦查守卫
题解
童年的回忆!
想当初,这是我考的第一次省选,我当时初二,我什么都不会,然后看着这个东西,是不是能用我一个月前才会的求lca,光这个lca我就调了一个多小时= =,然后整场五个小时,我觉得其他题不可做,一直杠这题的20分,然后。。。day1爆零了,之后day2手玩提答好像骗了一点,总归是没爆零
那么这个东西是可爱的树dp啦
首先我们考虑m = n(因为大体的代码都需要这个思路吧
我们设\(f[u][d]\)是u这个点不包括u,可以继续往上覆盖d个点最少需要的代价
\(g[u][d]\)是u这个点包括u自己还需要往下覆盖d个点最少需要的代价(和f有点开闭区间的意味哦,但是方便转移,g是闭区间,f是开区间
转移枚举u的儿子
\(f[u][d] = min(f[u][d],\sum_{v \in Son_u} f[v][d + 1] + \sum_{t = otherson}g[t][d])\)
\(g[u][d] = \sum_{v \in Son_u}g[v][d - 1]\)
\(g[u][ 0 ] = min(g[u][ 0 ],f[u][ 0 ])\)
如果选择某个点的话
\(f[u][d] = w[u] + \sum_{v \in Son_u} g[v][d]\)
我们还有一点,就是在计算完g数组和f数组后,要分别处理成g是前缀最小值,f是后缀最小值
可以这么想\(g[u][d - 1]\)能达到的最小值,\(g[u][d]\)也要能达到啊,显然我多覆盖一点,答案不会受影响
\(f[u][d + 1]\)能达到的最小值,\(f[u][d]\)也要能达到啊,显然我少覆盖一点,答案不会受影响
好的,我们刚才说的只是m = n的情况
如果不是,就证明有的点可以不覆盖,那么这么考虑,就是子树该覆盖的都需要被覆盖,自己不用被覆盖
那么就让\(f[u][ 0 ]\)的初始值为
\(f[u][ 0 ] = \sum_{v \in Son_u}g[v][ 0 ]\)
最后全部dp完成后答案就是\(f[1][ 0 ]\)
代码
#include <bits/stdc++.h>
#define MAXN 500005
//#define ivorysi
using namespace std;
typedef long long int64;
struct node {
int to,next;
}edge[MAXN * 2];
int head[MAXN],sumedge;
int n,d;
int f[MAXN][21],g[MAXN][21],w[MAXN];
bool appear[MAXN];
void add(int u,int v) {
edge[++sumedge].to = v;
edge[sumedge].next = head[u];
head[u] = sumedge;
}
void addtwo(int u,int v) {
add(u,v);add(v,u);
}
void dp(int u,int fa) {
for(int i = 0 ; i <= d ; ++i) g[u][i] = 0x7fffffff;
for(int i = 0 ; i <= d ; ++i) f[u][i] = 0x7fffffff;
vector<int> s;
for(int i = head[u] ; i ; i = edge[i].next) {
int v = edge[i].to;
if(v != fa) {
dp(v,u);
s.push_back(v);
}
}
if(!appear[u]) {
f[u][0] = 0;
for(auto v : s) {
f[u][0] += g[v][0];
}
}
for(int i = 0 ; i <= d; ++i) {
if(i != d) {
int sum = 0;
for(auto v : s) sum += g[v][i];
for(auto v : s) {
f[u][i] = min(f[u][i],f[v][i + 1] + sum - g[v][i]);
}
}
if(i != 0) {
g[u][i] = 0;
for(auto v : s) g[u][i] += g[v][i - 1];
}
}
f[u][d] = w[u];
for(auto v : s) {
f[u][d] += g[v][d];
}
for(int i = d - 1 ; i >= 0 ; --i) {
f[u][i] = min(f[u][i + 1],f[u][i]);
}
g[u][0] = min(g[u][0],f[u][0]);
for(int i = 1 ; i <= d ; ++i) {
g[u][i] = min(g[u][i],g[u][i - 1]);
}
s.clear();
}
void Init() {
scanf("%d%d",&n,&d);
for(int i = 1 ; i <= n ; ++i) scanf("%d",&w[i]);
int m,u,v;
scanf("%d",&m);
for(int i = 1 ; i <= m ; ++i) {
scanf("%d",&u);
appear[u] = 1;
}
for(int i = 1 ; i < n ; ++i) {
scanf("%d%d",&u,&v);
addtwo(u,v);
}
}
int main() {
#ifdef ivorysi
freopen("f1.in","r",stdin);
#endif
Init();
dp(1,0);
printf("%d\n",f[1][0]);
}
【LOJ】#2024. 「JLOI / SHOI2016」侦查守卫的更多相关文章
- loj #2024. 「JLOI / SHOI2016」侦查守卫
#2024. 「JLOI / SHOI2016」侦查守卫 题目描述 小 R 和 B 神正在玩一款游戏.这款游戏的地图由 nnn 个点和 n−1n - 1n−1 条无向边组成,每条无向边连接两个点, ...
- loj2024「JLOI / SHOI2016」侦查守卫
too hard #include <iostream> #include <cstdio> using namespace std; int n, d, m, uu, vv, ...
- loj #2026. 「JLOI / SHOI2016」成绩比较
#2026. 「JLOI / SHOI2016」成绩比较 题目描述 THU 的 G 系中有许许多多的大牛,比如小 R 的室友 B 神.B 神已经厌倦了与其他的同学比较 GPA(Grade Poin ...
- loj #2025. 「JLOI / SHOI2016」方
#2025. 「JLOI / SHOI2016」方 题目描述 上帝说,不要圆,要方,于是便有了这道题. 由于我们应该方,而且最好能够尽量方,所以上帝派我们来找正方形.上帝把我们派到了一个有 NNN ...
- LOJ #2026「JLOI / SHOI2016」成绩比较
很好的锻炼推柿子能力的题目 LOJ #2026 题意 有$n$个人$ m$门学科,第$ i$门的分数为不大于$U_i$的一个正整数 定义A「打爆」B当且仅当A的每门学科的分数都不低于B的该门学科的分数 ...
- 【LOJ】 #2025. 「JLOI / SHOI2016」方
题解 有什么LNOI啊,最后都是JLOI罢了 一道非常--懵逼的统计题 当然是容斥,所有的方案 - 至少有一个点坏掉的正方形 + 至少有两个点坏掉的正方形 - 至少有三个点坏掉的正方形 + 至少有四个 ...
- 【LOJ】#2026. 「JLOI / SHOI2016」成绩比较
题解 用\(f[i][j]\)表示考虑了前i个排名有j个人被碾压 \(f[i][j] = f[i - 1][k] \* C[k][j] \* C[N - k - 1][N - r[i] - j] \* ...
- loj2026 「JLOI / SHOI2016」成绩比较
orz #include <iostream> #include <cstdio> using namespace std; typedef long long ll; int ...
- Loj #2495. 「AHOI / HNOI2018」转盘
Loj #2495. 「AHOI / HNOI2018」转盘 题目描述 一次小 G 和小 H 原本准备去聚餐,但由于太麻烦了于是题面简化如下: 一个转盘上有摆成一圈的 \(n\) 个物品(编号 \(1 ...
随机推荐
- java基础-BigInteger类常用方法介绍
java基础-BigInteger类常用方法介绍 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.BigInteger类概述 Java中long型为最大整数类型,对于超过long ...
- [非常重要的总结] Linux C相关函数
(1)字符测试函数 isalnum(测试字符是否为英文字母或数字) isalpha(测试字符是否为英文字母) isascii(测试字符是否为ASCII码字符) isblank(测试字符是否为空格字符) ...
- element ui 栅格布局
<el-row> <el-col :span="24"><div class="grid-content bg-purple-dark&qu ...
- python获取url响应
前言 requests发请求时,接口的响应时间,也是我们需要关注的一个点,如果响应时间太长,也是不合理的.如果服务端没及时响应,也不能一直等着,可以设置一个timeout超时的时间 关于request ...
- 容斥 或者 单调栈 hihocoder #1476 : 矩形计数 和 G. Snake Rana 2017 ACM Arabella Collegiate Programming Contest
先说一个简单的题目(题目大意自己看去,反正中文):hihocoder上的:http://hihocoder.com/problemset/problem/1476 然后因为这个n和m的矩阵范围是100 ...
- Java并发编程原理与实战二十三:Condition原理分析
先来回顾一下java中的等待/通知机制 我们有时会遇到这样的场景:线程A执行到某个点的时候,因为某个条件condition不满足,需要线程A暂停:等到线程B修改了条件condition,使condit ...
- 【Foreign】动态规划 [分治][DP]
动态规划 Time Limit: 50 Sec Memory Limit: 128 MB Description 一开始有n个数,一段区间的价值为这段区间相同的数的对数. 我们想把这n个数切成恰好k ...
- Vue 表格内容根据后台返回状态位填充文字
本文地址:http://www.cnblogs.com/veinyin/p/8534365.html Vue 做表格时我们常用的就是 v-for ,直接把 prop 绑定上去,但是如果表格内容需要我 ...
- 【转】C#中PrintDocument类详解
PrintDocument组件是用于完成打印的类,其常用属性.方法和事件如下: 属性DocumentName:字符串类型,记录打印文档时显示的文档名(例如,在打印状态对话框或打印机队列中显示). 方法 ...
- C语言入门教程-(3)基本数据类型
1.数据类型 在C语言中,数据类型指的是用于声明不同类型的变量或函数的一个广泛的系统.C语言数据类型可以分为四种: 1.基本类型:它们是算术类型,包括两种类型:整数类型和浮点类型. 2.枚举类型:它们 ...