UVA DP入門テーマ
674 - Coin Change
17132
47.12%
5842
89.90%
10073734
674
Coin Change
Accepted
C++
2.076
2012-05-04 11:09:02
状態遷移方程式:dp[j]=dp[j]+dp[j-cost[i];
#include<iostream>
#include<cstdio>
#include<cstring>
#define MAXN 7500
using namespace std;
int cost[6]={0,1,5,10,25,50};
int dp[MAXN];
int main()
{
//freopen("input.txt","r",stdin);
int n,i,j;
while(scanf("%d",&n)!=EOF){
for(i=0; i<=n; ++i)
dp[i]=1;
for(i=2; i<=5; ++i){
for(j=cost[i]; j<=n; ++j)
dp[j]=dp[j]+dp[j-cost[i]];
}
printf("%d
",dp[n]);
}
return 0;
}
10131 - Is Bigger Smarter?
16114
37.77%
4752
84.85%
10074426
10131
Is Bigger Smarter?
Accepted
C++
0.016
2012-05-04 14:45:20
まず重量の大きさで並べ替えて、それからIQの最長の減算子のシーケンスを求めます
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<stack>
using namespace std;
class point{
public:
int weight;
int iq;
int number;
}elephant[1005];
bool cmp(const point& a,const point& b){
if(a.weight==b.weight)
return a.iq<b.iq;
return a.weight<b.weight;
}
class Ans{
public:
int num;
int pre;
}ans[1005];
int main()
{
//freopen("input.txt","r",stdin);
int index=0,i,j;
while(scanf("%d %d",&elephant[index].weight,&elephant[index].iq)!=EOF)
elephant[index].number=index+1, ++index;
sort(elephant,elephant+index,cmp);
for(i=0; i<index; ++i){
ans[i].num=1, ans[i].pre=i;
}
for(i=1; i<index; ++i){
int maxNum=0,pre;
for(j=0; j<i; ++j){
if(elephant[j].weight<elephant[i].weight && elephant[j].iq>elephant[i].iq && ans[j].num+1>maxNum)
maxNum=ans[j].num+1, pre=j;
}
if(maxNum>ans[i].num){
ans[i].num = maxNum;
ans[i].pre = pre;
}
}
int maxPos=0, maxNum=1;
for(i=1; i<index; ++i){
if(ans[i].num>maxNum)
maxNum=ans[i].num, maxPos=i;
}
printf("%d
",maxNum);
stack<int>st;
while(maxNum--){
st.push(elephant[maxPos].number);
maxPos = ans[maxPos].pre;
}
while(!st.empty()){
printf("%d
",st.top());
st.pop();
}
return 0;
}
562 - Dividing coins
16643
30.22%
3921
81.23%
10074473
562
Dividing coins
Accepted
C++
0.040
2012-05-04 15:03:51
01リュックサックの問題.
#include<cstdio>
#include<cstring>
int value[105],dp[25000];
int max(int a,int b){ return a>b?a:b; }
int main()
{
//freopen("input.txt","r",stdin);
int cas, m, i, j;
scanf("%d",&cas);
while(cas--){
scanf("%d",&m);
int sum=0;
for(i=0; i<m; ++i){
scanf("%d",&value[i]);
sum += value[i];
}
memset(dp,0,sizeof(dp));
for(i=0; i<m; ++i){
for(j=sum/2; j>=value[i]; --j)
dp[j] = max(dp[j-value[i]]+value[i],dp[j]);
}
printf("%d
",sum-dp[sum/2]*2);
}
return 0;
}
10066 - The Twin Towers
11414
44.45%
4465
86.58%
10074525
10066
The Twin Towers
Accepted
C++
0.012
2012-05-04 15:26:08
最長共通サブシーケンス
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
int max(int a,int b){return a>b?a:b;}
int dp[105][105];
int tow1[105],tow2[105];
int main()
{
// freopen("input.txt","r",stdin);
int cas=1;
int N1,N2,i,j;
while(scanf("%d %d",&N1,&N2)!=EOF){
if(!N1&&!N2) break;
// input
for(i=1; i<=N1; ++i)
scanf("%d",&tow1[i]);
for(i=1; i<=N2; ++i)
scanf("%d",&tow2[i]);
printf("Twin Towers #%d
",cas++);
memset(dp,0,sizeof(dp));
for(i=1; i<=N1; ++i){
for(j=1; j<=N2; ++j){
if(tow1[i]==tow2[j]) dp[i][j]=dp[i-1][j-1]+1;
else dp[i][j]=max(dp[i-1][j],dp[i][j-1]);
}
}
printf("Number of Tiles : %d
",dp[N1][N2]);
}
}
10130 - SuperSale
8092
49.68%
2865
92.11%
10074620
10130
SuperSale
Accepted
C++
0.216
グループの01リュックサックの問題を分けて、それぞれの家族の持つことができる最大の価値の物品を計算して、更に合計を求めます
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
int max(int a,int b){return a>b?a:b;}
int value[1005],cost[1005],group[105];
int dp[30000];
int main()
{
// freopen("input.txt","r",stdin);
int T,N,G,g,i,j;
scanf("%d",&T);
while(T--){
scanf("%d",&N);
for(i=1; i<=N; ++i)
scanf("%d %d",&value[i],&cost[i]);
scanf("%d",&G);
for(i=1; i<=G; ++i)
scanf("%d",&group[i]);
int maxValue=0;
for(g=1; g<=G; ++g){
memset(dp,0,sizeof(dp));
for(i=1; i<=N; ++i){
for(j=group[g]; j>=cost[i]; --j)
dp[j] = max(dp[j-cost[i]]+value[i],dp[j]);
}
maxValue += dp[group[g]];
}
printf("%d
",maxValue);
}
return 0;
}
10192 - Vacation
12967
35.15%
3953
88.24%
10074716
10192
Vacation
Accepted
C++
0.008
2012-05-04 16:06:02
最长の共通サブシーケンスは、缲り返しを判断するかと思いきや、渡した后に直接Aを発见した.
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
char str1[105],str2[105];
int dp[105][105];
int max(int a,int b){return a>b?a:b;}
int main(int i, int j)
{
// freopen("input.txt","r",stdin);
int cas=1;
while(gets(str1+1),gets(str2+1)){
if(str1[0]=='#') break;
int len1=strlen(str1+1);
int len2=strlen(str2+1);
memset(dp,0,sizeof(dp));
for(i=1; i<=len1; ++i){
for(j=1; j<=len2; ++j){
if(str1[i]==str2[j]) {
dp[i][j] = dp[i-1][j-1]+1;
}
else{
dp[i][j] = max(dp[i-1][j],dp[i][j-1]);
}
}
}
printf("Case #%d: you can visit at most %d cities.
",cas++,dp[len1][len2]);
}
return 0;
}
357 - Let Me Count The Ways
20522
27.54%
5907
66.84%
10074863
357
Let Me Count The Ways
Accepted
C++
0.708
2012-05-04 16:55:01
674と同じです.ただし、結果は1出力フォーマットとは異なり、64ビットintを用いることに注意してください.そのためWAは何度も
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
int value[6]={0,1,5,10,25,50};
long long dp[30005];
long long max(long long a,long long b){return a>b?a:b;}
int main(int i, int j)
{
// freopen("input.txt","r",stdin);
int n;
while(scanf("%d",&n)==1){
for(i=0; i<=n; ++i)
dp[i]=1;
for(i=2; i<=5; ++i){
for(j=value[i]; j<=n; ++j)
dp[j] += dp[j-value[i]];
}
if(dp[n]==1)
printf("There is only %lld way to produce %d cents change.
",dp[n],n);
else
printf("There are %lld ways to produce %d cents change.
",dp[n],n);
}
return 0;
}
10465 - Homer Simpson
8983
35.24%
2360
82.20%
10075027
10465
Homer Simpson
Accepted
C++
0.260
2012-05-04 17:49:56
この問題の意味は最初は分からなかったが、やっと分かった.Krusty-burgersはx個、Kwik-e-Martはy個、
m*x+n*y<=t、すなわちm*x+n*yをできるだけtに近づけるには、tに等しくなければ、出力とtの差はどのくらいあるか
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
int dp[10005],cost[2];
int max(int a,int b){return a>b?a:b;}
void swap(int &a,int &b){ int t=a; a=b; b=t; }
int main(int i, int j)
{
// freopen("input.txt","r",stdin);
int m,n,t;
while(scanf("%d %d %d",&m,&n,&t)!=EOF){
memset(dp,0,sizeof(dp));
if(m>n)
swap(m,n);
cost[0]=m, cost[1]=n;
for(i=0; i<2; ++i){
for(j=cost[i]; j<=t; ++j)
dp[j] = max(dp[j],dp[j-cost[i]]+cost[i]);
}
int cnt1=dp[t]/m;
int cnt2=0;
if(dp[t]==t){
while(cnt1*m+cnt2*n!=t){
--cnt1;
while(cnt1*m+cnt2*n<t)
++cnt2;
}
printf("%d
",cnt1+cnt2);
}
else{
while(cnt1*m+cnt2*n!=dp[t]){
--cnt1;
while(cnt1*m+cnt2*n<dp[t])
++cnt2;
}
printf("%d %d
",cnt1+cnt2,t-dp[t]);
}
}
return 0;
}
// 2
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
int dp[10005],num[10005],cost[2];
int max(int a,int b){return a>b?a:b;}
int main(int i, int j)
{
// freopen("input.txt","r",stdin);
int m,n,t;
while(scanf("%d %d %d",&cost[0],&cost[1],&t)!=EOF){
memset(dp,0,sizeof(dp));
memset(num,0,sizeof(num));
for(i=0; i<2; ++i){
for(j=cost[i]; j<=t; ++j){
if(dp[j-cost[i]]+cost[i]>dp[j]){
dp[j] = dp[j-cost[i]]+cost[i];
num[j] = num[j-cost[i]]+1;
}
else if(dp[j-cost[i]]+cost[i]==dp[j] && num[j-cost[i]]+1>num[j]){
num[j] = num[j-cost[i]]+1;
}
}
}
if(dp[t]==t)
printf("%d
",num[t]);
else
printf("%d %d
",num[t],t-dp[t]);
}
return 0;
}
-生命の意味は、その意味を与えることにある.
オリジナル
http://blog.csdn.net/shuangde800
, By D_Double