这题HDU 1595 find the longest of the shortest 就是一模一样的
但是出现了重边,就比较恶心人了
然后就将输入的数据先排序,然后出现重边的肯定在一块,然后对每条边存储的是它最短的情况和次短的情况即可
/*
ID: CUGB-wwj
PROG:
LANG: C++
*/
#include <iostream>
#include <vector>
#include <list>
#include <map>
#include <set>
#include <deque>
#include <queue>
#include <stack>
#include <bitset>
#include <algorithm>
#include <functional>
#include <numeric>
#include <utility>
#include <sstream>
#include <iomanip>
#include <cstdio>
#include <cmath>
#include <cstdlib>
#include <cctype>
#include <string>
#include <cstring>
#include <cmath>
#include <ctime>
#define INF 100000000
#define MAXN 1005
#define PI acos(-1.0)
using namespace std;
struct node
{
int v, next;
int w, worst;
void init(){worst = INF; }
}edge[MAXN * MAXN];
int vis[MAXN], pre[MAXN], head[MAXN], d[MAXN];
int q[MAXN * MAXN];
int n, m, e;
struct wwj
{
int u, v, w;
}in[MAXN * 100];
int uu[MAXN * 100], vv[MAXN * 100];
bool cmp(wwj x, wwj y)
{
if(x.u == y.u && x.v == y.v) return x.w < y.w;
else if(x.u == y.u) return x.v < y.v;
else return x.u < y.u;
}
void insert(int x, int y, int w)
{
int flag = 0;
for(int i = head[x]; i != -1; i = edge[i].next)
{
if(edge[i].v == y)
{
if(edge[i].worst == INF && edge[i].w != INF) edge[i].worst = w;
flag = 1;
break;
}
}
if(!flag)
{
edge[e].v = y;
edge[e].w = w;
edge[e].next = head[x];
head[x] = e++;
}
}
void spfa(int src)
{
for(int i = 1; i <= n; i++)
d[i] = INF;
int h = 0, t = 1;
memset(vis, 0, sizeof(vis));
d[src] = 0;
q[0] = src;
vis[src] = 1;
while(h < t)
{
int u = q[h++];
vis[u] = 0;
for(int i = head[u]; i != -1; i = edge[i].next)
{
int v = edge[i].v;
int w = edge[i].w;
if(d[v] > d[u] + w)
{
pre[v] = u;
d[v] = d[u] + w;
if(!vis[v])
{
q[t++] = v;
vis[v] = 1;
}
}
}
}
}
void init()
{
e = 0;
memset(head, -1, sizeof(head));
for(int i = 0; i < 5 * m; i++) edge[i].init();
}
void bechange(int u, int v)
{
for(int i = head[u]; i != -1; i = edge[i].next)
if(edge[i].v == v)
swap(edge[i].w, edge[i].worst);
}
int main()
{
int T;
scanf("%d", &T);
while(T--)
{
scanf("%d%d", &n, &m);
init();
for(int i = 0; i < m; i++)
{
scanf("%d%d%d", &in[i].u, &in[i].v, &in[i].w);
in[i + m].u = in[i].v;
in[i + m].v = in[i].u;
in[i + m].w = in[i].w;
}
sort(in, in + 2 * m, cmp);
for(int i = 0; i < 2 * m; i++)
insert(in[i].u, in[i].v, in[i].w);
spfa(1);
int ans = d[n];
int cnt = 0;
for(int i = n; i != 1; i = pre[i])
{
uu[cnt] = i;
vv[cnt] = pre[i];
cnt++;
}
for(int i = 0; i < cnt; i++)
{
bechange(uu[i], vv[i]);
bechange(vv[i], uu[i]);
spfa(1);
ans = max(ans, d[n]);
bechange(uu[i], vv[i]);
bechange(vv[i], uu[i]);
}
if(ans >= INF) puts("-1");
else
printf("%d\n", ans);
}
return 0;
}
分享到:
相关推荐
acm入门之枚举搜索,学校第一次acm培训,包括枚举及其优化,dfs和bfs
Problem Description 话说,经过了漫长的一个多月,小明已经成长了许多,所以他改了一个名字叫“大明”。 这时他已经不是那个只会做100以内加法的那个“小明”了,...请在一行里面输出输出A+B的值,请输出最简形式。
从简单入门到偏中等的几个题,线段树很灵活,主要懂了lazy操作,其他的自己yy吧。
本人准备2020年保研机试时刷的题目(虽然最后机试取消了,...来自某中流985,在HDU和vjudge平台上大概刷了400道。本文件地图(excel表格)包含了绝大部分我刷过的题目,笔记中具有思路、代码、总结和心得。 大佬勿入!
ACM题库,一些题目和答案,以及解题报告,传上来共享
杭电OnlineJudge 200-2099的解题报告
最短路(HDU-2544).rar
2、new做两件事,一是分配内存,二是调用类的构造函数 3、new建立的是一个对象,而malloc分配的是一块内存 4、new/delete是保留字,不需要头文
hdu 3341(ac自动机+状态压缩) 题意:容易理解... 思路:首先一开始容易想到要用到dp,开设一个dp[41][41][41][41][501]的数组来解决,但是明显内存已经超出范围了,于是就想如何减少内存呢?只要知道A、T、C、G其中...
acm hdu as easy as a+b
hdu 1695 GCD(欧拉函数+容斥原理).docx
HDU最全ac代码
300+ AC 代码 。 大数 , 线段树 , 字符串 , dp.....
hdu 5007 Post Robot 字符串枚举。 暴力一下就可以了。
HDU的1250,主要是利用高精度加法,但是代码有点繁琐,效率不是很高
HDU1059的代码
杭电ACMhdu1163
hdu1001解题报告
hdu 1574 passed sorce
HDU的一题........HDU DP动态规