07 day 2

又是惨烈的一天

第一题

多重背包。二进制拆分即可。

#include <stdio.h>
#define max(a,b) ((a)>(b)?(a):(b))
int n,m,i,j,k,l,a,b;
int d[11][11];
//  s: time  w: value
int s[10000],w[10000],pn,f[1000000];
int main(){
	freopen("medic.in","r",stdin);
	freopen("medic.out","w",stdout);
	scanf("%d%d",&n,&m);
	for(i=0;i<n;++i){
		scanf("%d%d",&a,&b);
		++d[a][b];
	}
	for(i=1;i<11;++i){
		for(j=1;j<=11;++j){
			k=1;
			while(d[i][j]){
				if(d[i][j]>k){
					s[pn]=i*k,w[pn]=j*k,++pn;
					d[i][j]-=k;
					k<<=1;
				}else{
					s[pn]=i*d[i][j],w[pn]=j*d[i][j],++pn;
					d[i][j]=0;
				}
			}
		}
	}
	for(i=0;i<pn;++i){
		for(j=m;j>=s[i];--j){
			f[j]=max(f[j],f[j-s[i]]+w[i]);
		}
	}
	printf("%d
", f[m]);
    return 0;
}

第二题

先将数读入为素因子里2和5分别的个数,然后求出一路乘到底最少的2和5的个数,取其小者。

(DP)

#include <cstdio>
#define min(a,b) ((a)<(b)?(a):(b))
int m,n,i,j,k,f2[2000][2000],f5[2000][2000],fmin[2000][2000],d2[2000][2000],d5[2000][2000],t;
inline int countbit(int n){
	t=0;
	while(!(k&1)) k>>=1,++t;
	return t;
}
inline int countf(int n){
	t=0;
	while(!(k%5)) k/=5,++t;
	return t;
}
int main(){
	freopen("matrix.in","r",stdin);
	freopen("matrix.out","w",stdout);
	scanf("%d%d",&n,&m);
	for(i=0;i<n;++i){
		for(j=0;j<m;++j){
			scanf("%d",&k);
			d2[i][j]=countbit(k);
			d5[i][j]=countf(k);
		//	printf("(2: %d | 5: %d) ",d2[i][j],d5[i][j]);
			if(!i){
				f2[i][j]=f2[i][j-1]+d2[i][j];
				f5[i][j]=f5[i][j-1]+d5[i][j];
			}else if(!j){
				f2[i][j]=f2[i-1][j]+d2[i][j];
				f5[i][j]=f5[i-1][j]+d5[i][j];
			}else{
				f2[i][j]=min(f2[i-1][j],f2[i][j-1])+d2[i][j];
				f5[i][j]=min(f5[i-1][j],f5[i][j-1])+d5[i][j];
			}
		}
		printf("
");
	}
	printf("%d
",min(f2[n-1][m-1],f5[n-1][m-1]));
	return 0;
}

第三题

递推。太懒不写。

原文地址:https://www.cnblogs.com/tmzbot/p/4004205.html