题目:
有一个体积为N的箱子和两种数量无限的宝物。宝物1的体积为S1,价值为V1;宝物2的体积为S2,价值为V2.输入均为32位带符号整数。计算最多能装多大价值的宝物,每种宝物都必须拿非负整数个。
思路:
看完紫书的分析,不知道怎么判断N/S1、N/S2到底在那个范围内较大、较小,于是就用了下面的方法,不过这个方法效率低的很
S1个宝物2的体积=S2个宝物1的体积,他们的价值就是S1*V2和S2*V1.
1.如果S1*V2 > S2*V1,那么宝物1最多拿S2-1个,因为一旦满了S2个宝物1,这些就可以转换成S1个宝物2。然后枚举宝物1就可以了,注意这里范围是min{S2-1,N/S1}。
2.如果S1*V2 < S2*V1,那么宝物2最多拿S1-1个,因为一旦满了S1个宝物2,这些就可以转换成S2个宝物1。然后枚举宝物2就可以了,注意这里范围是min{S1-1,N/S2}。
3.如果S1*V2 = S2*V1,那么如果V1 > V2,就尽量多拿宝物1,相反就尽量多拿宝物2。
然后写代码就ok了。
代码:
#include <bits/stdc++.h> #define inf 0x3f3f3f3f #define MAX 1000000009 #define FRE() freopen("in.txt","r",stdin) #define FRO() freopen("out.txt","w",stdout) using namespace std; typedef long long ll; const int maxn = 10; ll n,s1,v1,s2,v2; int main(){ //FRE(); ios::sync_with_stdio(false); int T,kase=0; cin>>T; while(T--){ cin>>n>>s1>>v1>>s2>>v2; ll a = s1*v2, b= s2*v1; ll ans = 0; if(a>b){//对应上述情况1 for(ll i=0; i<min(s2,n/s1+1); i++){ ll num = (n-i*s1)/s2; ans = max(ans,i*v1+num*v2); } }else if(a<b){//对应上述情况2 for(ll i=0; i<min(s1,n/s2+1); i++){ ll num = (n-i*s2)/s1; ans = max(ans,i*v2+num*v1); } }else{//情况3中按各个宝物价格大小决定枚举那个宝物 if(v1>v2){ ll i=0; for(; i<(n/s1); i++){ ll num = (n-i*s1)/s2; ans = max(ans, i*v1+num*v2); } }else { ll i=0; for(; i<(n/s2); i++){ ll num = (n-i*s2)/s1; ans = max(ans, i*v2+num*v1); } } } cout<<"Case #"<<++kase<<": "<<ans<<endl; } return 0; }
AC之后在VJ中发现了一个判断较大较小的方法做的,于是我又改了改自己写的代码,发现当这个界限设在1e3~1e4的时候,搜索的效率会达到最大。
当(N/S1)较小的时候,就枚举宝物1,尽量多拿宝物2,当(N/S2)较小的时候,就枚举宝物2,尽量多拿宝物1,如果两者都较小的时候,就按上边的1、2两个情况进行分析。
代码:
#include <bits/stdc++.h> #define inf 0x3f3f3f3f #define MAX 1e3 #define FRE() freopen("in.txt","r",stdin) #define FRO() freopen("out.txt","w",stdout) using namespace std; typedef long long ll; const int maxn = 10; ll n, s1, v1, s2, v2; int main() { //FRE(); ios::sync_with_stdio(false); int T, kase = 0; cin >> T; while(T--) { cin >> n >> s1 >> v1 >> s2 >> v2; ll c = n / s1, d = n / s2; ll a = s1 * v2, b = s2 * v1; ll ans = 0; if(c < MAX) { for(ll i = 0; i < (n / s1 + 1); i++) { ll num = (n - i * s1) / s2; ans = max(ans, i * v1 + num * v2); } } else if(d < MAX) { for(ll i = 0; i < (n / s2 + 1); i++) { ll num = (n - i * s2) / s1; ans = max(ans, i * v2 + num * v1); } } else if(a > b) { for(ll i = 0; i < min(s2, n / s1 + 1); i++) { ll num = (n - i * s1) / s2; ans = max(ans, i * v1 + num * v2); } } else if(a <= b) { for(ll i = 0; i < min(s1, n / s2 + 1); i++) { ll num = (n - i * s2) / s1; ans = max(ans, i * v2 + num * v1); } } cout << "Case #" << ++kase << ": " << ans << endl; } return 0; }