Magic Girl Haze

T组

n个点,m条有向含权边,可以选择不超过k条边,将其权值变为0.

问点1到点n的最短距离是多少?

1≤T≤5n≤105m≤2×105k≤10wi≤109
1\leq T \leq 5 \\
n \leq 10^5 \\
m \leq 2\times10^5 \\
k \leq 10 \\
w_i \leq 10^9
1≤T≤5n≤105m≤2×105k≤10wi​≤109

二维状态Dijkstra

原本dijkstra算法是dis[v],只有点的编号一个维度,dis[v]状态表示的是1到v最短的距离。

现在令dis[k][v]表示恰好k条边变0,点1到v最短的距离,状态变为二维

原本是uuu到vvv有边就可以尝试状态转移

现在是

  1. (k,u)(k,u)(k,u)到(k,v)(k,v)(k,v)可以转移,代价w;
  2. (k,u)(k,u)(k,u)到(k+1,u)(k+1,u)(k+1,u)可以转移,代价0;

正确性是显然的

对于状态k=k0,∀u(k,u)k = k_0,\forall u(k,u)k=k0​,∀u(k,u),假设它们的初始值已经设置好了,则固定k=k0k=k_0k=k0​,它们之间的转移更新就是传统的一维dijkstra,因此是正确的

k=k0,∀u(k,u)k = k_0,\forall u (k,u)k=k0​,∀u(k,u)初始值本是无穷大, 但是k=K0−1k=K_0-1k=K0​−1的状态可以优化它们的初始值。

code

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
int t;
ll n, m;
int k;
const int maxn = 1e+5 + 5;
const int maxm = 2e+5 + 5;
const int maxk = 10;
vector<int> ev[maxn];
vector<ll> ew[maxn];
// bool solved[maxk + 1][maxn];
bool vis[maxk + 1][maxn];
ll dis[maxk + 1][maxn];
struct vnd
{
int v;
ll dis;
bool operator<(const struct vnd &obj) const
{
return dis > obj.dis;
}
void print() const {
cout << "(v,dis) = "<<v<<","<<dis<<endl;
}
};
priority_queue<vnd> q[maxk + 1]; void init()
{
cin >> n >> m >> k;
for (int u = 1; u <= n; ++u)
{
ev[u].clear();
ew[u].clear();
}
int u, v, w;
for (int i = 1; i <= m; ++i)
{
cin >> u >> v >> w;
ev[u].push_back(v);
ew[u].push_back(w);
}
// memset(solved, false, sizeof(solved));
memset(vis, false, sizeof(vis));
for (int i = 0; i <= k; ++i)
q[i] = priority_queue<vnd>();
} int main()
{
ios::sync_with_stdio(false);
cin.tie(0);
cin >> t;
while (t--)
{
init();
bool first = true;
ll ans;
vis[0][1] = true;
dis[0][1] = 0;
int u, v;
ll w;
for (int i = 0; i <= k; ++i)
{
// push visited vertices into queue
for (int u = 1; u <= n; ++u)
if (vis[i][u]) {
q[i].emplace(vnd{u,dis[i][u]});
}
while (!q[i].empty())
{
vnd t = q[i].top();
q[i].pop();
u = t.v;
for (size_t j = 0; j < ev[u].size(); ++j)
{
v = ev[u][j];
w = ew[u][j];
// do not change w
if (!vis[i][v])
{
vis[i][v] = true;
dis[i][v] = t.dis + w;
q[i].emplace(vnd{v,dis[i][v]});
}
else if (dis[i][v] > t.dis + w)
{
dis[i][v] = t.dis + w;
q[i].emplace(vnd{v,dis[i][v]});
}
// change w to 0
if (i + 1 > k)
continue;
if (!vis[i + 1][v])
{
vis[i + 1][v] = true;
dis[i + 1][v] = t.dis;
}
else if (dis[i + 1][v] > t.dis)
{
dis[i + 1][v] = t.dis;
}
}
}
if (vis[i][n])
{
if (first)
{
first = false;
ans = dis[i][n];
}
else
{
ans = min(ans, dis[i][n]);
}
}
}
// the input ensure: first == false
cout << ans << endl;
}
return 0;
}

A1958的更多相关文章

  1. ICPC 2018 南京网络赛 J Magical Girl Haze(多层图最短路)

    传送门:https://nanti.jisuanke.com/t/A1958 题意:n个点m条边的路,你有k次机会将某条路上的边权变为0,问你最短路径长度 题解:最短路变形,我们需要在常规的最短路上多 ...

随机推荐

  1. Git浅谈随笔之---如何工作

    其他的版本控制工具我们常见的还有SVN,关于这两者的区别,我们不多谈,详见 Git 与 SVN 的区别 : Git是一种版本控制工具.用来记录文件内容的变化,备以后查阅某个版本的情况的系统:我们在Gi ...

  2. [Redis-Python]发布订阅通过Redis异步发送邮件

    接收订阅 #!/usr/bin/env pyhton # coding:utf-8 # @Time : 2020-02-16 21:36 # @Author : LeoShi # @Site : # ...

  3. Java基础环境配置及HelloWorld

    一.什么是JDK,JRE JDK(Java Development Kit Java开发工具包) JDK是提供给Java开发人员使用的,其中包含了java的开发工具,也包括了JRE.所以安装了JDK, ...

  4. 珠峰-6-koa-express

    ####  这里写await和return 的区别是, await后边还可以写代码,而return 不会. ##### koa这里需要注意 这样子方法1比起方法2,少写n个url类似这样的key.

  5. UnsupportedClassVersionError : 不支持的类版本错误

    UnsupportedClassVersionError : 不支持的类版本错误 listenerStart配置类的应用程序侦听器时出错 listenerStart Error configuring ...

  6. SOLID原则都不知道,还敢说自己是搞开发的!

    面向对象编程(OOP)给软件开发领域带来了新的设计思想.很多开发人员在进行面向对象编程过程中,往往会在一个类中将具有相同目的/功能的代码放在一起,力求以最快的方式解决当下的问题.但是,这种编程方式会导 ...

  7. url相对路径变成绝对路径

    var eleLink = document.createElement('a'); eleLink.href = "/wordpress/?p=9227"; console.lo ...

  8. 802.11有线等效加密WEP

    有线等效加密(WEP)标准是802.11无线安全早期的解决方案,WEP并不安全. 既然WEP并不安全,为什么还要学习WEP呢? WEP简单,相比后续出现的加密协议,它不要求有多么强大的计算能力.一些老 ...

  9. C#设计模式学习笔记:(5)原型模式

    本笔记摘抄自:https://www.cnblogs.com/PatrickLiu/p/7640873.html,记录一下学习过程以备后续查用.  一.引言 很多人说原型设计模式会节省机器内存,他们说 ...

  10. if 语句 总结笔记

    1.if 语句 语法: if(condition) statement1; else statement2; graph TD A[JAVA考试] -->|几天后| B(收到成绩单) B --& ...