Grid Speed
时间: 10ms 内存:128M
描述:
Consider a grid in which north-south streets, separated by gaps of 10 miles each, are elevated above east-west streets laid out in a similar fashion (see illustration for the case of a 6 by 6 grid). All streets are two-way. Entrance and exit ramps connect the streets at every intersection. Because there are no traffic lights, switching from a north-south street to an east-west street, and vice versa, takes essentially no time. The grid has very little traffic, but the local police patrol so carefully for speeding that there are virtually no speeders.
The speed limits follow an unusual pattern. The speed limits are separately posted for each street and are the same for the entire street in both directions. In the illustration above, let us label the intersections using their column and row numbers: the southwestern corner of the grid is (1, 1), the southeastern corner is (6, 1), and so on. Part of your task is to determine the shortest time in which we can get from (1, 1) to (6, 3) while obeying speed limits.
However, after the Kyoto disagreement, just being fast is not good enough, one also has to be fuel efficient. Fuel consumption of a car is given in miles-per-gallon (mpg) and depends on speed of the car. Speed of a car is given in miles-per-hour (mph) and, in this digital age, the speed of a car is always a positive integer multiple of 5. The formula relating mpg to mph is a very simple one: a car travelling at v mph makes 80-0.03*v2 mpg. In a given grid of streets we would like to travel from intersection (xs, ys) to intersection (xt, yt). You are to determine the fastest and the most fuel efficient way of making the trip such that:
- the car does not change speed between intersections,
- the car obeys all speed limits,
- the car travels the shortest possible distance between the start and finish, and
- the car arrives at the destination in the given time interval.
输入:
The first line of input contains an integer t, the number of scenarios to be processed. The data for each scenario occupy 5 lines. The first line contains an integer n<=10 which is the number of horizontal and vertical streets. The second line contains an integer which is the grid unit size in miles, smaller than 100. The third and fourth lines contain n integers each, specifying the speed limits on the horizontal and vertical streets, respectively. The largest speed limit is 50. The last line of data for a scenario contains 6 integers. The first four are xs, ys, xt, and yt. The last two integers give the shortest and the longest allowed time to travel in minutes, inclusive, both not bigger than 1000.
输出:
For each scenario, output two or three lines in the format given in the sample output. If the travel is possible then, on the second line of output, report the earliest possible arrival time (but within the imposed limits) and fuel consumed (least possible for this travel time) and, on the third line, report the earliest arrival time (but within the imposed limits) that consumes the minimum amount of fuel. The time is to be reported in minutes (integer), rounded up.
示例输入:
3
8
20
10 20 30 40 50 50 50 50
50 50 50 50 50 50 40 50
2 3 7 8 300 320
8
2
10 20 20 30 10 20 10 10 
10 20 20 30 10 20 10 20 
6 8 2 4 10 39
10
10
30 20 20 10 10 20 10 10 20 20
40 20 10 20 10 20 20 10 10 20
1 1 10 10 100 500
示例输出:
Scenario 1:
The earliest  arrival: 300 minutes, fuel 6.25 gallons
The economical travel: 318 minutes, fuel 5.60 gallons
Scenario 2:
IMPOSSIBLE
Scenario 3:
The earliest  arrival: 405 minutes, fuel 4.14 gallons
The economical travel: 498 minutes, fuel 2.76 gallons
提示:
参考答案(内存最优[19684]):
#include <stdio.h>
#include <assert.h>
#include <math.h>
#include <map>
#include <limits.h>
#include <float.h>
using namespace std;
#define MAXN 10+1
int main () {
  int t;
  scanf("%d",&t);
  for(int sc = 1; sc <=t; sc++) {
    printf("Scenario %d:\n",sc);
    int n, v[MAXN], h[MAXN];
    scanf("%d",&n);
    assert(1<=n && n<=MAXN);
    int gs;
    scanf("%d\n",&gs);
    for(int i=1;i<=n;i++) { scanf("%d",&h[i]); assert(1<=h[i]&&h[i]<=50); }
    for(int i=1;i<=n;i++) { scanf("%d",&v[i]); assert(1<=v[i]&&v[i]<=50); }
    int xs,ys,xt,yt,te,tl;
    scanf("%d%d%d%d%d%d",&xs,&ys,&xt,&yt,&te,&tl);
    assert(tl <= 1000);
    int dx = 0, dy = 0;
    map<int,double> m[MAXN][MAXN];
    pair<int,double> p;
    if (tl < te) goto B;
    if (xs<xt) dx = 1; if (xs>xt) dx = -1;
    if (ys<yt) dy = 1; if (ys>yt) dy = -1;
    m[xs][ys][0] = 0.0;
    for(int i=xs;;i+=dx) {
      if (dx > 0 && i>xt) break;
      if (dx < 0 && i<xt) break;
      for(int j=ys;;j+=dy) {
	//	printf("%d %d, qsize = %d \n",i,j, m[i][j].size());
	if (dy > 0 && j>yt) break;
	if (dy < 0 && j<yt) break;
	if (i==xt && j==yt) goto A;
	if (i!=xt) {
	  map<int,double> :: iterator pos;
	  for(pos=m[i][j].begin();pos!=m[i][j].end();pos++) {
	    for(int s=5;s<=h[j];s+=5) {
	      int t = (*pos).first+3600*gs/s;
	      if ( t > 60*tl) continue;
	      if (m[i+dx][j].find(t) == m[i+dx][j].end() ||
		  m[i+dx][j][t] > (*pos).second+gs/(80-0.03*s*s))
		m[i+dx][j][t] = (*pos).second+gs/(80-0.03*s*s);
	    }
	  }
	}
	if (j!=yt) {
	  map<int,double> :: iterator pos;
	  for(pos=m[i][j].begin();pos!=m[i][j].end();pos++) {
	    for(int s=5;s<=v[i];s+=5) {
	      int t = (*pos).first+3600*gs/s;
	      if (t > 60*tl) continue;
	      if (m[i][j+dy].find(t) == m[i][j+dy].end() ||
		  m[i][j+dy][t] > (*pos).second+gs/(80-0.03*s*s))
		m[i][j+dy][t] = (*pos).second+gs/(80-0.03*s*s);
	    }
	  }
	}
	if (!dy) break;
      }
      if (!dx) break;
    }
  A: if (m[xt][yt].empty()) goto B;
  else {
    p = *(m[xt][yt].begin());
    assert(p.first <= tl*60);
    double bt=INT_MAX,btf, bf=DBL_MAX, bft;
    map<int,double>:: iterator pos;
    for(pos=m[xt][yt].begin();pos!=m[xt][yt].end();pos++) {
      p = *pos;
      assert(p.first <= tl*60);
      if (p.first < 60*te || p.first > 60*tl) continue;
      if (p.second < bf) { bf = p.second; bft = p.first; }
      else if (p.second == bf && p.first < bft) bft = p.first;
      if (p.first < bt) { bt = p.first; btf = p.second; }
      else if (p.first == bt && p.second < btf) btf = p.second;
    }
    if (bt == -1) goto B;
    printf("The earliest  arrival: %.0f minutes, fuel %.2f gallons\n",
	   ceil(bt/60), btf);
    printf("The economical travel: %.0f minutes, fuel %.2f gallons\n",
	   ceil(bft/60), bf);
    continue;
  }
  B: printf("IMPOSSIBLE\n"); 
  }
}
参考答案(时间最优[7244]):
#include <stdio.h>
#include <assert.h>
#include <math.h>
#include <map>
#include <limits.h>
#include <float.h>
using namespace std;
#define MAXN 10+1
int main () {
  int t;
  scanf("%d",&t);
  for(int sc = 1; sc <=t; sc++) {
    printf("Scenario %d:\n",sc);
    int n, v[MAXN], h[MAXN];
    scanf("%d",&n);
    assert(1<=n && n<=MAXN);
    int gs;
    scanf("%d\n",&gs);
    for(int i=1;i<=n;i++) { scanf("%d",&h[i]); assert(1<=h[i]&&h[i]<=50); }
    for(int i=1;i<=n;i++) { scanf("%d",&v[i]); assert(1<=v[i]&&v[i]<=50); }
    int xs,ys,xt,yt,te,tl;
    scanf("%d%d%d%d%d%d",&xs,&ys,&xt,&yt,&te,&tl);
    assert(tl <= 1000);
    int dx = 0, dy = 0;
    map<int,double> m[MAXN][MAXN];
    pair<int,double> p;
    if (tl < te) goto B;
    if (xs<xt) dx = 1; if (xs>xt) dx = -1;
    if (ys<yt) dy = 1; if (ys>yt) dy = -1;
    m[xs][ys][0] = 0.0;
    for(int i=xs;;i+=dx) {
      if (dx > 0 && i>xt) break;
      if (dx < 0 && i<xt) break;
      for(int j=ys;;j+=dy) {
	//	printf("%d %d, qsize = %d \n",i,j, m[i][j].size());
	if (dy > 0 && j>yt) break;
	if (dy < 0 && j<yt) break;
	if (i==xt && j==yt) goto A;
	if (i!=xt) {
	  map<int,double> :: iterator pos;
	  for(pos=m[i][j].begin();pos!=m[i][j].end();pos++) {
	    for(int s=5;s<=h[j];s+=5) {
	      int t = (*pos).first+3600*gs/s;
	      if ( t > 60*tl) continue;
	      if (m[i+dx][j].find(t) == m[i+dx][j].end() ||
		  m[i+dx][j][t] > (*pos).second+gs/(80-0.03*s*s))
		m[i+dx][j][t] = (*pos).second+gs/(80-0.03*s*s);
	    }
	  }
	}
	if (j!=yt) {
	  map<int,double> :: iterator pos;
	  for(pos=m[i][j].begin();pos!=m[i][j].end();pos++) {
	    for(int s=5;s<=v[i];s+=5) {
	      int t = (*pos).first+3600*gs/s;
	      if (t > 60*tl) continue;
	      if (m[i][j+dy].find(t) == m[i][j+dy].end() ||
		  m[i][j+dy][t] > (*pos).second+gs/(80-0.03*s*s))
		m[i][j+dy][t] = (*pos).second+gs/(80-0.03*s*s);
	    }
	  }
	}
	if (!dy) break;
      }
      if (!dx) break;
    }
  A: if (m[xt][yt].empty()) goto B;
  else {
    p = *(m[xt][yt].begin());
    assert(p.first <= tl*60);
    double bt=INT_MAX,btf, bf=DBL_MAX, bft;
    map<int,double>:: iterator pos;
    for(pos=m[xt][yt].begin();pos!=m[xt][yt].end();pos++) {
      p = *pos;
      assert(p.first <= tl*60);
      if (p.first < 60*te || p.first > 60*tl) continue;
      if (p.second < bf) { bf = p.second; bft = p.first; }
      else if (p.second == bf && p.first < bft) bft = p.first;
      if (p.first < bt) { bt = p.first; btf = p.second; }
      else if (p.first == bt && p.second < btf) btf = p.second;
    }
    if (bt == -1) goto B;
    printf("The earliest  arrival: %.0f minutes, fuel %.2f gallons\n",
	   ceil(bt/60), btf);
    printf("The economical travel: %.0f minutes, fuel %.2f gallons\n",
	   ceil(bft/60), bf);
    continue;
  }
  B: printf("IMPOSSIBLE\n"); 
  }
}
题目和答案均来自于互联网,仅供参考,如有问题请联系管理员修改或删除。
