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

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

新規登録して質問してみよう
ただいま回答率
85.48%
アルゴリズム

アルゴリズムとは、定められた目的を達成するために、プログラムの理論的な動作を定義するものです。

C++

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

Q&A

解決済

1回答

455閲覧

深さ優先探索を用いた問題の解決

eguchinisi

総合スコア4

アルゴリズム

アルゴリズムとは、定められた目的を達成するために、プログラムの理論的な動作を定義するものです。

C++

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

0グッド

0クリップ

投稿2023/01/28 12:52

前提

再帰関数を用いた全探索で、問題を解きたいと考えていますが、うまく機能しません。
陸がつながっているということを判定するには、どのようなことを実装すればいいでしょうか。

(問題)
・島国の地図が 10 行にわたって与えられる。
・各行は 10 文字からなり、o は陸地を、x は海を表す。
・少なくとも 1 マスは陸地があることが保証される。
・少なくとも 1 マスは海があることが保証される。
・海を 1 マスだけ陸地にすることで全体を 1 つの島にできるなら YES 、できないなら NO を出力せよ。出力の末尾には改行をつけること。ただし、元から 1 つの島だった場合も YES を出力せよ。

実現したいこと

・まず何もしていない一番初めの状態の陸の数を数える
・海の一つを陸に変えて、全探索を用いて、陸から陸に行けたらsumを+1する
・sumと最初に数えた陸の数が同じなら、yesを出力する

発生している問題・エラーメッセージ

・陸がつながっているかいないかに関わらず、全ての場合においてyesを出力してしまう

該当のソースコード

#include <iostream> #include <vector> #include <math.h> #include <algorithm> using namespace std; const int h = 10; const int w = 10; vector<vector<char>> v(h,vector<char>(w)); vector<vector<bool>> judge(h,vector<bool>(w)); bool ans = false; int sum = 1; int counter = 0; void field_search(int x, int y) { //陸を探索する //探索範囲を限定する if(x < 0 || y < 0 || x > h - 1 || y > w - 1 || v[x][y] == 'x') return; if(v[x][y]) //一度来たことあるか判定する if(judge[x][y] == true) return; //来た場所にフラグを立てる judge[x][y] = true; //探索する field_search(x, y + 1); field_search(x, y - 1); field_search(x + 1, y); field_search(x - 1, y); } void sea_search(int x, int y) { //海の一つを陸に変える if(v[x][y] == 'x') { v[x][y] = 'o'; //陸を探索していって、全てを通ることができたら成功 field_search(x, y); } for(int i = 0; i < h; i++) { for(int j = 0; j < w; j++) { if(judge[i][j] == true) sum++; } } if(sum == counter) ans = true; else { v[x][y] = 'x'; sum = 0; } } int main() { //入力 for(int i = 0; i < h; i++) { for(int j = 0; j < w; j++) { cin >> v[i][j]; } } //陸の数を数える for(int i = 0; i < h; i++) { for(int j = 0; j < w; j++) { if(v[i][j] == 'o') counter++; } } //すべての海からスタートする for(int i = 0; i < h; i++) { for(int j = 0; j < w; j++) { if(v[i][j] == 'x') sea_search(i,j); if(ans) break; } if(ans) break; } if(ans) cout << "YES" << endl; else cout << "NO" << endl; }

試したこと

ここに問題に対して試したことを記載してください。

補足情報(FW/ツールのバージョンなど)

ここにより詳細な情報を記載してください。

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

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

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

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

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

actorbug

2023/01/28 20:59

問題のURLを載せてください。 現状、入力例・出力例が無いので、動作確認が難しいです。
Zuishin

2023/01/28 23:51

マスの数が少ないので、力技でできます。 まず陸のマスを数えます。 次に海のマスを一つ選び、そこから地続きの陸を深さ優先探索し、地続きのマスが予め数えた陸のマスと同じ数になった時、そこを埋め立てれば一つの島になります。
dameo

2023/01/28 23:52

そもそも海にしたのに陸かどうかを判定しに行ったり、判定を何度も使ってるのに誰も元に戻そうとしてなかったり、いろいろおかしいけど、そもそもatcoderの問題を題材にした質問は個人的にあまり推奨したくないですね。バグ探しとか回答依頼というのはちょっと。。。
fana

2023/01/30 04:44

(1)ラベリング処理を実施する→陸の個数Nがわかる. (2)隣接する陸マスのラベル種類がN種類になる海マスがあるかどうか でやれるように思う.
guest

回答1

0

ベストアンサー

問題点が2つあります。

1つ目は、sum == counterで判定していることです。
海の一つを陸に変えているのだから、sumcounter(最初に数えた陸の数)より1つ多いはずです。

2つ目は、海を陸に変えるたびにjudgeをリセットしていないことです。
入力例2で、左上を海から陸にした後、judgeは以下の状態になります(Fがfalse、Tがtrue)

TFFFFFFFFF FFFFFFFFFF FFFFFFFFFF FFFFFFFFFF FFFFFFFFFF FFFFFFFFFF FFFFFFFFFF FFFFFFFFFF FFFFFFFFFF FFFFFFFFFF

次に、その右隣りの海を陸にするわけですが、jugdeは前回修正した内容がそのまま残るため、以下のようになります。

TTFFFFFFFF FTTTTTTTFF FFTTTTTFFF FFFTTTFFFF FFFFFFFFFF FFFFFFFFFF FFFFFFFFFF FFFFFFFFFF FFFFFFFFFF FFFFFFFFFF

これを繰り返すと、左上から1つずつTが増えていくことになり、いずれ最初に数えた陸の数と等しくなってYESを出力してしまうことになります。

投稿2023/01/29 01:14

actorbug

総合スコア2224

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

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

eguchinisi

2023/01/29 02:45

ありがとうございます! もう一回見直してみたいと思います!
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問