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