P7276-送给好友的礼物【dp】
正题
题目链接:https://www.luogu.com.cn/problem/P7276?contestId=39577
题目大意
\(n\)个点的一棵树,\(k\)个关键点,两个人从根出发分别走一段路径回到根。要求每个关键点至少被一个人经过,求最短时间。
解题思路
相当于求两个覆盖所有关键点的联通子图,使得最大那棵最小。
树上只留下关键点和它们的祖先节点,这样就变为了所有点都要经过,也就是所有叶子都要经过。
然后比较显然的结论是一定会按照\(dfs\)从小到大的顺序走,所以可以直接\(dp\)。设\(f_{i,j,k}\)表示两个人分别走到\(i,j\),目前第二棵树的大小为\(k\)时第一棵树的最小大小。
注意转移\(x->i\)的时候权值得是\(dis(\ LCA(x,i),i\ )\),因为是联通子图上延伸下来一条路径。
时间复杂度\(O(n^3)\)
因为是比赛的时候写的代码改的,所以比较丑。
code
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int N=420;
struct node{
int to,next;
}a[N<<1];
int n,k,tot,num,ls[N],dep[N],f[N],lca[N][N],ans,dp[N][N][N],son[N];
bool mark[N];
void addl(int x,int y){
a[++tot].to=y;
a[tot].next=ls[x];
ls[x]=tot;return;
}
bool calc(int x,int fa){
dep[x]=dep[fa]+1;
for(int i=ls[x];i;i=a[i].next){
int y=a[i].to;
if(y==fa)continue;
f[y]=x;mark[x]|=calc(y,x);
}
return mark[x];
}
void dfs(int x,int fa){
if(!mark[x])return;
bool flag=1;
for(int i=ls[x];i;i=a[i].next){
int y=a[i].to;
if(y==fa||!mark[y])continue;
flag=0;break;
}
if(flag||x==1)son[++num]=x;
for(int i=ls[x];i;i=a[i].next){
int y=a[i].to;
if(y==fa)continue;
dfs(y,x);
}
return;
}
int LCA(int x,int y){
while(dep[x]>dep[y])x=f[x];
while(dep[y]>dep[x])y=f[y];
while(x!=y)x=f[x],y=f[y];
return x;
}
int dis(int x,int y)
{return dep[y]-dep[lca[x][y]];}
int main()
{
scanf("%d%d",&n,&k);
for(int i=1;i<n;i++){
int x,y;
scanf("%d%d",&x,&y);
addl(x,y);addl(y,x);
}
for(int i=1;i<=k;i++){
int x;scanf("%d",&x);
mark[x]=1;
}
calc(1,1);dfs(1,1);
for(int i=1;i<=n;i++)
for(int j=1;j<=n;j++)
lca[i][j]=LCA(i,j);
memset(dp,0x3f,sizeof(dp));
ans=dp[1][1][0];dp[1][1][0]=0;son[++num]=1;
for(int i=2;i<=num;i++)
for(int x=1;x<i;x++)
for(int y=1;y<i;y++){
if(x!=i-1&&y!=i-1)continue;
for(int z=0;z<n;z++){
int d=dis(son[x],son[i]);
dp[i][y][z]=min(dp[i][y][z],dp[x][y][z]+d);
d=dis(son[y],son[i]);
if(z>=d)dp[x][i][z]=min(dp[x][i][z],dp[x][y][z-d]);
}
}
for(int i=0;i<=num;i++)
for(int j=0;j<=num;j++){
if(i!=num&&j!=num)continue;
for(int z=0;z<n;z++)
ans=min(ans,max(dp[i][j][z],z));
}
printf("%d\n",ans*2);
}
P7276-送给好友的礼物【dp】的更多相关文章
- hpuoj--校赛--送给新生的礼物(水题)
问题 A: 感恩节KK专场--送给新生的礼物 时间限制: 1 Sec 内存限制: 128 MB 提交: 631 解决: 187 [提交][状态][讨论版] 题目描述 学长KK要送给学弟学妹们礼物, ...
- 【Codeforces 506E】Mr.Kitayuta’s Gift&&【BZOJ 4214】黄昏下的礼物 dp转有限状态自动机+矩阵乘法优化
神题……胡乱讲述一下思维过程……首先,读懂题.然后,转化问题为构造一个长度为|T|+n的字符串,使其内含有T这个子序列.之后,想到一个简单的dp.由于是回文串,我们就增量构造半个回文串,设f(i,j, ...
- Python3实现QQ机器人自动爬取百度文库的搜索结果并发送给好友(主要是爬虫)
一.效果如下: 二.运行环境: win10系统:python3:PyCharm 三.QQ机器人用的是qqbot模块 用pip安装命令是: pip install qqbot (前提需要有request ...
- NSURLSession 网络库 - 原生系统送给我们的礼物
大家在进行iOS开发的时候一定会用到网络操作.但由于早期原生的 NSURLConnection 操作起来有很多不便,使得大家更愿意使用第三方库的解决方案,比如鼎鼎大名的 AFNetworking.正是 ...
- 【COGS1049】天空中的繁星
[题目背景] 第二届『Citric』杯NOIP提高组模拟赛 第二题 [题目描述] Lemon最近买了一台数码相机.某天Lemon很无聊,于是对着夜空拍了一张照片,然后把照片导入了电脑.Lemon想依靠 ...
- 皮裤原理和运营微信公众号dotNET跨平台
经常碰到有同学对.NET跨平台存在各种疑惑和误解,原因是什么呢?当然我是知道.NET的跨平台不是问题,而且微软2014年的努力可圈可点,而且还有很多人对.NET的前景感到困惑.春节期间突然明白了,这就 ...
- K米点歌APP评测
K米APP评测 产品简介 K米点歌是一款免费的社交K歌手机应用,其手机点歌功能主要在KTV.夜总会,酒吧等K歌场所中使用,当前提供iPhone版本及安卓版本下载使用.——百度百科 评测版本 K米点歌4 ...
- C#知识体系(二)用案例来理解委托与事件
上一篇博客讲到了LinQ和lambda的常用方法 还有很多我们未知但c#设计团队已经为我们封装好的类和方法.随着我们不断的熟悉C#语言,渐渐的就会接触到其他的知识点,委托.事件.反射.线程.同步,异步 ...
- "2013":爱你不容易
2013对我来说确实像年初时曾给自己定义的那样,真的是非常不平常的一年.依稀记得去年年终时,BOSS和我深聊了1多钟头,谈到职业规划.人生还有家庭的林林种种.春节在家时也仔细考虑过2013自己该如何规 ...
随机推荐
- leaflet获取arcgis服务图层所有信息
L.esri.query({ url: "http://127.0.0.1:6080/arcgis/rest/services/demo/ditu/MapServer/0" }). ...
- IDEA不自动提示仓库中有的包maven
- 微信小程序学习笔记二 数据绑定 + 事件绑定
微信小程序学习笔记二 1. 小程序特点概述 没有DOM 组件化开发: 具备特定功能效果的代码集合 体积小, 单个压缩包体积不能大于2M, 否则无法上线 小程序的四个重要的文件 *js *.wxml - ...
- Git修改历史commit的author信息
前言 "嘀嗒嘀嗒",抬头看向墙上的钟表,此时已是凌晨1点.小明终于把Go语言圣经第二章的笔记写完,保存commit,提交,然后睡觉. 额,等等,不对,小明发现他用的是公司的git账 ...
- 用C++实现的有理数(分数)四则混合运算计算器
实现目标 用C++实现下图所示的一个console程序: 其中: 1.加减乘除四种运算符号分别用+.-.*./表示, + 和 - 还分别用于表示正号和负号. 2.分数的分子和分母以符号 / 分隔. 3 ...
- Ansible基础使用
原文转自:https://www.cnblogs.com/itzgr/p/10233932.html作者:木二 目录 一 Ansible命令用法 1.1 免密钥 1.2 Ad-Hoc基础命令 1.3 ...
- 堆栈相关的经典题(c++)
1.定义队列 typedef struct node{ int data; struct node * next; }Node; typedef struct linkQueue { Node * f ...
- awk 命令-随笔
awk语法: awk [option] 'pattern{action}' file ... awk [参数] '条件{动作}' 文件 ... 解析: 命令: awk 参数: -F "&qu ...
- Java基础(二)——内部类
一.内部类 内部类(Inner Class)就是定义在一个类里面的类.与之对应,包含内部类的类被称为外部类.内部类可以用private修饰. 1.为什么要定义内部类?或者内部类的作用是什么? 内部类提 ...
- Appium问题解决方案(6)- Java堆栈错误:java.lag.ClassNotFoundException:org.eclipse.swt.widets.Control
背景 运行脚本出现 SWT folder '..\lib\location of your Java installation.' does not exist. Please set ANDROID ...