传送门


强行二合一最为致命

第一问直接最短路+$DFS$解决

考虑第二问,与深度相关,可以考虑长链剖分。

设$f_{i,j}$表示长度为$i$,经过边数为$j$时的最大边权和,考虑到每一次从重儿子转移过来的时候,不仅要将$f$数组右移一格,还需要同时加上一个值。显然用线段树等数据结构额外维护是不现实的,我们考虑维护一个影响范围为整个$f_i$的加法标记$tag_i$,将$f_{i,0}$设置为$-tag_i$,每一次上传的时候把标记也一起上传,合并轻儿子、计算答案的时候将这个$tag$加上,就能够做到快速地维护了。

长链剖分代码比点分治还长……

 #include<bits/stdc++.h>
 #define P pair < int , int >
 #define int long long
 //This code is written by Itst
 using namespace std;

 inline int read(){
     ;
     ;
     char c = getchar();
     while(c != EOF && !isdigit(c)){
         if(c == '-')
             f = ;
         c = getchar();
     }
     while(c != EOF && isdigit(c)){
         a = (a << ) + (a << ) + (c ^ ');
         c = getchar();
     }
     return f ? -a : a;
 }

 ;
 vector < P > e[MAXN];
 struct edge{
     int end , upEd , w;
 }Ed[MAXN << ];
 ] , g[MAXN << ] , sz[MAXN] , tag[MAXN] , *dp[MAXN] , *cnt[MAXN];
 int *p1 = f , *p2 = g , N , M , K , cntEd , ans , times;
 priority_queue < P > q;
 bool vis[MAXN];

 inline void addEd(int a , int b , int c){
     Ed[++cntEd].end = b;
     Ed[cntEd].upEd = head[a];
     Ed[cntEd].w = c;
     head[a] = cntEd;
 }

 void Dijk(){
     q.push(P( , ));
     memset(dis , 0x3f , sizeof(dis));
     dis[] = ;
     while(!q.empty()){
         P t = q.top();
         q.pop();
         if(-t.first > dis[t.second])
             continue;
          ; i < sz[t.second] ; ++i)
             if(dis[e[t.second][i].first] > dis[t.second] + e[t.second][i].second){
                 dis[e[t.second][i].first] = dis[t.second] + e[t.second][i].second;
                 q.push(P(-dis[e[t.second][i].first] , e[t.second][i].first));
             }
     }
 }

 void create(int now){
     vis[now] = ;
      ; i < sz[now] ; ++i)
         if(!vis[e[now][i].first] && dis[e[now][i].first] == dis[now] + e[now][i].second){
             addEd(now , e[now][i].first , e[now][i].second);
             addEd(e[now][i].first , now , e[now][i].second);
             create(e[now][i].first);
         }
 }

 void dfs1(int now , int pre){
     md[now] = dep[now] = dep[pre] + ;
     for(int i = head[now] ; i ; i = Ed[i].upEd)
         if(!dep[Ed[i].end]){
             dfs1(Ed[i].end , now);
             if(md[Ed[i].end] > md[now]){
                 md[now] = md[Ed[i].end];
                 son[now] = Ed[i].end;
                 len[now] = Ed[i].w;
             }
         }
 }

 void dfs2(int now){
     if(son[now]){
         dp[son[now]] = dp[now] + ;
         cnt[son[now]] = cnt[now] + ;
         dfs2(son[now]);
         tag[now] = tag[son[now]] + len[now];
         dp[now][] = -tag[now];
     }
     cnt[now][] = ;
     if(ans < dp[now][K] + tag[now]){
         ans = dp[now][K] + tag[now];
         times = cnt[now][K];
     }
     else
         if(ans == dp[now][K] + tag[now])
             times += cnt[now][K];
     for(int i = head[now] ; i ; i = Ed[i].upEd)
          && Ed[i].end != son[now]){
             dp[Ed[i].end] = p1;
             cnt[Ed[i].end] = p2;
             p1 += (md[Ed[i].end] - dep[Ed[i].end] + ) << ;
             p2 += (md[Ed[i].end] - dep[Ed[i].end] + ) << ;
             dfs2(Ed[i].end);
              ; j <= md[Ed[i].end] - dep[Ed[i].end] && j <= K -  ; ++j)
                  - j)
                      - j] + Ed[i].w){
                         ans = tag[Ed[i].end] + dp[Ed[i].end][j] + tag[now] + dp[now][K -  - j] + Ed[i].w;
                         times = cnt[Ed[i].end][j] * cnt[now][K -  - j];
                     }
                     else
                          - j] + Ed[i].w)
                             times += cnt[Ed[i].end][j] * cnt[now][K -  - j];
              ; j <= md[Ed[i].end] - dep[Ed[i].end] +  && j <= K ; ++j)
                 ] + tag[Ed[i].end] + Ed[i].w - tag[now]){
                     dp[now][j] = dp[Ed[i].end][j - ] + tag[Ed[i].end] + Ed[i].w - tag[now];
                     cnt[now][j] = cnt[Ed[i].end][j - ];
                 }
                 else
                     ] + tag[Ed[i].end] + Ed[i].w - tag[now])
                         cnt[now][j] += cnt[Ed[i].end][j - ];
         }
 }

 signed main(){
 #ifndef ONLINE_JUDGE
     freopen("2993.in" , "r" , stdin);
     //freopen("2993.out" , "w" , stdout);
 #endif
     N = read();
     M = read();
     K = read() - ;
      ; i <= M ; ++i){
         int a = read() , b = read() , c = read();
         e[a].push_back(P(b , c));
         e[b].push_back(P(a , c));
         ++sz[a];
         ++sz[b];
     }
      ; i <= N ; ++i)
         sort(e[i].begin() , e[i].end());
     Dijk();
     create();
     dfs1( , );
     dp[] = p1;
     p1 += md[] << ;
     cnt[] = p2;
     p2 += md[] << ;
     dfs2();
     cout << ans << ' ' << times;
     ;
 }

Luogu2993 FJOI2014 最短路径树问题 最短路树、长链剖分的更多相关文章

  1. (持续更新)虚树,KD-Tree,长链剖分,后缀数组,后缀自动机

    真的就是讲课两天,吸收一个月呢! \(1.\)虚树 \(2.\)KD-Tree \(3.\)长链剖分 \(4.\)后缀数组 后缀数组 \(5.\)后缀自动机 后缀自动机

  2. [FJOI2014]最短路径树问题 长链剖分

    [FJOI2014]最短路径树问题 LG传送门 B站传送门 长链剖分练手好题. 如果你还不会长链剖分的基本操作,可以看看我的总结. 这题本来出的很没水平,就是dijkstra(反正我是不用SPFA)的 ...

  3. Bzoj4016/洛谷P2993 [FJOI2014] 最短路径树问题(最短路径问题+长链剖分/点分治)

    题面 Bzoj 洛谷 题解 首先把最短路径树建出来(用\(Dijkstra\),没试过\(SPFA\)\(\leftarrow\)它死了),然后问题就变成了一个关于深度的问题,可以用长链剖分做,所以我 ...

  4. bzoj 4016 [FJOI2014]最短路径树问题(最短路径树+树分治)

    4016: [FJOI2014]最短路径树问题 Time Limit: 5 Sec  Memory Limit: 512 MBSubmit: 426  Solved: 147[Submit][Stat ...

  5. BZOJ_4016_[FJOI2014]最短路径树问题_最短路+点分治

    BZOJ_4016_[FJOI2014]最短路径树问题_最短路+点分治 Description 给一个包含n个点,m条边的无向连通图.从顶点1出发,往其余所有点分别走一次并返回. 往某一个点走时,选择 ...

  6. 洛谷 [FJOI2014]最短路径树问题 解题报告

    [FJOI2014]最短路径树问题 题目描述 给一个包含\(n\)个点,\(m\)条边的无向连通图.从顶点\(1\)出发,往其余所有点分别走一次并返回. 往某一个点走时,选择总长度最短的路径走.若有多 ...

  7. [BZOJ4016][FJOI2014]最短路径树问题

    [BZOJ4016][FJOI2014]最短路径树问题 试题描述 给一个包含n个点,m条边的无向连通图.从顶点1出发,往其余所有点分别走一次并返回. 往某一个点走时,选择总长度最短的路径走.若有多条长 ...

  8. [BZOJ4016][FJOI2014]最短路径树问题(dijkstra+点分治)

    4016: [FJOI2014]最短路径树问题 Time Limit: 5 Sec  Memory Limit: 512 MBSubmit: 1796  Solved: 625[Submit][Sta ...

  9. 【BZOJ4016】[FJOI2014]最短路径树问题 最短路径树+点分治

    [BZOJ4016][FJOI2014]最短路径树问题 Description 给一个包含n个点,m条边的无向连通图.从顶点1出发,往其余所有点分别走一次并返回. 往某一个点走时,选择总长度最短的路径 ...

随机推荐

  1. MySQL 性能优化--优化数据库结构之优化数据大小

    MySQL性能优化--优化数据库结构之优化数据大小   By:授客  QQ:1033553122 尽量减少表占用的磁盘空间.通常,执行查询期间处理表数据时,小表占用更少的内存. 表列 l   尽可能使 ...

  2. 性能测试 Apache参数配置与性能调优

    Apache性能调优 by:授客 QQ:1033553122 环境: Apache 2.4 1.选择合适的MPM(Multi -Processing Modules, 多处理模块) Unix/Linu ...

  3. XSS(跨站脚本攻击)漏洞解决方案

    首先,简单介绍一下XSS定义: 一 . XSS介绍 XSS是跨站脚本攻击(Cross Site Scripting)的缩写.为了和层叠样式表CSS(Cascading Style Sheets)加以区 ...

  4. bs4爬虫入门

    # -*- coding: utf-8 -*- """ Created on Fri Nov 16 13:35:33 2018 @author: zhen "& ...

  5. SQL Server ->> DAC(Dedicated Administrator Connection)专用管理员连接

    专用管理员连接是一种特殊的SQL Server服务器诊断性质的连接,用于在正常或者说标准连接形式无法连接SQL Server服务器的情况下连接到服务器进行服务器性能或者错误诊断.DAC同样支持安全加密 ...

  6. mybatis 初始

    接下来带着大家建立一个mybatis的初级项目 首先我们利用idea利用maven建立一个空项目 然后输入名称什么的就会创建一个空的maven项目了 然后我们需要在项目总得pom.xml中进行配置信息 ...

  7. Unity 4.6 GUI

    一起来窥探Unity的下一代GUI 预览 UI组件 UI结构 Canvas Button Selection List(滑动列表)

  8. dumpe2fs 命令的使用,转储 ext2/ext3/ext4 文件系统信息

    使用man 命令可以查看 dumpe2fs 命令具体的使用的方法: NAME dumpe2fs - dump ext2/ext3/ext4 filesystem information SYNOPSI ...

  9. 最大公约数&&最小公倍数

    //最大公约数(greatest common divisor),运用递归 int gcd(int a,int b){//注意a要求大于b return !b?a:gcd(b,a%b); } //最小 ...

  10. python 从外部获取传入的参数

    有时候我们在执行python程序的时需要接收到外部传入的参数 python的 sys.argv[]就能实现 # test.py import sys #引入模块 str = sys.argv[1]pr ...