2015年1月4日日曜日

150104(2)

Ruby


「n-クイーン」パズル

バリエーション解を全て出力するコードを、
「アルゴリズムとデータ構造」第10章と、
スタック・オーバーフローにおける回答を
もとに作成。
http://ja.stackoverflow.com/questions/3021/%E3%82%A8%E3%82%A4%E3%83%88%E3%82%AF%E3%82%A4%E3%83%BC%E3%83%B3%E3%83%91%E3%82%BA%E3%83%AB%E3%81%AE%E8%A7%A3%E6%B3%95%E3%81%AB%E3%81%A4%E3%81%84%E3%81%A6

N = 5
@count = 0
@board = Array.new(N){Array.new(N, false)}

# (x,y)にクィーンがあるかどうかチェック
def check(x, y)
  # 左方向をチェック
  p = x
  while p > 0
    return false if @board[p -= 1][y]
  end
  # 左上方向をチェック
  p, q = x, y
  while p > 0 && q > 0
    return false if @board[p -= 1][q -= 1]
  end
  # 左下方向をチェック
  p, q = x, y
  while p > 0 && q < N - 1
    return false if @board[p -= 1][q += 1]
  end
  return true
end

def showBoard
  for y in (0..N - 1)
    for x in (0..N - 1)
      printf @board[x][y]? 'Q ' : '. '
    end
    printf "\n"
  end
end

def solve(x)
  if x == N
    # すべての列にクィーンを置けたので、解を表示する。
    @count += 1
    showBoard
    puts "--" * N
  else
    # (x, 0) ... (x, N-1) にクィーンを置くことを試していく。
    for i in (0..N - 1)
      if check(x, i)
        # (x, i) にクィーンが置けるので、実際に置く。
        @board[x][i] = true
        # 次の列に進む
        solve(x + 1)
        # (x, i) に置いたクィーンを取り除く。
        @board[x][i] = false
      end
    end
  end
end

solve(0)
puts @count

0 件のコメント:

コメントを投稿

注: コメントを投稿できるのは、このブログのメンバーだけです。