题意:给出一个有\(n\)个结点的有向图,边有边权。有\(q\)组询问,每次给出\(s,t,k\),问从\(s\)到\(t\)至少经过\(k\)条边的最短路。

\(n \leq 50, \, q \leq 10^5, \, k \leq 10^4\)

首先,注意到\(n\)非常小这个性质。对于很多这类点数少,询问不易维护也不复杂的图论题,可以用矩阵来做。

我们设原图的邻接矩阵为\(G\),并定义矩阵的二元运算\(\bigotimes\)为:

\[(A \bigotimes B)_{ij} = \min_{k} A_{ik} + B_{kj}
\]

那么,令\(A^k = \underbrace{A \bigotimes A \cdots A}_{k\text{ times}}\),那么\((G^k)_{ij}\)就等于从\(i\)到\(j\)恰好走\(k\)步的最短路。因此,每次询问的答案就是\(G^k \bigotimes S\),其中\(S\)为图floyd后得到的邻接矩阵。

接下来,让我们考虑这样一个暴力:一开始对于所有可能的路径长度\(l \space (l \leq k + n )\),求出\(G^l \bigotimes S\)。这样预处理是\(O(n^3 k)\),询问是\(O(1)\)。

这样做的话,预处理复杂过高,但每次询问能允许更高的复杂度(\(O(n)\))。注意到,询问要求的只是一个矩阵上的一个元素的罢了。因此,如果我们能把所有\(G^k \bigotimes S\)都表示为\(A \bigotimes B\)的形式,并且所有可能的\(A\)和\(B\)的总数量可以接受,就可以了。这也相当于把所有\(k\)拆分为两个数。

答案是分块。令块大小为\(H(k)\),那么我们可以把\(k\)分为\(\left\lfloor \frac {k} {H(k)} \right\rfloor \times H(k) + k \mod H(k)\)的形式。因此,我们令\(H(k) = \sqrt k\),那么,只要求出所有\(A_i = G^{i\sqrt k}, \ B_i = G_i \bigotimes S\),就可以\(O(n)\)回答每次询问,且\(|A| = |B| = \sqrt k\),故预处理复杂度也能达到\(O(n^3 \sqrt k)\)。

时间复杂度\(O(n^3 \sqrt k + nq)\)。

#include <bits/stdc++.h>
using namespace std;
const int N = 55, BAS = 100, INF = 0x3f3f3f3f;
typedef int mat[N][N];
int n,m,q;
mat a[N * 3],b[N * 3];
void mul(mat& x,mat y,mat z) {
memset(x,0x3f,sizeof(mat));
for (int k = 1 ; k <= n ; ++ k)
for (int i = 1 ; i <= n ; ++ i)
for (int j = 1 ; j <= n ; ++ j)
x[i][j] = min(x[i][j],y[i][k] + z[k][j]);
}
int main() {
int T,x,y,z,ret;
scanf("%d",&T);
while (T --) {
scanf("%d%d",&n,&m);
memset(a,0x3f,sizeof a);
memset(b,0x3f,sizeof b);
for (int i = 1 ; i <= m ; ++ i) {
scanf("%d%d%d",&x,&y,&z);
b[1][x][y] = min(b[1][x][y],z);
}
for (int i = 1 ; i <= n ; ++ i)
a[0][i][i] = 0, b[0][i][i] = 0;
for (int i = 2 ; i <= BAS + n ; ++ i)
mul(b[i],b[i-1],b[1]);
for (int i = 1 ; i <= n ; ++ i)
for (int j = 1 ; j <= n ; ++ j)
a[1][i][j] = b[100][i][j];
for (int i = 2 ; i <= BAS ; ++ i)
mul(a[i],a[i-1],a[1]);
for (int k = BAS + n - 1 ; k >= 0 ; -- k)
for (int i = 1 ; i <= n ; ++ i)
for (int j = 1 ; j <= n ; ++ j)
b[k][i][j] = min(b[k][i][j],b[k+1][i][j]);
scanf("%d",&q);
for (int i = 1 ; i <= q ; ++ i) {
scanf("%d%d%d",&x,&y,&z);
ret = INF;
for (int k = 1 ; k <= n ; ++ k)
ret = min(ret,a[z/BAS][x][k] + b[z%BAS][k][y]);
if (ret != INF) printf("%d\n",ret);
else puts("-1");
}
}
return 0;
}

小结:感觉分块白学了……还是只会死板地使用。

【做题】HDU6331 Walking Plan——矩阵&分块的更多相关文章

  1. hdu6331 Walking Plan

    题意: sol: 考虑floyed 直接暴力做的话复杂度是kn^3会炸. 考虑一个比较神仙的分块做法. 注意到我们是可以直接求单独某个k的矩阵,使用矩阵快速幂即可(取min的矩阵乘法). 单独求一次的 ...

  2. HDU6331 Problem M. Walking Plan【Floyd + 矩阵 + 分块】

    HDU6331 Problem M. Walking Plan 题意: 给出一张有\(N\)个点的有向图,有\(q\)次询问,每次询问从\(s\)到\(t\)且最少走\(k\)条边的最短路径是多少 \ ...

  3. [日记&做题记录]-Noip2016提高组复赛 倒数十天

    写这篇博客的时候有点激动 为了让自己不颓 还是写写日记 存存模板 Nov.8 2016 今天早上买了两个蛋挞 吃了一个 然后就做数论(前天晚上还是想放弃数论 但是昨天被数论虐了 woc noip模拟赛 ...

  4. 2018HDU多校训练-3-Problem M. Walking Plan

    链接:http://acm.hdu.edu.cn/showproblem.php?pid=6331 Walking Plan  Problem Description There are n inte ...

  5. HDU6331Problem M. Walking Plan

    传送门 分块floyd $f[i][j][k]$表示从i走k步到j的最短路 $g[i][j][k]$表示从i走k*100步到j的最短路 $h[i][j][k]$表示从i至少走k步到j的最短路 询问从x ...

  6. 最小割 总结&&做题记录

    模型要点: 1.一般适用于二取一问题或者01规划. 2.利用最小割=最大流,转化为最大流求之. 建议阅读胡伯涛的论文 <<最小割模型在信息学竞赛的应用>>,有精彩有序的证明和各 ...

  7. project euler做题记录

    ProjectEuler_做题记录 简单记录一下. problem 441 The inverse summation of coprime couples 神仙题.考虑答案为: \[\begin{a ...

  8. AtCoder Grand Contest 11~17 做题小记

    原文链接https://www.cnblogs.com/zhouzhendong/p/AtCoder-Grand-Contest-from-11-to-20.html UPD(2018-11-16): ...

  9. AtCoder Grand Contest 1~10 做题小记

    原文链接https://www.cnblogs.com/zhouzhendong/p/AtCoder-Grand-Contest-from-1-to-10.html 考虑到博客内容较多,编辑不方便的情 ...

随机推荐

  1. eclipse设置字体_字符编码_快捷键

    eclipse设置字体.字符编码.快捷键 1.设置字体: preferences->general->appearnce->colors and fonts-->basic-- ...

  2. oracle 排序 row_number() over(partition by 排序字段)

    业务描述:按t.truckId,t.riskCode 分组,每个分组里有分数,取分组中分数最大的那条记录. 如:A1 B1   5  6 A1  B1   5  3 A1  B2   2  5 A1 ...

  3. 解决 samba 服务器 windows 多重连接

    samba连接,用户名密码均正确.失败提示:不允许一个用户使用一个以上用户名与一个服务器或共享资源的多重连接. 事实上,这个不是samba的限制.是Windows的限制. 在打开存在public = ...

  4. ajax评论

    评论有好几种格式:有评论树.评论楼等的格式 发表评论注意事项: 1. 展示评论 1. 评论楼(Django模板语言渲染) 1. 从后端查询出所有的评论 2. 如果有父评论就展示父评论 2. 评论树 通 ...

  5. tft屏图像显示也成功完成

    2010-04-30 14:18:00 tft屏图像显示也成功完成. 其实有了刷屏的经验,图像显示就很简单. void address_set(uint x1,uint y1,uint x2,uint ...

  6. Nginx配置服务器静态文件支持跨域访问

    在server中配置 add_header Access-Control-Allow-Origin *; add_header Access-Control-Allow-Headers X-Reque ...

  7. Django项目----快速实现增删改查组件(起步阶段!!!)

    一.相关知识点回顾 1.什么是反射?   可以用字符串的方式去访问对象的属性 2.反射有四种方法? hasattr(object,name):判断一个对象是不是有name属性或者方法 getattr: ...

  8. Linux 执行程序 报错误:Permission denied.

    是对此文件所在位置没有权限导致的 chmod +x /home/yourfile 即可

  9. Mysql初级第三天(wangyun)

    1.JDBC简介 1).数据库驱动 2).SUN公司为统一数据库的操作,定义了一套Java操作数据库的规范,称之为JDBC. 3).JDBC全称:Java Database Connectivity( ...

  10. PyTorch 常用方法总结1:生成随机数Tensor的方法汇总(标准分布、正态分布……)

    在使用PyTorch做实验时经常会用到生成随机数Tensor的方法,比如: torch.rand() torch.randn() torch.normal() torch.linespace() 在很 ...