uva 301

题目的意思是说列车从起始点开出的时候,他要收集每个站(包括起始站)的乘客的搭车订单,这些订单能够知道多少人从哪个站上车然后从哪个站下车。但由于列车载人的容量有限,它必须有所规划,即放弃有些站的订单票,在这样的情况下,求这列车能够得到的最大的利润值,单价是人从上车开始每经过一个站收权值为1的价格

#include <iostream>
#include <cstdio>
#include <cstring>
using namespace std;
const int MAXN = 300 ;

int n,no_b,numTicket,maxSum,mark[MAXN];

struct Ticket
{
	int start,end,num;
	int earn;
	int leftSum;
}arr[MAXN];

void dfs(int cur,int sum)
{
	if ( sum > maxSum )
		maxSum = sum ;

	for ( ; cur < numTicket ; cur++)
	{
		int i;
		if (sum + arr[cur].leftSum < maxSum) //如果后面的客人全接也比最大值小
			return ;
		for (i = arr[cur].start ; i < arr[cur].end ; i++ ) //题目要求我们是要么全部接收,要么不接收
		{
			mark[i] += arr[cur].num ;
			if(mark[i] > n )
				break;
		}
		if (i == arr[cur].end)
		{
			dfs(cur+1,sum+arr[cur].earn);
			i--;
		}
		for ( ; i >= arr[cur].start ; i--)   // 回溯 
			mark[i] -= arr[cur].num ;
	}
}

int main()
{
	int order;
	while (scanf("%d %d %d",&n,&no_b,&order) != EOF)
	{
		if (!n && !no_b && !order)
			break;
		numTicket = 0 ;
		int a,b,c;

		for (int i = 0 ; i < order ; i++ )
		{
			scanf("%d %d %d",&a,&b,&c);
			if ( c <= n )
			{
				arr[numTicket].start = a;
				arr[numTicket].end = b ;
				arr[numTicket].num = c ;
				arr[numTicket].earn = (b-a) * c ;
				arr[numTicket++].leftSum = (b-a) * c; //后面的减枝会用到
			}
		}
		for (int i = numTicket - 2 ; i >= 0 ; i--)
			arr[i].leftSum += arr[i+1].leftSum ;

		memset(mark,0,sizeof(mark));
		maxSum = -1234455 ;
		dfs(0,0);
		printf("%d\n",maxSum);		
	}
	return 0 ;
}


你可能感兴趣的:(uva 301)