【BZOJ1578】【USACO2009Feb】股票市场 背包DP

题目大意

  告诉你\(n\)只股票在这\(m\)天内的价格,给你\(s\)元的初始资金,问你\(m\)天后你最多拥有多少钱。

  \(n\leq 50,m\leq 10,s\leq 200000,\)答案\(\leq 500000\)

题解

  首先我们可以认为第\(i\)天买的所有股票都会在第\(i+1\)天卖出(如果不卖出就可以看成卖出再买入)。

  那么我们可以把第\(i\)天的价格看成体积,把第\(i+1\)天的价格\(-\)\(i\)天的价格看成价值,做\(m-1\)次完全背包即可。

  时间复杂度:\(O(nms)\)

代码

#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cstdlib>
#include<ctime>
#include<utility>
#include<cmath>
#include<functional>
using namespace std;
typedef long long ll;
typedef unsigned long long ull;
typedef pair<int,int> pii;
typedef pair<ll,ll> pll;
void sort(int &a,int &b)
{
	if(a>b)
		swap(a,b);
}
void open(const char *s)
{
#ifndef ONLINE_JUDGE
	char str[100];
	sprintf(str,"%s.in",s);
	freopen(str,"r",stdin);
	sprintf(str,"%s.out",s);
	freopen(str,"w",stdout);
#endif
}
int f[500010];
int a[110][110];
int main()
{
	int n,m,u;
	scanf("%d%d%d",&n,&m,&u);
	int i,j,k;
	for(j=1;j<=n;j++)
		for(i=1;i<=m;i++)
			scanf("%d",&a[i][j]);
	for(i=2;i<=m;i++)
	{
		memset(f,0,sizeof f);
		for(j=1;j<=n;j++)
		{
			int v=a[i][j]-a[i-1][j],w=a[i-1][j];
			if(v<0)
				continue;
			for(k=w;k<=u;k++)
				f[k]=max(f[k],f[k-w]+v);
		}
		u+=*max_element(f+1,f+u+1);
	}
	printf("%d\n",u);
	return 0;
}
posted @ 2018-03-06 11:06  ywwyww  阅读(155)  评论(0编辑  收藏  举报