ponpokosan
@ponpokosan

Are you sure you want to delete the question?

Leaving a resolved question undeleted may help others!

atcoder ABC147 C問題(ビット全探索の問題)についての質問

解決したいこと

Atcoder ABC147 C - HonestOrUnkind2
ビット演算についての質問

発生している問題

ビット表記のi番目のフラグが立っているかを調べる際、以下の表記について同じ働きをすると認識しているのですが、上で記述するとWA、下で記述するとACになります。
この表記の違いを教えてほしいです。
誤)

if(bit & 1 << x[i][j])

正)

if(bit >> x[i][j] & 1)

以下にソースコード全文載せます。(56行目辺りです)

該当するソースコード

#include <atcoder/all>
#include <bits/stdc++.h>
#include <string>

using namespace std;
using namespace atcoder;
using ll = long long;

#define rep(i, n) for (int i = 0; i < (int)(n); i++) // 単純なfor文
#define reps(i, m, n) for (int i = (int)(m); i < (int)(n); i++) // 指定for文
#define pb push_back
#define eb emplace_back

//--------------------------------------------------//

//--------------------------------------------------//

int main() {

  //---------------text temprate--------------------//
  ifstream in("in.txt");
  if (in.is_open()) {
    cin.rdbuf(in.rdbuf());
  }
  //---------------text temprate--------------------//

  //------------------------------------------------//
  //---------------koko kara kizyutu----------------//

  int n;
  cin >> n;
  vector<int> a(n);
  vector<vector<int>> x(n), y(n);

  rep(i, n) {
    cin >> a[i];
    rep(j, a[i]) {
      int u, t;
      cin >> u >> t;
      u--;
      x[i].pb(u);
      y[i].pb(t);
    }
  }

  bool hantei;
  int ans = 0;
  rep(bit, 1 << n) {
    hantei = true;
    rep(i, n) {
      if (!(bit & (1 << i)))
        continue;

      //該当箇所ここです
      rep(j, a[i]) {
        if ((bit & (1 << x[i][j])) ^ y[i][j])
          hantei = false;
      }
    }
    if (hantei)
      ans = max(ans, __builtin_popcount(bit));
  }

  cout << ans << endl;

  //---------------koko made kizyutu----------------//
  //------------------------------------------------//
}

自分で試したこと

調べたところ、前者は1をiだけシフトした後すべてのビットをチェックするのに対して、後者はシフトした後1ビット目のみを比較するため2つの式が異なる結果を返す可能性があるそうです。
この説明でなんとなくわかったような気がしたのですが、それであればすべてのビットを比較する前者を使うメリットが全く分かりません。

お聞きしたいこと

①上で記述した2つのコードの働きの違いについて
②「自分で試したこと」で書いた2つのコードの違いの仮説が正しかった場合、なぜ色々なサイトや記事で前者のコードが多く使われているのか。

最後に

初学者なので頓珍漢な事しているかもしれませんが、教えていただけると嬉しいです。T^T

0

4Answer

この表記の違いを教えてほしいです。

(問題文とかは見てないので,そういう状況が WA と AC に影響したのか?等はわかりませんが…)

例えば x[i][j] が int 型の bit 数よりも大きい場合とかで,両者の結果が異なるのではないでしょうか.

if(bit & 1 << x[i][j])
の場合, bit & 0 になって,常に if の条件を満たさないことになると思います.

if(bit >> x[i][j] & 1)
の方だと,結果は bit の符号bit次第になるのではないでしょうか.

1Like

Comments

  1. @ponpokosan

    Questioner

    ありがとうございます!自分にはない着眼点でした。。
    感謝です!!

  2. すみません.
    ちょっと試してみたら話が全然違ってました.
    bit幅以上のシフト量は未定義動作っぽいです.

Comments

  1. @ponpokosan

    Questioner

    回答ありがとうございます!!ご丁寧にURLまで、、助かります。。!!

51行目 if (!(bit & (1 << i)))

56行目 if ((bit & (1 << x[i][j])) ^ y[i][j])

どちらの if文でしょうか?

51行目はどちらの書き方でも同じと思いますが、
56行目はその後、xor するので結果が違ってくると思いますが。

1Like

Comments

  1. @ponpokosan

    Questioner

    回答感謝です!bit&1<<iがてっきり1or0で返してくるものだと勘違いしていて、XORのこと完全に考慮から外していました。。無事解決しました!ありがとうございますm(__)m

回答していただいたお三方、本当にありがとうございましたm(__)m
ビット演算で勘違いしていた部分があったらしくそれに気づけたので無事解決しました!

0Like

Your answer might help someone💌