题解 [APIO2014]连珠线

题面

解析

首先这连成的是一棵树啊.

并且\(yy\)一下,如果钦定一个根,

那么这上面的蓝线都是爸爸->儿子->孙子这样的,因为像下图这样的构造不出来:

(兄弟到兄弟的特殊情况不用考虑,因为会在一个端点作为根的情况考虑的)

那么首先还是来简单的写法,

设\(f[i][0/1]\)表示\(i\)是否为一根蓝线的中点的最大分数,

也可以理解为从\(i\)的一个儿子到\(i\)在上去还有没有蓝线.

并且,\(f[i][1]\)要算上它到父亲的边权.

然后再设\(c[i]\)=\(\max(f[i][0],f[i][1])\),

主要是懒得写

那么\(f[i][0]=\sum_{k=son[i]}c[k]\),

而\(f[i][1]=f[i][0]+\max(f[k][0]+w[k]-c[k])\),

其中\(w[k]\)表示\(k\)到父亲的边权(也就是i到k)

跑\(n\)遍dfs即可.

但这显然可以换根DP啊.

设\(dp[i]\)表示以\(i\)为根的最大分数,

\(v[i]\)表示\(i\)的父亲作为一条蓝边的中点,而\(i\)是一个端点的分数,并且也要再算上\(fa\)到\(i\)这条边.

(可以理解为f[fa][1]伸出去的那条边到了\(i\)这里)

那么有\(dp[i]=f[i][0]+max(dp[fa]-c[i],v[i])\)

就是\(i\)子树里的贡献加上父亲的贡献.

而父亲的贡献要么是不连边(\(dp[fa]-c[i]\)),要么就连边(v[i]).

(把\(f[i][0]\)式子里的\(c[k]\)换成\(c\)的定义就会发现很像)

然后考虑怎么求\(v\).

这里我们是用父亲去求儿子,

也就是当前是\(i\)时,我们考虑求\(i\)的儿子\(k\)(们)的\(v[k]\).

首先\(k\)是一个端点,那么我们要在\(i\)的儿子里再找出一个端点,

这里我们记一个\(mx1\)代表更新\(f[x][1]\)时后面那一串max(f[k][0]+w[k]-c[k])的最大值,

\(mx2\)表示次大值,\(id\)表示值为\(mx1\)的\(k\).

然后在求\(v[k]\)时,我们就有:

  • \(v[k]=dp[i]-c[k]+mx1+w[i]\),\(k\not=id\)

    这时我们可以直接拿最大值来贡献到\(k\)

  • \(v[k]=dp[i]-c[k]+mx2+w[i]\),\(k=id\)

    因为\(k\)已经是最大值的端点了,所以只能拿次大值来更新.

注意,\(mx1\)和\(mx2\)都要算上父亲!!!

显然父亲也会有贡献.

而父亲的贡献是dp[fa]-c[x]+w[i]-max(dp[fa]-c[x],v[x])

其实和上面的式子的结构是一样的(\(dp[fa]-c[x]\)就是\(f[k][0]\),\(\max(dp[fa]-c[x],v[x])\)就是\(c\))

然后就没有然后了

code:


  1. #include <iostream>
  2. #include <cstdio>
  3. #include <cstring>
  4. #define int long long
  5. using namespace std;
  6. inline int read(){
  7. int sum=0,f=1;char c=getchar();
  8. while(c>'9'||c<'0'){if(c=='-') f=-1;c=getchar();}
  9. while(c<='9'&&c>='0'){sum=(sum<<3)+(sum<<1)+c-'0';c=getchar();}
  10. return sum*f;
  11. }
  12. const int N=200005;
  13. const int INF=1e18;
  14. struct edge{int to,next,w;}e[N<<1];
  15. struct node{int mx1,mx2,id;}a[N];
  16. int n;
  17. int f[N][2],c[N],v[N],dp[N];
  18. int head[N],cnt=0;
  19. inline void add(int x,int y,int w){
  20. e[++cnt]=(edge){head[x],y,w};head[x]=cnt;
  21. }
  22. inline void dfs(int x,int fa){
  23. int ok=0;
  24. for(int i=head[x];i;i=e[i].to){
  25. int k=e[i].next;
  26. if(k==fa) continue;
  27. f[k][1]+=e[i].w;
  28. dfs(k,x);ok=1;
  29. f[x][0]+=c[k];
  30. if(f[k][0]+e[i].w-c[k]>a[x].mx1)
  31. a[x].mx2=a[x].mx1,a[x].mx1=f[k][0]+e[i].w-c[k],a[x].id=k;
  32. else if(f[k][0]+e[i].w-c[k]>a[x].mx2)
  33. a[x].mx2=f[k][0]+e[i].w-c[k];
  34. }
  35. f[x][1]+=f[x][0]+a[x].mx1;
  36. if(!ok) f[x][1]=-INF;
  37. c[x]=max(f[x][0],f[x][1]);
  38. }
  39. inline void dfs1(int x,int fa){
  40. dp[x]=f[x][0]+max(dp[fa]-c[x],v[x]);
  41. for(int i=head[x];i;i=e[i].to){
  42. int k=e[i].next;
  43. if(k==fa) continue;
  44. if(k==a[x].id) v[k]=dp[x]-c[k]+a[x].mx2+e[i].w;
  45. else v[k]=dp[x]-c[k]+a[x].mx1+e[i].w;
  46. int ret=dp[x]-c[k]+e[i].w-max(dp[x]-c[k],v[k]);
  47. if(ret>a[k].mx1) a[k].mx2=a[k].mx1,a[k].mx1=ret,a[k].id=x;
  48. else if(ret>a[k].mx2) a[k].mx2=ret;
  49. dfs1(k,x);
  50. }
  51. }
  52. signed main(){
  53. n=read();
  54. for(int i=1;i<n;i++){
  55. int x=read(),y=read(),w=read();
  56. add(x,y,w);add(y,x,w);
  57. }
  58. for(int i=1;i<=n;i++) a[i].mx1=a[i].mx2=-INF;
  59. dfs(1,0);dfs1(1,0);
  60. int ans=0;
  61. for(int i=1;i<=n;i++) ans=max(ans,dp[i]);
  62. printf("%lld\n",ans);
  63. return 0;
  64. }

题解 [APIO2014]连珠线的更多相关文章

  1. 【BZOJ3677】[Apio2014]连珠线 换根DP

    [BZOJ3677][Apio2014]连珠线 Description 在列奥纳多·达·芬奇时期,有一个流行的童年游戏,叫做“连珠线”.不出所料,玩这个游戏只需要珠子和线,珠子从1到礼编号,线分为红色 ...

  2. 【LG3647】[APIO2014]连珠线

    [LG3647][APIO2014]连珠线 题面 洛谷 题解 首先考虑一下蓝线连起来的情况,一定是儿子-父亲-另一个儿子或者是儿子-父亲-父亲的父亲. 而因为一开始只有一个点在当前局面上,将一条红边变 ...

  3. [Bzoj3677][Apio2014]连珠线(树形dp)

    3677: [Apio2014]连珠线 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 434  Solved: 270[Submit][Status] ...

  4. bzoj3677: [Apio2014]连珠线

    Description 在列奥纳多·达·芬奇时期,有一个流行的童年游戏,叫做“连珠线”.不出所料,玩这个游戏只需要珠子和线,珠子从1到礼编号,线分为红色和蓝色.游戏 开始时,只有1个珠子,而接下来新的 ...

  5. APIO2014 连珠线

    题目链接:戳我 换根DP 由于蒟蒻不会做这个题,所以参考了大佬. 本来想的是有三种情况,一种是该节点不作为两个蓝线的中点(我们称这种不是关键节点),一种是该节点作为关键点.连两个子节点,一种是作为关键 ...

  6. 并不对劲的bzoj3677:p3647:[APIO2014]连珠线

    题目大意 有一种生成\(n\)个点的树的方法为: 一开始有一个点,\(n-1\)次操作,每次可以有两种操作:1.选一个点,用一条红边将它与新点连接:2.将新点放在一条红边上,新点与这条红边两端点直接的 ...

  7. bzoj 3677: [Apio2014]连珠线【树形dp】

    参考:http://www.cnblogs.com/mmlz/p/4456547.html 枚举根,然后做树形dp,设f[i][1]为i是蓝线中点(蓝线一定是父子孙三代),f[i][0]为不是,转移很 ...

  8. Luogu P3647 [APIO2014]连珠线

    题目 换根dp. 显然对于给定的一棵有根树,蓝线都不能拐弯. 设\(f_{u,0}\)表示\(u\)不是蓝线中点时子树内的答案,\(f_{u,1}\)表示\(u\)是蓝线中点时子树内的答案.(以\(1 ...

  9. 洛谷$P3647\ [APIO2014]$连珠线 换根$dp$

    正解:换根$dp$ 解题报告: 传送门! 谁能想到$9102$年了$gql$居然还没写过换根$dp$呢,,,$/kel$ 考虑固定了从哪个点开始之后,以这个点作为根,蓝线只可能是直上直下的,形如&qu ...

随机推荐

  1. JVM -- 对象的概述和引用

    一.概述 说起垃圾收集(Garbage Collection,GC),大部分人都把这项技术当做java语言的伴生产物,然后GC出现历史比java久远. GC需要完成的3件事情: 1.哪些内存需要回收 ...

  2. 数据库相关概念讲解(java)

    1.常用类或接口介绍 1.DataSource接口 通过javaAPI中javax.sql.DataSource接口注释了解. 1.DataSource功能 如下图: 翻译: DataSource对象 ...

  3. django_视图相关

    使用通用视图(返回静态页面) from django.conf.urls.defaults import * from django.views.generic.simple import direc ...

  4. linux中部署jenkins(war包)及jenkins忘记登录账号密码

    未登录状态 登录状态 一:部署jenkins(war包) 1.直接下载war包jenkins.war,下载地址https://jenkins.io/download 2.将下载的war包放到服务器上t ...

  5. MySQL AND 和 OR 联合使用带来的坑

    MySQL 基础篇 三范式 MySQL 军规 MySQL 配置 MySQL 用户管理和权限设置 MySQL 常用函数介绍 MySQL 字段类型介绍 MySQL 多列排序 MySQL 行转列 列转行 M ...

  6. (二)ActiveMQ之点对点消息实现

    一.点对点消息实现概念 在点对点或队列模型下,一个生产者向一个特定的队列发布消息,一个消费者从该队列中读取消息.这里,生产者知道消费者的队列,并直接将消息发送到消费者的队列.这种模式被概括为:只有一个 ...

  7. C# 钉钉第三方开发接入

    钉钉开放平台 本文是针对钉钉开放平台的基于dotNetCore服务端开发和配置的描述 钉钉可开发的程序包括 企业内部应用,第三方企业应用,第三方个人应用 一.环境搭建 1.钉钉开发需要企业钉钉账号,如 ...

  8. C#常用数据结构

    常碰到的几种数据结构:Array,ArrayList,List,LinkedList,Queue,Stack,Dictionary<K,T>: 1.数组是最简单的数据结构.其具有如下特点: ...

  9. 利用Supervisor 管理自己部署的应用程序

    首先,在centos7下安装supervisor yum install python-setuptools easy_install supervisor 然后新建配置文件 #新建superviso ...

  10. express 和 vue-cli 的博客网站

    已经上传到github地址:https://github.com/13476075014/node-vue/tree/master/mynodeproject/15.TimeBlog # 个人博客系统 ...