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

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

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

Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

Q&A

解決済

3回答

321閲覧

長さMで要素は0または1であるリストを全て列挙する

hoshi1996

総合スコア53

Python

Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

1グッド

2クリップ

投稿2019/12/27 16:25

編集2019/12/27 16:29

タイトルの通り長さMが与えられたとき要素が0または1のリストを全て列挙したいです.

python

1M = 10 2padding = '0' + str(M) + 'b' 3arrays = [] 4for i in range(2 ** M): 5 binary = format(i, padding) 6 binary_array = [int(binary[j]) for j in range(len(binary))] 7 arrays.append(binary_array)

これで実際に実装はできました.
しかし,Mの値が大きくなると計算量が多く,このコードは現実的でありません.
もっと効率的な方法はありますか.

DrqYuto👍を押しています

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

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

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

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

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

hayataka2049

2019/12/28 15:22 編集

何のためにこれを行いたいのですか? 目的によってはより適切な解法が存在するかもしれません。 (どのみち2**Mの長さのlistを作るのは非効率です)
guest

回答3

0

ベストアンサー

itertools.product同様にジェネレータにすればメモリ使用量は減ります。

def bit_pattern(n): for i in range(2**n): yield [(i >> b) & 1 for b in range(n - 1, -1, -1)] M = 10 for bit in bit_pattern(M): print(bit)

処理内容によっては、バイナリのままrangeにしておいて、必要なときに2進配列化する手もあるのでは?
itertool.product で処理した方が高速ですけど。

>>> M = 10 >>> pattern = range(1<<M) >>> def to_array(b, n): ... return [(i >> b) & 1 for b in range(n - 1, -1, -1)] ... >>> print(to_array(pattern[5], M)) [0, 0, 0, 0, 0, 0, 0, 1, 0, 1]

とか

>>> M = 10 >>> for i in range(2**M): ... print(f"{i:0{M}b}") ... 0000000000 0000000001 ... 1111111110 1111111111

投稿2019/12/27 22:54

編集2019/12/28 06:32
shiracamus

総合スコア5406

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

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

0

こんにちは

速くするために

  • 文字列を経由しない。
  • ビット演算を使う。(2で割るときに/ 2 ではなく右シフトする、など)

といった工夫をすることで、以下でどうでしょう?

python3

1M = 10 2 3arrays = [[i >> j & 0b1 for j in reversed(range(M))] for i in range(2 ** M)] 4

参考になれば幸いです。

(※ビット演算を使ったコードは、shiracamusさんのご回答で既に提示されていましたので、別の方法を下記に追記しました。)

追記

1または0を要素とするリストを引数に取る、次のような関数incrementを考えます。

  • たとえば、[0, 1, 0, 1] というリスト a があったとして、この a を引数に与えられると、 a の要素の並びを2進数とみたてて、1を加えた結果のビットの並びを表す [0, 1, 1, 0] に変更し、変更後の a を返します。ただし、 [1, 1, 1, 1] が与えられたときは、リストの長さは変更されず、 [0, 0, 0, 0] にして返します。

このような increment は一例として再帰を使って以下のように書けます。(再帰を使わなくても書けるでしょう)

python3

1def increment(lis, length=None): 2 if length is None: 3 length = len(lis) 4 if length == 0: 5 return 6 7 if lis[length-1]: 8 lis[length-1] = 0 9 increment(lis, length-1) 10 else: 11 lis[length-1] = 1 12 13 return lis 14

上記の incrementを使うと、ご質問の arrays は、以下で得られます。

python3

1M = 10 2a = [1] * M 3arrays = [increment(a).copy() for _ in range(2 ** M)]

投稿2019/12/28 02:06

編集2019/12/29 08:35
jun68ykt

総合スコア9058

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

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

0

速いかどうか確認してませんが、コンパクトに書いてライブラリに任せるとすると、

Python

1import itertools 2 3M=5 4iter=itertools.product(*([[0,1]]*M)) 5print(list(iter))

リスト化せずに、イテレータのままで扱うとよいと思います。

投稿2019/12/27 16:42

otn

総合スコア84538

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

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

otn

2019/12/27 16:50 編集

Rubyなら、 [0,1].repeated_permutation(M) と、そのためのメソッド(重複ありの順列)があります。
shiracamus

2019/12/27 22:37

productならrepeat引数も便利です。 import itertools M = 10 arrays = list(itertools.product([0,1], repeat=M)
otn

2019/12/27 23:49 編集

ああ、そういう引数があったんですね。 補足ありがとうございます。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問