A1958
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有边就可以尝试状态转移
现在是
- (k,u)(k,u)(k,u)到(k,v)(k,v)(k,v)可以转移,代价w;
- (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的更多相关文章
- ICPC 2018 南京网络赛 J Magical Girl Haze(多层图最短路)
传送门:https://nanti.jisuanke.com/t/A1958 题意:n个点m条边的路,你有k次机会将某条路上的边权变为0,问你最短路径长度 题解:最短路变形,我们需要在常规的最短路上多 ...
随机推荐
- PHP常见代码执行后门函数
PHP常见代码执行后门函数 代码执行: 应用程序在调用一些能够将字符串转换为代码的函数(例如php中的eval中),没有考虑用户是否控制这个字符串,将造成代码执行漏洞. 常见php代码执行函数介绍 常 ...
- 手把手教你使用数据可视化BI软件创建仓库可视化管理大屏
灯果数据可视化BI软件是新一代人工智能数据可视化大屏软件,内置丰富的大屏模板,可视化编辑操作,无需任何经验就可以创建属于你自己的大屏.大家可以在他们的官网下载软件. 本文以仓库可视化管理大屏为例为 ...
- SQL Server 2019 表无法修改问题
SQL Server 2019 表无法修改问题 问题描述: 解决方法: 1.在菜单栏中,点击工具->选项,示例: 2.在选项中单击设计器->表设计器和数据库设计器->取消勾选阻止保存 ...
- Linux运维--11.手动部署Galera Cluster
1.搭建galera集群 yum install epel-release yum install centos-release-openstack-stein #1.1 安装mariadb yum ...
- Java 中常见排序算法
经典的排序算法总结 冒泡排序算法 算法描述: 比较相邻的元素:如果第一个比第二个大,就交换它们两个: 对每一对相邻元素作同样的工作,从开始第一对到结尾的最后一对,这样在最后的元素应该会是最大的数: 针 ...
- Jmeter后置处理器,正则表达式提取器的使用
[使用场景]:下一个请求参数需要从上一个请求的响应数据中获取 [jmeter正则表达式说明]:使用perl正则表达式(可参考:http://www.runoob.com/perl/perl-regul ...
- Java架构-高并发的解决实战总结方案
Java架构-高并发的解决实战总结方案 1.应用和静态资源分离 刚开始的时候应用和静态资源是保存在一起的,当并发量达到一定程度的时候就需要将静态资源保存到专门的服务器中,静态资源主要包括图片.视频.j ...
- codeforces 995C
题意:从L到R 找有几个x,使x=a^p(a>0,p>1) 题解: 一开始把所有符合的次方都存到vector,然后MLE 可以看到1e6^3=1e18,所以可以将二次方单独来求,其他次方存 ...
- 02-flink时间语义 与 Window 基础概念与实现原理
Flink 多种时间语义对比 Flink 在流应用程序中支持不同的 Time 概念,就比如有 Processing Time.Event Time 和 Ingestion Time.下面我们一起来看看 ...
- SpringCloud踩坑
今天在使用 SpringCloud 时遇到了一个问题,感觉有不少小伙伴会遇到,所以记录下来 版本说明 SpringBoot 2.2.4.RELEASE SpringCloud Greenwich.SR ...