Python辞書の理解


386

Pythonで(キーの)辞書内包を作成することは可能ですか?

リスト内包表記がなければ、次のようなものを使用できます。

l = []
for n in range(1, 11):
    l.append(n)

これをリスト内包表記に短縮できますl = [n for n in range(1, 11)]

しかし、辞書のキーを同じ値に設定したいとします。できます:

d = {}
for n in range(1, 11):
     d[n] = True # same value for each

私はこれを試しました:

d = {}
d[i for i in range(1, 11)] = True

しかし、私はSyntaxError上に乗るfor

さらに(この部分は必要ありませんが、疑問に思っています)、次のように、辞書のキーをさまざまな値の束に設定できます。

d = {}
for n in range(1, 11):
    d[n] = n

これは辞書の理解で可能ですか?

d = {}
d[i for i in range(1, 11)] = [x for x in range(1, 11)]

これにより、も発生SyntaxErrorforます。


3
今後の読者のために:NumPy配列を使用すると、複数の要素を単一の値または値のリストに設定できます。NumPyを使用する理由がまだない場合は、この機能だけではおそらく価値がありません。
David Z

回答:


520

Python 2.7+に辞書内包表記がありますが、それらはあなたがしようとしているようには機能しません。リスト内包のように、それらは新しい辞書を作成します。これらを使用して、既存の辞書にキーを追加することはできません。また、キーと値を指定する必要がありますが、必要に応じてダミー値を指定することもできます。

>>> d = {n: n**2 for n in range(5)}
>>> print d
{0: 0, 1: 1, 2: 4, 3: 9, 4: 16}

それらをすべてTrueに設定する場合:

>>> d = {n: True for n in range(5)}
>>> print d
{0: True, 1: True, 2: True, 3: True, 4: True}

あなたが求めているように見えるのは、既存の辞書に一度に複数のキーを設定する方法です。そのための直接のショートカットはありません。既に示したようにループするか、辞書内包を使用して新しい値で新しい辞書を作成oldDict.update(newDict)し、新しい値を古い辞書にマージすることができます。


15
FWIWはdict.updatedictコンストラクターのように反復可能なキーと値のペアを受け入れることもできます
mgilson

6
すべての値が同じであるディクショナリを作成する場合は、を使用することに注意してくださいdict.fromkeys()。したがって、すべての値をに設定するにはTrue、を使用しますdict.fromkeys(range(5), True)。値はコピーされないため、変更可能な値がある場合は、これを回避する必要があることに注意してください。すべてのキー間で共有されます。
Martijn Pieters

2
注:キーはメソッドの結果である場合もあります{ n*2 : n for n in range(3) } => {0: 0, 2: 1, 4: 2}。どちらも同じ式で実行できます{ n*2 : n*3 for n in range(3) } => { 0: 0, 2: 3, 4: 6 }
Zaaier 2017

151

あなたはdict.fromkeysクラスメソッドを使うことができます...

>>> dict.fromkeys(range(5), True)
{0: True, 1: True, 2: True, 3: True, 4: True}

これは、すべてのキーが同じ値にマップされる辞書を作成する最も速い方法です。

ただし、これを変更可能なオブジェクトで使用ないでください

d = dict.fromkeys(range(5), [])
# {0: [], 1: [], 2: [], 3: [], 4: []}
d[1].append(2)
# {0: [2], 1: [2], 2: [2], 3: [2], 4: [2]} !!!

実際にすべてのキーを初期化する必要がない場合は、a defaultdictも役立ちます。

from collections import defaultdict
d = defaultdict(True)

2番目の部分に答えるには、dict理解が必要なものです。

{k: k for k in range(10)}

おそらくこれを行うべきではありませんが、オーバーライドしdictdefaultdict場合のように機能するサブクラスを作成することもできます__missing__

>>> class KeyDict(dict):
...    def __missing__(self, key):
...       #self[key] = key  # Maybe add this also?
...       return key
... 
>>> d = KeyDict()
>>> d[1]
1
>>> d[2]
2
>>> d[3]
3
>>> print(d)
{}

の場合、d = defaultdict(lambda: True)Trueは変更可能(または変更不可)であるため、ラムダは必要ありません。
rhbvkleef

28
>>> {i:i for i in range(1, 11)}
{1: 1, 2: 2, 3: 3, 4: 4, 5: 5, 6: 6, 7: 7, 8: 8, 9: 9, 10: 10}

22

@mgilsonのコメントが本当に好きです。2つのイテラブル(キーに対応するものと値に対応するもの)がある場合は、次のことも実行できます。

keys = ['a', 'b', 'c']
values = [1, 2, 3]
d = dict(zip(keys, values))

与える

d = {'a':1、 'b':2、 'c':3}


4
口述読解の使用:{i:j for i in keys for j in values}
LoMaPh

11

タプルのリストでdict()を使用します。このソリューションでは、同じ長さである限り、各リストに任意の値を含めることができます

i_s = range(1, 11)
x_s = range(1, 11)
# x_s = range(11, 1, -1) # Also works
d = dict([(i_s[index], x_s[index], ) for index in range(len(i_s))])

12
d = dict(zip(i_s,x_s))
余談

10

辞書内包表記を使用してリスト内の単語の出現をカウントするこの例を考えてみます

my_list = ['hello', 'hi', 'hello', 'today', 'morning', 'again', 'hello']
my_dict = {k:my_list.count(k) for k in my_list}
print(my_dict)

そして結果は

{'again': 1, 'hi': 1, 'hello': 3, 'today': 1, 'morning': 1}

これは興味深いですが、「hello」のようなキーを複数回カウントするため、最も効率的ではありません
FuriousGeorge

7

リスト内包の主な目的は、元のリストを変更または破棄せずに、別のリストに基づいて新しいリストを作成することです。

書く代わりに

    l = []
    for n in range(1, 11):
        l.append(n)

または

    l = [n for n in range(1, 11)]

あなただけ書くべきです

    l = range(1, 11)

上位2つのコードブロックでは、新しいリストを作成し、それを繰り返し処理して、各要素を返します。これは、リストのコピーを作成するための高価な方法です。

すべてのキーが別の辞書に基づいて同じ値に設定された新しい辞書を取得するには、次のようにします。

    old_dict = {'a': 1, 'c': 3, 'b': 2}
    new_dict = { key:'your value here' for key in old_dict.keys()}

あなたが書くとき、SyntaxErrorを受け取ります

    d = {}
    d[i for i in range(1, 11)] = True

あなたは基本的に言っています:「私のキー 'i for i in range(1、11)'をTrueに設定してください。そして、「i for i in range(1、11)」は有効なキーではなく、単なる構文エラーです。辞書がサポートするリストをキーとして使用する場合は、次のようにします

    d[[i for i in range(1, 11)]] = True

ではなく

    d[i for i in range(1, 11)] = True

ただし、リストはハッシュ可能ではないため、リストをdictキーとして使用することはできません。


-3

そのようなリストをハッシュすることはできません。代わりにこれを試してください、それはタプルを使用します

d[tuple([i for i in range(1,11)])] = True
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.