题目链接:
https://vjudge.net/problem/ZOJ-2750
题目大意:
给定一本字典,字典里有很多成语,要求从字典里的第一个成语开始,运用字典里的成语变到最后一个成语,变得过程就是成语接龙,后一个成语的第一个字必须有前一个成语的最后一个字相等,给定的成语是4位16进制数字,每个成语前边跟的数字代表着找到这个成语之后再找到下个成语还需要t分钟 。(这里是找到该成语后还需要时间t,才能找到下一个成语)
思路:
每个成语有前面四个字符和后面四个字符,如果第i个成语后面字符等于第j个成语前面四个字符,那么这中间的消耗就是第i个成语自带的消耗T,按照这个原理就可以建图啦。然后用dijkstra求出从第一个成语到其他各个成语的最短路,然后判断最后一个成语是否可达,可达就输出d[n],不能就输出-1
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<queue>
#include<stack>
#include<map>
#include<sstream>
using namespace std;
typedef long long ll;
const int maxn = + ;
const int INF = << ;
int T, n, m, cases;
int Map[maxn][maxn];
bool v[maxn];
int d[maxn];
void dijkstra(int u0)
{
memset(v, , sizeof(v));
for(int i = ; i <= n; i++)d[i] = (i == u0 ? : INF);
for(int i = ; i <= n; i++)
{
int x, m = INF;
for(int j = ; j <= n; j++)if(!v[j] && d[j] <= m)m = d[x = j];
v[x] = ;
for(int j = ; j <= n; j++)
d[j] = min(d[j], d[x] + Map[x][j]);//松弛操作
}
}
struct node
{
string s1, s2;
int time;
};
node a[maxn];
int main()
{
while(cin >> n && n)
{
string s;
int x;
for(int i = ; i <= n; i++)
{
cin >> x >> s;
a[i].time = x;
a[i].s1 = a[i].s2 = "";
for(int j = , k = s.size() - ; j < ; j++, k++)
{
a[i].s1 += s[j];
a[i].s2 += s[k];
}
//cout<<a[i].s1<<" "<<a[i].s2<<endl;
}
for(int i = ; i <= n; i++)//建图
{
for(int j = ; j <= n; j++)
{
Map[i][j] = INF;
if(i == j)continue;
if(a[i].s2 == a[j].s1)Map[i][j] = a[i].time;
}
}
dijkstra();
if(d[n] >= INF)cout<<"-1"<<endl;
else cout<<d[n]<<endl;
}
return ;
}