明天就SX AFO了交篇题解%一下

这题大概是我第一道有独立思考切掉的紫题

之前的都是各种抄借鉴题解

为什么写这题的题解呢?另一个重要的原因是这样的↓

翻了翻已有题解中的几篇,下面几种情况屡见不鲜

  1. 样例2都过不去论题解是如何过审的

  2. 变量名不统一,有一些小的笔误

  3. 疯狂的防作弊系统,复制编译试图对拍然鹅数十行报错

正因如此,本蒟蒻想道自己写一篇题解,尽量保证不出现上面的错误,如果依然有笔误请轻喷

因为作者忙着复习颓废,\(Markdown\)及\(Latex\)会有一点偷懒,敬请谅解


说了一堆废话,下面进入正题

树形dp+换根

两遍dfs

这俩思想应该还是比较容易想到,像这种暴力搞不了的树上问题一看就是dp,再一个题目要求每一个点的情况,暴力跑一定会爆炸,所以需要用到换根

观察题目性质

  • 问题可以简化成,在某一节点\(i\)聚会时,此时的答案即为司机把所有人送回家再返回该点的距离减掉从点\(i\)到最远的人的家的距离。

  • 如果我们走的是简单路径(不知道简单路径是什么的可以当场AFO了),那么是不存在所谓的最短路的,因为这是一棵树,先送道远的和先送道近的没有本质区别

下面我们来设计状态

首先我们以\(1\)号节点为根

\(g[u]\) 从点u出发把所有在u的子树里的人送回家并返回u的距离

\(f[u]\) 从点\(u\)出发把所有人出发并返回\(u\)的距离

\(len[u]\) 从\(u\)出发,在\(u\)的子树内,距离\(u\)最远的那个人的家到\(u\)的距离

\(dlen[u]\) 从\(u\)出发,在\(u\)的子树内,距离\(u\)次远第二远的那个人的家到\(u\)的距离

\(up[u]\) 不在u的子树内,距离\(u\)最远的那个人的家到\(u\)的距离

\(siz[u]\) \(u\)及\(u\)的子树内有多少个人的家

则最终的答案应为

$ans[i] = f[i] - max(up[i],len[i] ) $

那么我们考虑代码

第一次\(dfs\)我们主要是处理子树内的问题

我们需要处理出$ g[]\(,\)len[]\(,\)dlen[]\(,\)siz[] $

这应该很简单

第二次就是换根操作了

我们需要处理出\(up[],f[]\)

我们需要分类讨论,设当前的点为\(u\),父亲节点为\(fa\),这条边的边权为 _\(w\)

  1. 当所有人的家都在\(u\)的子树内即\(siz[u]=k\)时,这时候\(f[u]\)即是\(g[u]\),而\(up[u]\)应当是\(0\),这很显然,因为\(u\)的子树外没有家

  2. 当所有人的家都不在u的子树内即\(siz[u]=0\)时,这是\(f[u]=f[fa]+2\)_\(w\),因为你首先要走到父亲节点这样才能送人回家,\(up[u]\)则完全由父亲节点更新(详见代码)

  3. \(u\)的子树内有家,\(u\)的子树外也有家时,这种情况可以手玩一下,\(f[u]\)就等于\(f[fa]\) ,那么$ up[u]\(呢?我们感性理解一下,\)up[u]\(时\)u\(从他的父亲那里继承而来的最长链,而\)len[u]\(,\)dlen[u]$则是从儿子那里继承过来的最长次长链那么我们显然要分两种情况

  • 当\(u\)在他父亲的\(len[fa]\)上时,\(up[u]\)就需要从\(dlen[fa]\)更新过来因为\(len[fa]\)要经过\(u\)

  • 当\(u\)不在父亲的\(len[fa]\)上时,同理由\(len[fa]\)转移过来

讲的比较抽象,建议画图+阅读代码理解

十年OI一场空,不开long long见祖宗

Code:

#include <bits/stdc++.h>
#define ll long long
using namespace std;
const int maxn = 5e+5+100;
struct Edge{
int v;
ll w;
};
int n,k;
int he[maxn],ne[maxn<<1],cnt=1;
Edge G[maxn<<1];
bool p[maxn];
int siz[maxn];
ll f[maxn],g[maxn],up[maxn],len[maxn],dlen[maxn]; void add_Edge(int u,int v,ll w){
G[cnt].v = v;
G[cnt].w = w;
ne[cnt] = he[u];
he[u] = cnt++;
}
void dfs(int u,int fa){
siz[u] = p[u];
for(int i=he[u];i;i=ne[i]){
int v=G[i].v;
int w=G[i].w;
if(v==fa) continue;
dfs(v,u);
siz[u]+=siz[v];
if(siz[v]){
g[u] += g[v] + 2*w;
if(len[v]+w>len[u]){
dlen[u] = len[u];
len[u] = len[v] + w;
}
else if(len[v]+w>dlen[u]){
dlen[u] = len[v] + w;
}
}
}
}
void dfs(int u,int fa,ll _w){
if(siz[u]==k){
f[u] = g[u];
up[u] = 0;
}
else if(siz[u]==0){
f[u] = f[fa] + 2*_w;
up[u] = max(up[fa],len[fa]) + _w;
}
else{
f[u] = f[fa];
if(len[fa]-len[u]==_w) up[u] = max(up[fa],dlen[fa]) + _w;
else up[u] = max(up[fa],len[fa]) + _w;
}
for(register int i=he[u];i;i=ne[i]){
int v=G[i].v;
int w=G[i].w;
if(v==fa) continue;
dfs(v,u,w);
}
}
int main(){
cin>>n>>k;
for(register int i=1;i<n;++i){
int u,v;
ll w;
scanf("%d%d%lld",&u,&v,&w);
add_Edge(u,v,w);
add_Edge(v,u,w);
}
for(register int i=1;i<=k;++i){
int x;
scanf("%d",&x);
p[x] = true;
}
dfs(1,0);
dfs(1,0,0);
for(register int i=1;i<=n;++i) printf("%lld\n",f[i]-max(up[i],len[i]));
return 0;
}

洛谷 P6419 Kamp 题解的更多相关文章

  1. 洛谷NOIp热身赛题解

    洛谷NOIp热身赛题解 A 最大差值 简单树状数组,维护区间和.区间平方和,方差按照给的公式算就行了 #include<bits/stdc++.h> #define il inline # ...

  2. 洛谷P2827 蚯蚓 题解

    洛谷P2827 蚯蚓 题解 题目描述 本题中,我们将用符号 ⌊c⌋ 表示对 c 向下取整. 蛐蛐国最近蚯蚓成灾了!隔壁跳蚤国的跳蚤也拿蚯蚓们没办法,蛐蛐国王只好去请神刀手来帮他们消灭蚯蚓. 蛐蛐国里现 ...

  3. 洛谷P1816 忠诚 题解

    洛谷P1816 忠诚 题解 题目描述 老管家是一个聪明能干的人.他为财主工作了整整10年,财主为了让自已账目更加清楚.要求管家每天记k次账,由于管家聪明能干,因而管家总是让财主十分满意.但是由于一些人 ...

  4. [POI 2008&洛谷P3467]PLA-Postering 题解(单调栈)

    [POI 2008&洛谷P3467]PLA-Postering Description Byteburg市东边的建筑都是以旧结构形式建造的:建筑互相紧挨着,之间没有空间.它们共同形成了一条长长 ...

  5. [NOI 2020 Online] 入门组T1 文具采购(洛谷 P6188)题解

    原题传送门 题目部分:(来自于考试题面,经整理) [题目描述] 小明的班上共有 n 元班费,同学们准备使用班费集体购买 3 种物品: 1.圆规,每个 7 元. 2.笔,每支 4 元. 3.笔记本,每本 ...

  6. [洛谷P3948]数据结构 题解(差分)

    [洛谷P3948]数据结构 Description 最开始的数组每个元素都是0 给出n,opt ,min,max,mod 在int范围内 A: L ,R ,X 表示把[l,R] 这个区间加上X(数组的 ...

  7. [CodePlus 2017 11月赛&洛谷P4058]木材 题解(二分答案)

    [CodePlus 2017 11月赛&洛谷P4058]木材 Description 有 n棵树,初始时每棵树的高度为 Hi ,第 i棵树每月都会长高 Ai.现在有个木料长度总量为 S的订单, ...

  8. 洛谷P1189 SEARCH 题解 迭代加深

    题目链接:https://www.luogu.com.cn/problem/P1189 题目大意: 给你一个 \(n \times m\) 的矩阵,其中有一些格子可以走,一些各自不能走,然后有一个点是 ...

  9. 洛谷 P5221 Product 题解

    原题链接 庆祝!第二道数论紫题. 推式子真是太有趣了! \[\prod_{i=1}^n \prod_{j=1}^n \frac{\operatorname{lcm}(i,j)}{\gcd(i,j)} ...

随机推荐

  1. [LeetCode]1084. 销售分析III(Mysql,having+聚合函数)

    题目 Table: Product +--------------+---------+ | Column Name | Type | +--------------+---------+ | pro ...

  2. 1. spring5源码 -- Spring整体脉络 IOC加载过程 Bean的生命周期

    可以学习到什么? 0. spring整体脉络 1. 描述BeanFactory 2. BeanFactory和ApplicationContext的区别 3. 简述SpringIoC的加载过程 4. ...

  3. Java 15 正式发布, 14 个新特性,刷新你的认知!!

    JDK 15 2020/09/15 如期而至! 这个时间牛逼啊,和苹果发布会同天? OracleJDK 15 发布地址: https://www.oracle.com/java/technologie ...

  4. 最好用的流程编辑器bpmn-js系列之基本使用

    最好用的流程编辑器bpmn-js系列文章 BPMN(Business Process Modeling Notation)是由业务流程管理倡议组织BPMI(The Business Process M ...

  5. 利用python简单实现unittest

    python3的eval方法 eval() 函数用来执行一个字符串表达式,并返回表达式的值 # 例如 a = [1,2,3,4] b = "a" print(eval(b)) # ...

  6. 关于bat中日期时间字符串的格式化

    在其他编程语言中,要实现日期时间字符串的格式化,包括时间计算,都是比较简单的 但在bat或者说cmd.dos中要实现这些功能.还是有一定难度的 首先,windows的cmd中可以使用%date%表示日 ...

  7. Oracle学习(十四)分表分区

    一.前言 大数据量的查询,不仅查询速度非常慢,而且还会导致数据库经常宕机,在尝试添加索引及查询方式修改后,还有没有更有效的解决方案呢? 分库.分表.分区这些概念咱就应该了解一下. 二.分表 假如一个大 ...

  8. BUU reverse xxor

    下载下来的是个elf文件,因为懒得上Linux,直接往IDA里扔, 切到字符串的那个窗口,发现Congratulation!,应该是程序成功执行的表示, 双击,按'x',回车跟入 找到主函数: 1 _ ...

  9. Java 审计之XXE篇

    Java 审计之XXE篇 0x00 前言 在以前XXE漏洞了解得并不多,只是有一个初步的认识和靶机里面遇到过.下面来 深入了解一下该漏洞的产生和利用. 0x01 XXE漏洞 当程序在解析XML输入时, ...

  10. Python-如何在一个for循环中迭代多个可迭代对象?

    案例: 某班学生期末考试成绩,语文.数学.英语分别存储在3个列表中,同时迭代三个列表.,计算每个学生的总分(并行) 某年级有4个班,某次英语成绩分别记录在4个列表中,依次迭代每个列表,统计全年级高于9 ...