程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> C語言 >> C++ >> C++入門知識 >> POJ 1742 Coins(背包問題)

POJ 1742 Coins(背包問題)

編輯:C++入門知識

題意:給你一定數量和面值的紙幣,在不超過總錢數m的情況下,用所給的紙幣共能形成多少種錢數
思路:本來想到多重背包上面了,用多重背包寫了一下,tle,後來用二進制優化,還是tle,後來看dis,說用優先隊列,網上搜資料,沒看懂,,後來看acmj1991的博客,說是用數組標記法,然後不懂神馬是數組標記法。後來才知道,其實就是個常規的方法,不過很強大。具體來說,對於每件物品,循環錢數從自身價值一直到總價值,每次循環時判斷,若該值沒出現過,則ans加1.用flag[i]標記i這個值是否出現過,num[i]表示當前物品的價值,sum表示當前物品的數量,則if(flag[i - num[i]] && cnt[i - num[i]] < sum && !flag[i])成立時,ans加1。其中cnt[i]表示當前物品所形成的價值為i時所用的當前物品的數量。
代碼:
[cpp]
#include <iostream> 
#include <cstdio> 
#include <string.h> 
using namespace std; 
 
#define CLR(arr,val) memset(arr,val,sizeof(arr)) 
const int N = 100010; 
int num[110]; 
bool flag[N]; 
int cnt[N]; 
int main(){ 
    //freopen("1.txt","r",stdin); 
    int n,totalvalue; 
    while(scanf("%d%d",&n,&totalvalue)&&n&&totalvalue){ 
      for(int i = 0;i < n;++i) 
          scanf("%d",&num[i]); 
      CLR(flag,false); 
      flag[0] = true; 
      int sum = 0,ans = 0; 
      for(int i = 0;i < n;++i){ 
         scanf("%d",&sum); 
         CLR(cnt,0); 
         for(int j = num[i];j <= totalvalue;++j){ 
             if(flag[j-num[i]] && cnt[j-num[i]] < sum && !flag[j]){ 
                flag[j] = true; 
                ans++; 
                cnt[j] = cnt[j-num[i]] + 1; 
             } 
         } 
      } 
      printf("%d\n",ans); 
    } 
    return 0; 

作者:wmn_wmn

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved