题目描述

给定一张n个点m条边的带权有向图,每条边的边权只可能是1,2,3中的一种。
将所有可能的路径按路径长度排序,请输出第k小的路径的长度,注意路径不一定是简单路径,即可以重复走同一个点。

输入

第一行包含三个整数n,m,k(1<=n<=40,1<=m<=1000,1<=k<=10^18)。
接下来m行,每行三个整数u,v,c(1<=u,v<=n,u不等于v,1<=c<=3),表示从u出发有一条到v的单向边,边长为c。
可能有重边。

输出

包含一行一个正整数,即第k短的路径的长度,如果不存在,输出-1。

样例输入

6 6 11
1 2 1
2 3 2
3 4 2
4 5 1
5 3 1
4 6 3

样例输出

4

提示

长度为1的路径有1->2,5->3,4->5。
长度为2的路径有2->3,3->4,4->5->3。
长度为3的路径有4->6,1->2->3,3->4->5,5->3->4。
长度为4的路径有5->3->4->5。

因为边权有三种,但边数比较多,因此不能拆边。但点数比较少可以把每个点拆成三个点,同一个点拆成的三个点要连上边,这样就能使边权都是1了。

很容易想到用二分答案来求第k短路径,但这是log2,显然过不去,因此可以预处理出矩阵乘法的2i的矩阵,每次像倍增lca一样如果能走这么多步就走,不能走就尝试2i-1的矩阵的答案数。

那么怎么统计答案?

可以建一个原点(0号点)连向所有拆点后的原图节点,再将原点连向自己,这样第一行每个数就是原点到达对应点步数小于等于矩阵幂次的总路径数。

但这样求的是2i-1步数的答案,因此还要记录每个点的出度,统计时将每个答案乘上对应点的出度即可。

因为k比较大,矩阵乘法过程中会爆longlong,对于两个数加起来爆longlong,那么结果一定是负数,实际结果也就一定大于k,矩乘和求答案时判一下即可。

#include<set>
#include<map>
#include<queue>
#include<stack>
#include<cmath>
#include<bitset>
#include<vector>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#define ll long long
using namespace std;
ll a[65][125][125];
ll b[125][125];
ll c[125][125];
int mask;
ll ans;
ll K;
int cnt;
int f[45][3];
int v[121];
int n,m;
int x,y,z;
void multiply(ll a[125][125],ll b[125][125],ll c[125][125])
{
for(int i=0;i<=cnt;i++)
{
for(int j=0;j<=cnt;j++)
{
c[i][j]=0;
for(int k=0;k<=cnt;k++)
{
if(a[i][k]&&b[k][j])
{
if(a[i][k]<0||b[k][j]<0)
{
c[i][j]=-1;
break;
}
if(a[i][k]>K/b[k][j])
{
c[i][j]=-1;
break;
}
c[i][j]+=a[i][k]*b[k][j];
if(c[i][j]<0)
{
c[i][j]=-1;
break;
}
}
}
}
}
}
bool check()
{
ll res=0;
for(int i=0;i<=cnt;i++)
{
if(c[0][i]&&v[i])
{
if(c[0][i]<0)
{
return 0;
}
if(c[0][i]>K/v[i])
{
return 0;
}
res+=c[0][i]*v[i];
if(res<0)
{
return 0;
}
}
}
return res<K;
}
int main()
{
scanf("%d%d%lld",&n,&m,&K);
for(int i=1;i<=n;i++)
{
for(int j=0;j<=2;j++)
{
f[i][j]=++cnt;
}
}
a[0][0][0]++;
for(int i=1;i<=n;i++)
{
for(int j=0;j<=1;j++)
{
a[0][f[i][j]][f[i][j+1]]++;
}
a[0][0][f[i][0]]++;
}
while(m--)
{
scanf("%d%d%d",&x,&y,&z);
a[0][f[x][z-1]][f[y][0]]++;
v[f[x][z-1]]++;
}
for(mask=0;(1ll<<mask)<=K*3;mask++);
mask--;
for(int i=1;i<=mask;i++)
{
multiply(a[i-1],a[i-1],a[i]);
}
for(int i=0;i<=cnt;i++)
{
b[i][i]=1;
}
for(int i=mask;i>=0;i--)
{
multiply(b,a[i],c);
if(check())
{
ans|=1ll<<i;
for(int j=0;j<=cnt;j++)
{
for(int k=0;k<=cnt;k++)
{
b[j][k]=c[j][k];
}
}
}
}
ans++;
if(ans>K*3)
{
ans=-1;
}
printf("%lld",ans);
}

BZOJ4386[POI2015]Wycieczki——矩阵乘法+倍增的更多相关文章

  1. 【bzoj4386】[POI2015]Wycieczki 矩阵乘法

    题目描述 给定一张n个点m条边的带权有向图,每条边的边权只可能是1,2,3中的一种.将所有可能的路径按路径长度排序,请输出第k小的路径的长度,注意路径不一定是简单路径,即可以重复走同一个点. 输入 第 ...

  2. BZOJ4386 [POI2015]Wycieczki 矩阵+倍增

    题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=4386 题解 一眼就可以看出来是邻接矩阵快速幂. 可是这里的边权不为 \(1\).不过可以发现, ...

  3. BZOJ 4386 Luogu P3597 [POI2015]Wycieczki (矩阵乘法)

    题目链接: (bzoj) https://www.lydsy.com/JudgeOnline/problem.php?id=4386 (luogu) https://www.luogu.org/pro ...

  4. CF_576D_Flights for Regular Customers_矩阵乘法+倍增floyd+bitset+bfs

    CF_576D_Flights for Regular Customers_矩阵乘法+倍增floyd+bitset https://www.luogu.org/problemnew/show/CF57 ...

  5. BZOJ4386 : [POI2015]Wycieczki

    将每个点拆成三个点,并将转移转化为矩阵乘法,然后倍增即可求出第$k$短路的长度,注意对爆long long情况的处理. 时间复杂度$O(n^3\log k)$. #include<cstdio& ...

  6. bzoj 2165: 大楼【Floyd+矩阵乘法+倍增+贪心】

    1<<i的结果需要是long long的话i是long long是没用的--要写成1ll<<i--我别是个傻子吧 虽然写的是二进制贪心,但是我觉得二分可能更好写吧(但是会慢) ...

  7. BZOJ4386[POI2015]Wycieczki / Luogu3597[POI2015]WYC - 矩乘

    Solution 想到边权为$1$的情况直接矩乘就可以得出长度$<=t$ 的路径条数, 然后二分check一下即可 但是拓展到边权为$2$,$3$ 时, 需要新建节点 $i+n$ 和 $i+2n ...

  8. 【BZOJ-4386】Wycieczki DP + 矩阵乘法

    4386: [POI2015]Wycieczki Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 197  Solved: 49[Submit][Sta ...

  9. CF781D Axel and Marston in Bitland [倍增 矩阵乘法 bitset]

    Axel and Marston in Bitland 好开心第一次补$F$题虽然是$Div.2$ 题意: 一个有向图,每条边是$0$或$1$,要求按如下规则构造一个序列然后走: 第一个是$0$,每次 ...

随机推荐

  1. java 面向对象抽象类和接口

    1.abstract:可以修饰类和方法,被abstract修饰过的称为抽象类和抽象方法. 抽象类: 除了不可以创建对象, 其他和普通类一致. 可以有成员方法.静态方法和构造方法(提供给子类调用)等. ...

  2. ASP.NET的生命周期

    我主要参考了这些文章 ASP.NET应用程序与页面生命周期, IIS处理Asp.net请求和 Asp.net页面生命周期 asp.net页面的生命周期 页面生命周期开始 (一)页面生命周期的主要阶段包 ...

  3. 阿里云ubuntu 16.04搭建odoo11服务器

    ubuntu 16.04 具体如何搭建odoo11网站的具体步骤可以参考这一篇文章 按上面的文章配置环境后,自己网站的启动具体步骤如下: 1.登录阿里云 [远程连接],进入命令行界面1 2.cd到目录 ...

  4. C# 如何物理删除有主外键约束的记录?存储过程实现

    十年河东,十年河西,莫欺少年穷 本篇主旨是如何物理删除有主外键约束的记录!那么,我们从主外键走起! 下面新建三张有主外键约束的表,分别为:系/学院表,专业班表,学生表,如下: CREATE TABLE ...

  5. Vue 开发环境搭建 (Mac)

    一.初识 由于个人工作原因以及技术需要一个提升,略晚的开始初探Vue ~.~ 二.那么Vue是什么呢? 他就是一个前端的框架,特点是数据双向绑定.组件化. 三.推荐开发环境 四.环境安装 打开终端运行 ...

  6. Jlink使用技巧之烧写SPI Flash存储芯片

    前言 大多数玩单片机的人都知道Jlink可以烧写Hex文件,作为ARM仿真调试器,但是知道能烧写SPI Flash的人应该不多,本篇文章将介绍如何使用JLink来烧写或者读取SPI Flash存储器, ...

  7. 记一次拿webshell踩过的坑(如何用PHP编写一个不包含数字和字母的后门)

    0x01 前言 最近在做代码审计的工作中遇到了一个难题,题目描述如下: <?php include 'flag.php'; if(isset($_GET['code'])){ $code = $ ...

  8. BugkuCTF web基础$_GET

    前言 写了这么久的web题,算是把它基础部分都刷完了一遍,以下的几天将持续更新BugkuCTF WEB部分的题解,为了不影响阅读,所以每道题的题解都以单独一篇文章的形式发表,感谢大家一直以来的支持和理 ...

  9. Redis Cluster日常操作命令梳理

    在之前的一篇文章已经介绍了Redis Cluster及其部署,下面说下Redis Cluster日常操作命令: 一.以下命令是Redis Cluster集群所独有的,执行下面命令需要先登录redis: ...

  10. nginx域名访问的白名单配置梳理

    在日常运维工作中,会碰到这样的需求:设置网站访问只对某些ip开放,其他ip的客户端都不能访问.可以通过下面四种方法来达到这种效果:1)针对nginx域名配置所启用的端口(比如80端口)在iptable ...