您的位置:首页 > Web前端 > JavaScript

[0513]tyvj P1842 [JSOI2010]快递服务

2012-05-13 15:15 363 查看
背景 Background
JSOI2009第二轮一试
描述 Description
  「飞奔」快递公司成立之后,已经分别与市内许多中小企业公司签订邮件收送服务契约。由于有些公司是在同一栋大楼内,所以「飞奔」公司收件的地点(收件点)最多只有m点 (1, 2, …, m),因此「飞奔」仅先行采购了三辆货車并聘用了三名司机,每天早上分别从收件地点 「1」, 「2」 及 「3」出发。而在与客户的服务契约中有明确订约:「飞奔」必须在客户提出邮件寄送要求的隔天派人至该公司(地点)收件。
  为了能更有效率的服务客户并节省收件时间,该公司设立了收件服务登记网站,客户如有邮件需要寄送,必须在需要收件的前一天就先上网登记。为了节省油量,「飞奔」就利用晚上先行安排三位司机隔天的收件路线。每位司机至各地点收件的顺序应与各公司上网登记的顺序相符且必须能在最省油的情况下完成当天所有的收件服务。因此每位司机有可能需要在不同时间重复到同一地点收件,或不同的司机有可能需在不同的时间点前往同一地点收件。
  如下面范例二(收件公司地点依序为: 4 2 4 1 5 4 3 2 1)所示,虽然司机1一开始就已经在收件地点「1」了,但是他却不能先把后面第四个登记的公司(地点「1」)邮件先收了再前往第一、第二、或第三个登记收件地点(地点「4」,「2」,「4」)收件。但是如果前三个登记收件的服务是由司机2或3來负责,则司机1就可以在地点「1」收了第四个登记的邮件后再前往后面所登记的地点收件。此外,在某些情况下,不一定每辆车都要收到货,也就是說,最佳收件方式也有可能是只需出动一或兩辆车去收货。请写一个程序來帮「飞奔」公司计算每天依预约顺序至各收件地点收件的最少总耗油量。
输入格式 Input Format
  输入文件第一行有一个整数 m(3 ≤ m ≤ 200),代表「飞奔」公司收件的地点数,以1至m之间的整数代号來表示每个地点。
  接下來的m行(第2到第m+1行),每行有m个整数,代表一个矩阵D。第 i +行的第 j 个整数是D(i, j),D(i, j) 代表司机开车从收件点 i 到收件点 j 所需耗油量。最后有一行数串,数串之数字依序为前一天上网登记要求收件的公司地点代号,最多会有1000个收件请求。输入文件中任兩个相邻的整数都以一个空白隔开。
注意:油量矩阵D满足三角不等式,也就是說 D(i, j) ≤ D(i, k) + D(k, j),1 ≤ i, j, k ≤ m。因此,每辆车前往下一个收件地点时一定是直接前往,不必先绕道至其它地点再抵达下个收件地点。
输出格式 Output Format
输出一个整数,代表收件所需最少总耗油量

#include<cstdio>
#include<iostream>
#include<algorithm>
#include<string>
#include<vector>
#include<cstring>
#define pb push_back
using namespace std;
typedef long long ll;
const int maxn=200+10,
maxp=1000+10;
const int inf=~0U>>1;
int Dist[maxn][maxn];
int Dp[2][maxn][maxn];
int A[maxp],n,p=0;
int Init(void)
{
cin>>n;
for(int i=0;i<n;i++)
for(int j=0;j<n;j++)
{
cin>>Dist[i][j]; }
A[p++]=0;
while(scanf("%d",A+p)==1)--A[p++];
}
inline void Update(int&x,int c){x=min(x,c);}
int done(void)
{
int now=0,next=1;int cost;
for(int j=0;j<n;j++) for(int k=0;k<n;k++)
Dp[next][j][k]=inf;
Dp[next][1][2]=0;
for(int i=0;i<p-1;i++)
{ swap(now,next);
for(int j=0;j<n;j++) for(int k=0;k<n;k++) Dp[next][j][k]=inf;
for(int j=0;j<n;j++) for(int k=0;k<n;k++)if((cost=Dp[now][j][k])!=inf)
{

Update(Dp[next][j][k],cost+Dist[A[i]][A[i+1]]);
Update(Dp[next][A[i]][j],cost+Dist[k][A[i+1]]);
Update(Dp[next][A[i]][k],cost+Dist[j][A[i+1]]);
}
}
int Ans=inf;
for(int j=0;j<n;j++) for(int k=0;k<n;k++)Update(Ans,Dp[next][j][k]);
cout<<Ans<<endl;
}
int main()
{
Init();
done();
return 0;
}

VijosNT Mini 2.0.5.6

foo.cpp: In function 'int Init()':
foo.cpp:25:1: warning: no return statement in function returning non-void [-Wreturn-type]
foo.cpp: In function 'int done()':
foo.cpp:47:1: warning: no return statement in function returning non-void [-Wreturn-type]

#01: Accepted (0ms, 788KB)
#02: Accepted (0ms, 788KB)
#03: Accepted (0ms, 788KB)
#04: Accepted (0ms, 788KB)
#05: Accepted (553ms, 788KB)
#06: Accepted (0ms, 788KB)
#07: Accepted (0ms, 788KB)
#08: Accepted (553ms, 788KB)
#09: Accepted (553ms, 788KB)
#10: Accepted (0ms, 788KB)

Accepted / 100 / 1659ms / 788KB
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: