传送门


强行二合一最为致命

第一问直接最短路+$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. 关于input的焦点事件

    关于input的焦点事件 $(".scanf_integral").focus(function(){//获取焦点//获取焦点后触发的事件 }) $(".scanf_in ...

  2. JMeter 参数化之利用CSV Data Set Config从文件读取参数并关联变量

    参数化之利用CSV Data Set Config从文件读取参数并关联变量   by:授客 QQ:1033553122 1.   添加CSV Data Set Config 右键线程组->配置元 ...

  3. JavaScript 字符串转json格式

    第一种:浏览器支持的转换方式(Firefox,chrome,opera,safari,ie)等浏览器: JSON.parse(jsonstr); //可以将json字符串转换成json对象 JSON. ...

  4. (网页)12种不宜使用的Javascript语法(转)

    转自阮一峰: 最近写的一些小东西,总是出各种各样的问题,用了angular.js反应居然比我的jQuery还慢,客户吐槽了,我又把一个小操作,改成了jQuery.浏览一下大神的的博客.转载一点东西: ...

  5. 《R数据挖掘入门》彩色插图(第9章)

    图9.5  图9.9

  6. NoSQL&Redis

    1.介绍NoSQL NoSQL(Not Only SQL):不仅仅是SQL,是一项全新的数据库理念,泛指非关系型数据库,原来我们所使用的MySQL.Oracle.Microsoft SQL Serve ...

  7. 使用NPOI导入导出Excel(xls/xlsx)数据到DataTable中

    using System; using System.Collections.Generic; using System.Text; using System.IO; using NPOI.SS.Us ...

  8. SELinux 是什么?

    一.SELinux的历史 SELinux全称是Security Enhanced Linux,由美国国家安全部(National Security Agency)领导开发的GPL项目,它拥有一个灵活而 ...

  9. idea+spring-boot+devtools热部署

    idea+spring-boot+devtools热部署 标签: spring-boot 2017-03-20 14:45 2635人阅读 评论(1) 收藏 举报  分类: spring-boot m ...

  10. jQuery 往table添加新内容有以下四个方法:

    Query 添加新内容有以下四个方法: append() - 在被选元素的结尾插入内容 prepend() - 在被选元素的开头插入内容 after() - 在被选元素之后插入内容 before() ...