<题目链接>

题目大意:
有n个城市,每一个城市有一个拥挤度$A_i$,从一个城市I到另一个城市J的时间为:$(A_v-A_u)^3$。问从第一个城市到达第$k$个城市所花的时间,如果不能到达,或者时间小于3输出?否则输出所花的时间。

解题分析:

很明显,此题路段的权值可能为负,所以我们就不能用Dijkstra算法求最短路了。需要注意的是,当点存在负环的时候,就要将负环所能够到达的所有点全部标记,从起点到这些点的最短路是不存在的(因为假设如果存在最短路,那么只要途中在负环上多走几遍,那么重新算得的时间一定会变少,所以不存在最短路)。所以,总的来说,对于本题,对那些负环能够到达的点,和从起点无法到达的点,和时间小于3的点,全部输出“?”,其他满足条件的直接输出最短时间就行。

#include <bits/stdc++.h>
using namespace std;
const int N = ,INF = 0x3f3f3f3f; struct Edge { int to,nxt,val; } edge[N * N]; int res, n;
int h[N];
bool vis[N]; //记录该点是否在队列内
bool cir[N]; //记录该点是否为负环上的点
int a[N], dist[N], cnt[N]; // cnt[]数组记录该数在队列中出现的次数 void dfs(int u) { //将该负环所能够达到的所有点全部标记
cir[u] = true;
for (int i = h[u]; i != -; i = edge[i].nxt) {
int v = edge[i].to;
if (!cir[v]) dfs(v);
}
}
void init() {
memset(h, -, sizeof(h));
res = ;
}
void add(int u, int v, int w) {
edge[res]=Edge{ v,h[u],w };h[u]=res++;
}
void SPFA(int st) {
memset(vis, false, sizeof(vis));
memset(cir, false, sizeof(cir));
memset(cnt, , sizeof(cnt));
for (int i = ; i <= n; i++) dist[i] = INF;
dist[st] = ;
queue<int> q;q.push(st);
cnt[st] = ;vis[st] = true;
while (!q.empty()) {
int u = q.front(); q.pop();
vis[u] = false; //当该点从队列中pop掉之后,就要清除vis标记
for (int i = h[u]; i != -; i = edge[i].nxt) {
int v = edge[i].to;
if (cir[v]) continue; //如果是负环上的点
if (dist[v] > dist[u] + edge[i].val) {
dist[v] = dist[u] + edge[i].val;
if (!vis[v]) { //如果该点不在队列中
vis[v] = true;
q.push(v);
cnt[v]++;
if (cnt[v] > n) dfs(v); //若存在负环,就用dfs将该负环能够达到的所有点标记
}
}
}
}
}
int main() {
int t; scanf("%d", &t);
int ncase = ;
while (t--) {
init();
scanf("%d", &n);
for (int i = ; i <= n; i++) scanf("%d", &a[i]);
int m; scanf("%d", &m);
while (m--) {
int u, v; scanf("%d %d", &u, &v);
add(u, v, (a[v] - a[u]) * (a[v] - a[u]) * (a[v] - a[u]));
}
SPFA();
printf("Case %d:\n", ++ncase);
scanf("%d", &m);
while (m--) {
int u;scanf("%d", &u);
if (cir[u] || dist[u] < || dist[u] ==INF) //如果询问的点能由负环达到、或者到起点的最小受益小于3、或者询问的点不可达
puts("?");
else printf("%d\n", dist[u]);
}
}
}

本题直接标记能够被负环上的点松弛>n次的点(包括负环上的点)也能过。

但是个人认为这种做法是不够严谨的,因为我的松弛是有终止条件的(因为没有必要,但是实际的问题时是能够在负环上一直转的),即,如果被松弛大于n次,就不再松弛,所以那些会被负环到达的点可能不会被松弛大于n次,比如它还能被别的路松弛,使得它到起点的距离非常短,这样负环上的点转了几圈之后才会小于它之前松弛的到起点的最短路,这样可能使得这个点进入队列的次数<n次(因为负环上的点我设置的是>n次自动终止)。

#include <bits/stdc++.h>
using namespace std;
const int N = , inf = 0x3f3f3f3f; struct Edge { int to,nxt,val; } edge[N * N]; bool vis[N], cir[N];
int n, h[N], res, a[N], dist[N], cnt[N]; //cnt[]数组记录该数在队列中出现的次数 void init() {
memset(h, -, sizeof(h));
res = ;
}
void add(int u, int v, int w) {
edge[res]=Edge{ v,h[u],w };h[u]=res++;
}
void SPFA(int st) {
memset(vis, false, sizeof(vis));
memset(cir, false, sizeof(cir));
memset(cnt, , sizeof(cnt));
for (int i = ; i <= n; i++) dist[i] = inf;
dist[st] = ;
queue<int> q;q.push(st);
cnt[st] = ;vis[st] = true;
while (!q.empty()) {
int u = q.front(); q.pop();
vis[u] = false;
for (int i = h[u]; i != -; i = edge[i].nxt) {
int v = edge[i].to;
if (cir[v]) continue;
if (dist[v] > dist[u] + edge[i].val) {
dist[v] = dist[u] + edge[i].val;
if (!vis[v]) {
vis[v] = true;
q.push(v);
cnt[v]++;
if(cnt[v]>n) cir[v]=true; //本题直接标记负环中的点以及能够被负环中的点松弛>n次的点也能过
}
}
}
}
}
int main() {
int t; scanf("%d", &t);
int ncase = ;
while (t--) {
init();
scanf("%d", &n);
for (int i = ; i <= n; i++) scanf("%d", &a[i]);
int m; scanf("%d", &m);
while (m--) {
int u, v; scanf("%d %d", &u, &v);
add(u, v, (a[v] - a[u]) * (a[v] - a[u]) * (a[v] - a[u]));
}
SPFA();
printf("Case %d:\n", ++ncase);
scanf("%d", &m);
while (m--) {
int u;scanf("%d", &u);
if (cir[u] || dist[u] < || dist[u] ==inf) puts("?");
else printf("%d\n", dist[u]);
}
}
}

类似于评论区里对最短距离的判断也能过(这与上面这个问题类似,但是也有一点区别)。

#include <bits/stdc++.h>
using namespace std;
const int N = , inf = 0x3f3f3f3f; struct Edge { int to,nxt,val; } edge[N * N]; bool vis[N], cir[N];
int n, h[N], res, a[N], dist[N], cnt[N]; void init() {
memset(h, -, sizeof(h));
res = ;
}
void add(int u, int v, int w) {
edge[res]=Edge{ v,h[u],w };h[u]=res++;
}
void SPFA(int st) {
memset(vis, false, sizeof(vis));
memset(cir, false, sizeof(cir));
memset(cnt, , sizeof(cnt));
for (int i = ; i <= n; i++) dist[i] = inf;
dist[st] = ;
queue<int> q;q.push(st);
cnt[st] = ;vis[st] = true;
while (!q.empty()) {
int u = q.front(); q.pop();
vis[u] = false;
for (int i = h[u]; i != -; i = edge[i].nxt) {
int v = edge[i].to;
if (cir[v]) continue;
if (dist[v] > dist[u] + edge[i].val) {
dist[v] = dist[u] + edge[i].val;
if (!vis[v]) {
vis[v] = true;
q.push(v);
cnt[v]++;
if(cnt[v]>n) cir[v]=true;
}
}
}
}
}
int main() {
int t; scanf("%d", &t);
int ncase = ;
while (t--) {
init();
scanf("%d", &n);
for (int i = ; i <= n; i++) scanf("%d", &a[i]);
int m; scanf("%d", &m);
while (m--) {
int u, v; scanf("%d %d", &u, &v);
add(u, v, (a[v] - a[u]) * (a[v] - a[u]) * (a[v] - a[u]));
}
SPFA();
printf("Case %d:\n", ++ncase);
scanf("%d", &m);
while (m--) {
int u;scanf("%d", &u);
if (dist[u] < || dist[u] == inf) puts("?");
else printf("%d\n", dist[u]);
}
}
}

LightOJ 1074 - Extended Traffic 【SPFA】(经典)的更多相关文章

  1. LightOj 1074 Extended Traffic (spfa+负权环)

    题目链接: http://lightoj.com/volume_showproblem.php?problem=1074 题目大意: 有一个大城市有n个十字交叉口,有m条路,城市十分拥挤,因此每一个路 ...

  2. LightOJ 1074 Extended Traffic SPFA 消负环

    分析:一看就是求最短路,然后用dij,果断错了一发,发现是3次方,有可能会出现负环 然后用spfa判负环,然后标记负环所有可达的点,被标记的点答案都是“?” #include<cstdio> ...

  3. LightOJ 1074 Extended Traffic(spfa+dfs标记负环上的点)

    题目链接:https://cn.vjudge.net/contest/189021#problem/O 题目大意:有n个站点,每个站点都有一个busyness,从站点A到站点B的花费为(busynes ...

  4. LightOJ - 1074 Extended Traffic (SPFA+负环)

    题意:N个点,分别有属于自己的N个busyness(简称b),两点间若有边,则边权为(ub-vb)^3.Q个查询,问从点1到该点的距离为多少. 分析:既然是差的三次方,那么可能有负边权的存在,自然有可 ...

  5. LightOJ 1074 - Extended Traffic (SPFA)

    http://lightoj.com/volume_showproblem.php?problem=1074 1074 - Extended Traffic   PDF (English) Stati ...

  6. LightOJ 1074 Extended Traffic (最短路spfa+标记负环点)

    Extended Traffic 题目链接: http://acm.hust.edu.cn/vjudge/contest/122685#problem/O Description Dhaka city ...

  7. lightoj 1074 - Extended Traffic(spfa+负环判断)

    题目链接:http://www.lightoj.com/volume_showproblem.php?problem=1074 题意:有n个城市,每一个城市有一个拥挤度ai,从一个城市I到另一个城市J ...

  8. SPFA(负环) LightOJ 1074 Extended Traffic

    题目传送门 题意:收过路费.如果最后的收费小于3或不能达到,输出'?'.否则输出到n点最小的过路费 分析:关键权值可为负,如果碰到负环是,小于3的约束条件不够,那么在得知有负环时,把这个环的点都标记下 ...

  9. (简单) LightOJ 1074 Extended Traffic,SPFA+负环。

    Description Dhaka city is getting crowded and noisy day by day. Certain roads always remain blocked ...

随机推荐

  1. window Maven私服搭建——nexus

    注:本文来源于 <window   Maven私服搭建--nexus> Maven私服搭建--nexus 1.下载nexus https://www.sonatype.com/downlo ...

  2. Oracle 闪回

    Oracle 闪回特性(FLASHBACK DATABASE) 本文来源于:gerainly 的<Oracle 闪回特性(FLASHBACK DATABASE) > -========== ...

  3. Confluence 6 启用 HTTP 压缩

    在屏幕的右上角单击 控制台按钮 ,然后选择 基本配置(General Configuration) 链接. 在左侧的面板中选择 通用配置(General Configuration). 启用 HTTP ...

  4. Confluence 6 在编辑器中控制参数的显示

    你可以决定宏参数在 Confluence 编辑器中如何进行显示的. 在默认的情况下,在宏占位符下尽可能显示能显示的所有参数: 你可以控制这里显示的参数数量,通过这种控制你可能尽量的为编辑者提供有效的信 ...

  5. js 判断输入的内容是否是整数

    需求简介:列表有一列排序,其值只能是整数,不能是小数,在js中判断,并给出提示 解决思路:在js中先获取表单的值,然后用isNaN,然后查一下怎么把小数排除在外.我靠( ‵o′)凸,这只能算是半路把! ...

  6. Eclipse搭建.C#和..NET Core环境

    1.在上一篇博客中我介绍了如何使用Eclipse搭建C++.C开发环境,顺带把搭建 .NET Core 和C#也做个介绍.配置任何环境关键是找到要开发语言的编辑器和SDK.eclipse是java开发 ...

  7. Java 获取屏幕的宽、高

    import java.awt.Toolkit; public class GetScreenSize { public static void main(String[] args) { int s ...

  8. C++ 关闭显示器

    好困,想躺一下,关灯.上床,笔记本的屏幕还亮着,好刺眼,睡不着! 脑子里出现一个疑问,怎么用C++写一个关闭屏幕的小程序呢? 参考了网上已有的例子,最简化: #include <windows. ...

  9. python字符编码和文件处理

    一.了解字符编码的知识储备 1.文本编辑器存取文件的原理(nodepad++,python,word) 打开编辑器就打开了启动了一个进程,是在内存中的,所以,用编辑器编写的内容也都是存放于内存中的,断 ...

  10. spring集成jwt验证方式,token验证

    为什么要告别session?有这样一个场景,系统的数据量达到千万级,需要几台服务器部署,当一个用户在其中一台服务器登录后,用session保存其登录信息,其他服务器怎么知道该用户登录了?(单点登录), ...