質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.48%
C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

Q&A

解決済

2回答

397閲覧

AtcoderのB問題をdpで解きたいです。

oioimeister

総合スコア15

C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

0グッド

0クリップ

投稿2022/01/25 07:28

編集2022/01/25 08:16

https://atcoder.jp/contests/abc087/tasks/abc087_b
この問題をdpで解こうとしています。

C++

1#include <bits/stdc++.h> 2using namespace std; 3 4const int INF = 1<<29; 5 6int main(){ 7 int B[3]; 8 for(int i=0;i<3;i++) cin >> B[i]; 9 int X; cin >> X; 10 int dp[6][21000]; 11 int A[3] = {500 , 100 ,50}; 12 for (int i = 0; i < 6; ++i) for (int j = 0; j < 21000; ++j) dp[i][j] = INF; 13 dp[0][0] = 0; // dp[0][0] だけ 0 に 14 15 int count = 0 ; 16 17 for (int i = 0; i < 3; ++i) { 18 for (int j = 0; j <= X; ++j) { 19 if (dp[i][j] < INF) dp[i+1][j] = 0; 20 if (j >= A[i]) { 21 if (dp[i][j-A[i]] < INF) { 22 dp[i+1][j] = min(dp[i+1][j], 1); 23 } 24 if (dp[i+1][j-A[i]] < B[i]) { 25 dp[i+1][j] = min(dp[i+1][j], dp[i+1][j-A[i]] + 1); 26 } 27 } 28 if(dp[3][X] < INF){ 29 count++; 30 dp[3][X] = INF; 31 } 32 } 33 } 34 35 cout << count << endl; 36}

基本的な考え方は https://algo-method.com/tasks/313
を参考にしています。これだと答えが思い通りに出ないのですが、どなたかご教示いただけないでしょうか?
例えば、入力で
5
1
0
150
となっているとき、このようなパターンは存在しないので0のはずが、1を出力してしまいます。
他にも
30
40
50
6000
で213パターンあるのに、1しか出力しないなどです

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

y_waiwai

2022/01/25 07:54

どういう答えが出るんでしょうか そして、それがどうなればいいんでしょう。 具体的に説明しましょう
guest

回答2

0

int dp[6][21000]; の 6 と 21000 はどういう意味ですか?
dp[i][j] の i は 0~3 しか使っていないようですね。
for (int j = 0; j <= X; ++j) { ですが、X は 50 の倍数なのに、
j を 1ずつ増やすのは無駄ではないのですか?

私だったら次のようなコードを書きます。

C++

1#include <iostream> 2using namespace std; 3 4int count; 5 6void step(int num[], int i, int x) 7{ 8 static int coin[3] = { 500, 100, 50 }; 9 if (x == 0) count++; 10 else if (x > 0 && i < 3) 11 for (int j = min(num[i], x / coin[i]); j >= 0; j--) 12 step(num, i+1, x - coin[i]*j); 13} 14 15int main() 16{ 17 int num[3], x; 18 for (int i= 0; i < 3; i++) cin >> num[i]; 19 cin >> x; 20 step(num, 0, x); 21 cout << count << endl; 22}

投稿2022/01/25 14:01

kazuma-s

総合スコア8224

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

0

ベストアンサー

結論から言うと、「基本的な考え方」のリンク先の方法そのままで解くのは無理です。

リンク先の問題はパターン数が必要ないので、途中段階で複数のパターンがあった場合、minを使って最も数値の個数が少ないパターン1つのみを残す作りになっています。
そのために、いくらパターンがあっても出力は1になります。

あとは、前提条件が異なります。
リンク先の問題は各数字が1つ以上存在する前提ですが、AtCoderの問題は0の場合があります。
5,1,0,150という入力で1が出力されるのは、こちらが原因です。

投稿2022/01/25 13:32

actorbug

総合スコア2224

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問