您的位置:首页 > 其它

POJ 2686 Traveling by Stagecoach 壮压DP

2013-09-16 18:56 267 查看
大意是有一个人从某个城市要到另一个城市(点数<=30)

然后有n个马车票,相邻的两个城市走的话要消耗掉一个马车票。

花费的时间呢,是马车票上有个速率值,用边/速率就是花的时间。

问最后这个人花费的最短时间是多少

然后就是壮压DP了

dp[S][v] 代表当前消耗了S集合的车票走到v花费的最小时间

可以用spfa转移。

也可以直接转移。

直接转的原因是,这个图由于走路要消耗车票,所以实质上图是个DAG

看两种代码

#include <iostream>
#include <cstdio>
#include <cstring>
#include <cmath>
#include <vector>
#include <algorithm>
#define MAXN 2222
#define INF 1000000007
using namespace std;
double dp[333][33];
typedef pair<int, int> P;
vector<P>g[33];
int t, n, m, src, des;
int num[33];
int main ()
{
int u, v, w;
while(scanf("%d%d%d%d%d", &t, &n, &m, &src, &des) != EOF)
{
if(!t && !n && !m && !src && !des) break;
for(int i = 0; i < t; i++) scanf("%d", &num[i]);
for(int i = 0; i <= n; i++) g[i].clear();
for(int i = 0; i < m; i++)
{
scanf("%d%d%d", &u, &v, &w);
g[u].push_back(make_pair(v, w));
g[v].push_back(make_pair(u, w));
}
for(int i = 0; i <= 300; i++)
for(int j = 0; j < 33; j++)
dp[i][j] = INF;
dp[0][src] = 0;
double res = INF;
for(int i = 0; i < (1 << t); i++)
{
for(u = 1; u <= n; u++)
for(int k = 0; k < t; k++)
if(!(i & (1 << k)))
{
for(int j = 0; j < g[u].size(); j++)
{
v = g[u][j].first;
w = g[u][j].second;
dp[i | (1 << k)][v] = min(dp[i | (1 << k)][v], dp[i][u] + (double)w / num[k]);
}
}
res = min(res, dp[i][des]);
}
if(res == INF) puts("Impossible");
else printf("%.3f\n", res);

}
return 0;
}


然后是SPFA

#include <iostream>
#include <cstdio>
#include <cstring>
#include <cmath>
#include <vector>
#include <queue>
#include <algorithm>
#define MAXN 2222
#define INF 1000000007
using namespace std;
double dp[333][33];
typedef pair<int, int> P;
vector<P>g[33];
int t, n, m, src, des;
int num[33], vis[333][33];
queue<P>q;
void spfa()
{
for(int i = 0; i <= 300; i++)
for(int j = 0; j < 33; j++)
dp[i][j] = INF;
memset(vis, 0, sizeof(vis));
dp[0][src] = 0;
vis[0][src] = 1;
while(!q.empty()) q.pop();
q.push(make_pair(0, src));
while(!q.empty())
{
P top = q.front();
q.pop();
int S = top.first;
int u = top.second;
for(int j = 0; j < t; j++)
{
if(S & (1 << j)) continue;
for(int i = 0; i < g[u].size(); i++)
{
int v = g[u][i].first;
int w = g[u][i].second;
if(dp[S | (1 << j)][v] > dp[S][u] + (double)w / num[j])
{
dp[S | (1 << j)][v] = dp[S][u] + (double)w / num[j];
if(!vis[S | (1 << j)][v])
{
q.push(make_pair(S | (1 << j), v));
vis[S | (1 << j)][v] = 1;
}
}
}
}
}
double res = INF;
for(int i = 0; i < (1 << t); i++)
res = min(res, dp[i][des]);
if(res == INF) puts("Impossible");
else printf("%.3f\n", res);
}
int main ()
{
int u, v, w;
while(scanf("%d%d%d%d%d", &t, &n, &m, &src, &des) != EOF)
{
if(!t && !n && !m && !src && !des) break;
for(int i = 0; i < t; i++) scanf("%d", &num[i]);
for(int i = 0; i <= n; i++) g[i].clear();
for(int i = 0; i < m; i++)
{
scanf("%d%d%d", &u, &v, &w);
g[u].push_back(make_pair(v, w));
g[v].push_back(make_pair(u, w));
}
spfa();
}
return 0;
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: