解凍、拡張解凍、ネストされた拡張解凍


105

次の式について考えてみます。一部の表現は「コンテキスト」を表すために繰り返されていることに注意してください。

(これは長いリストです)

a, b = 1, 2                          # simple sequence assignment
a, b = ['green', 'blue']             # list asqignment
a, b = 'XY'                          # string assignment
a, b = range(1,5,2)                  # any iterable will do


                                     # nested sequence assignment

(a,b), c = "XY", "Z"                 # a = 'X', b = 'Y', c = 'Z' 

(a,b), c = "XYZ"                     # ERROR -- too many values to unpack
(a,b), c = "XY"                      # ERROR -- need more than 1 value to unpack

(a,b), c, = [1,2],'this'             # a = '1', b = '2', c = 'this'
(a,b), (c,) = [1,2],'this'           # ERROR -- too many values to unpack


                                     # extended sequence unpacking

a, *b = 1,2,3,4,5                    # a = 1, b = [2,3,4,5]
*a, b = 1,2,3,4,5                    # a = [1,2,3,4], b = 5
a, *b, c = 1,2,3,4,5                 # a = 1, b = [2,3,4], c = 5

a, *b = 'X'                          # a = 'X', b = []
*a, b = 'X'                          # a = [], b = 'X'
a, *b, c = "XY"                      # a = 'X', b = [], c = 'Y'
a, *b, c = "X...Y"                   # a = 'X', b = ['.','.','.'], c = 'Y'

a, b, *c = 1,2,3                     # a = 1, b = 2, c = [3]
a, b, c, *d = 1,2,3                  # a = 1, b = 2, c = 3, d = []

a, *b, c, *d = 1,2,3,4,5             # ERROR -- two starred expressions in assignment

(a,b), c = [1,2],'this'              # a = '1', b = '2', c = 'this'
(a,b), *c = [1,2],'this'             # a = '1', b = '2', c = ['this']

(a,b), c, *d = [1,2],'this'          # a = '1', b = '2', c = 'this', d = []
(a,b), *c, d = [1,2],'this'          # a = '1', b = '2', c = [], d = 'this'

(a,b), (c, *d) = [1,2],'this'        # a = '1', b = '2', c = 't', d = ['h', 'i', 's']

*a = 1                               # ERROR -- target must be in a list or tuple
*a = (1,2)                           # ERROR -- target must be in a list or tuple
*a, = (1,2)                          # a = [1,2]
*a, = 1                              # ERROR -- 'int' object is not iterable
*a, = [1]                            # a = [1]
*a = [1]                             # ERROR -- target must be in a list or tuple
*a, = (1,)                           # a = [1]
*a, = (1)                            # ERROR -- 'int' object is not iterable

*a, b = [1]                          # a = [], b = 1
*a, b = (1,)                         # a = [], b = 1

(a,b),c = 1,2,3                      # ERROR -- too many values to unpack
(a,b), *c = 1,2,3                    # ERROR - 'int' object is not iterable
(a,b), *c = 'XY', 2, 3               # a = 'X', b = 'Y', c = [2,3]


                                     # extended sequence unpacking -- NESTED

(a,b),c = 1,2,3                      # ERROR -- too many values to unpack
*(a,b), c = 1,2,3                    # a = 1, b = 2, c = 3

*(a,b) = 1,2                         # ERROR -- target must be in a list or tuple
*(a,b), = 1,2                        # a = 1, b = 2

*(a,b) = 'XY'                        # ERROR -- target must be in a list or tuple
*(a,b), = 'XY'                       # a = 'X', b = 'Y'

*(a, b) = 'this'                     # ERROR -- target must be in a list or tuple
*(a, b), = 'this'                    # ERROR -- too many values to unpack
*(a, *b), = 'this'                   # a = 't', b = ['h', 'i', 's']

*(a, *b), c = 'this'                 # a = 't', b = ['h', 'i'], c = 's'

*(a,*b), = 1,2,3,3,4,5,6,7           # a = 1, b = [2, 3, 3, 4, 5, 6, 7]

*(a,*b), *c = 1,2,3,3,4,5,6,7        # ERROR -- two starred expressions in assignment
*(a,*b), (*c,) = 1,2,3,3,4,5,6,7     # ERROR -- 'int' object is not iterable
*(a,*b), c = 1,2,3,3,4,5,6,7         # a = 1, b = [2, 3, 3, 4, 5, 6], c = 7
*(a,*b), (*c,) = 1,2,3,4,5,'XY'      # a = 1, b = [2, 3, 4, 5], c = ['X', 'Y']

*(a,*b), c, d = 1,2,3,3,4,5,6,7      # a = 1, b = [2, 3, 3, 4, 5], c = 6, d = 7
*(a,*b), (c, d) = 1,2,3,3,4,5,6,7    # ERROR -- 'int' object is not iterable
*(a,*b), (*c, d) = 1,2,3,3,4,5,6,7   # ERROR -- 'int' object is not iterable
*(a,*b), *(c, d) = 1,2,3,3,4,5,6,7   # ERROR -- two starred expressions in assignment


*(a,b), c = 'XY', 3                  # ERROR -- need more than 1 value to unpack
*(*a,b), c = 'XY', 3                 # a = [], b = 'XY', c = 3
(a,b), c = 'XY', 3                   # a = 'X', b = 'Y', c = 3

*(a,b), c = 'XY', 3, 4               # a = 'XY', b = 3, c = 4
*(*a,b), c = 'XY', 3, 4              # a = ['XY'], b = 3, c = 4
(a,b), c = 'XY', 3, 4                # ERROR -- too many values to unpack

そのような表現の結果を手動で正しく推測する方法は?


28
正直なところ、これらのほとんどは、日常のコードで見られるものよりもはるかに複雑です。リスト/タプルのアンパックの基本を学ぶと、大丈夫です。
Rafe Kettler

2
これらは再帰的であることに注意してください。したがって、最初の数個をアンダーサンドした場合、すべてを処理できます。交換してみてください* xは、例えば、*(*、b)は、アンパックをxは何を把握して、xの、などで(*、b)はバックプラグ
Peteris

4
@greengit私は自分自身がPythonの高度な知識を持っていると考えており、一般的なルールを知っているだけです:)すべてのコーナーケースを知る必要はありません。インタプリタを起動して何かをテストする必要があるだけです。
Rafe Kettler

素晴らしいリストです。a, *b = 1, 2, 3開梱の種類は本当に知りませんでした。しかし、これはPy3kでしょ?
Niklas R

回答:


113

この投稿の長さについてお詫びしますが、完全性を選択することにしました。

いくつかの基本的なルールを理解したら、それらを一般化することは難しくありません。いくつかの例で説明するために最善を尽くします。これらを「手動」で評価することについて話しているので、簡単な置換ルールをいくつか提案します。基本的に、すべてのイテラブルが同じ方法でフォーマットされていると、式を理解しやすくなります。

展開のみを目的として、次の置換はの右側で有効です=(つまり、右辺値の場合)。

'XY' -> ('X', 'Y')
['X', 'Y'] -> ('X', 'Y')

値がアンパックされないことがわかった場合は、置換を元に戻します。(詳細については、以下を参照してください。)

また、「裸の」コンマが表示された場合は、トップレベルのタプルがあると想定します。これを左側と右側の両方で行います(つまり、左辺値右辺値の場合)。

'X', 'Y' -> ('X', 'Y')
a, b -> (a, b)

これらの単純なルールを念頭に置いて、いくつかの例を示します。

(a,b), c = "XY", "Z"                 # a = 'X', b = 'Y', c = 'Z'

上記のルールを適用して、に変換"XY"('X', 'Y')、裸のコンマを括弧で囲みます。

((a, b), c) = (('X', 'Y'), 'Z')

ここでの視覚的な対応は、割り当てがどのように機能するかをかなり明白にします。

ここに誤った例があります:

(a,b), c = "XYZ"

上記の置換規則に従って、以下を取得します。

((a, b), c) = ('X', 'Y', 'Z')

これは明らかに誤りです。ネストされた構造が一致しません。次に、少し複雑な例でどのように機能するかを見てみましょう。

(a,b), c, = [1,2],'this'             # a = '1', b = '2', c = 'this'

上記のルールを適用すると、

((a, b), c) = ((1, 2), ('t', 'h', 'i', 's'))

しかし、今では'this'アンパックされないが直接に割り当てられる構造から明らかcです。したがって、置換を元に戻します。

((a, b), c) = ((1, 2), 'this')

cタプルでラップするとどうなるか見てみましょう:

(a,b), (c,) = [1,2],'this'           # ERROR -- too many values to unpack

なる

((a, b), (c,)) = ((1, 2), ('t', 'h', 'i', 's'))

繰り返しますが、エラーは明らかです。cはもはやネイキッド変数ではなく、シーケンス内の変数なので、右側の対応するシーケンスはにアンパックされ(c,)ます。ただし、シーケンスの長さが異なるため、エラーが発生します。

次に、*演算子を使用して拡張アンパックします。これはもう少し複雑ですが、それでもかなり簡単です。が前に付いた変数*はリストになり、リストには、変数名に割り当てられていない、対応するシーケンスの項目が含まれます。かなり単純な例から始めます。

a, *b, c = "X...Y"                   # a = 'X', b = ['.','.','.'], c = 'Y'

これは

(a, *b, c) = ('X', '.', '.', '.', 'Y')

これを分析する最も簡単な方法は、端から作業することです。'X'に割り当てられa'Y'に割り当てられcます。シーケンスの残りの値はリストに入れられ、に割り当てられbます。

以下のような左辺値(*a, b)とは、(a, *b)上記の単なる特殊なケースです。*あいまいであるため、1つの左辺値シーケンス内に2つの演算子を含めることはできません。値はこのようなもののどこに行くのでしょうか(a, *b, *c, d)- bまたはc?ネストされたケースをすぐに検討します。

*a = 1                               # ERROR -- target must be in a list or tuple

ここでエラーはかなり自明です。ターゲット(*a)はタプル内になければなりません。

*a, = (1,2)                          # a = [1,2]

これは、裸のカンマがあるため機能します。ルールを適用しています...

(*a,) = (1, 2)

以外の変数がないので*a*a右辺値シーケンス内のすべての値をアップslurps。を(1, 2)単一の値に置き換えるとどうなりますか?

*a, = 1                              # ERROR -- 'int' object is not iterable

なる

(*a,) = 1

繰り返しますが、ここでのエラーは自明です。シーケンスではないものをアンパックすることはできず、アンパックする*aために何かが必要です。だから私たちはそれをシーケンスに入れます

*a, = [1]                            # a = [1]

これは

(*a,) = (1,)

最後に、これは一般的な混乱のポイントです。これは(1)同じ1です-タプルと算術ステートメントを区別するにはカンマが必要です。

*a, = (1)                            # ERROR -- 'int' object is not 

入れ子になりました。実際、この例は「ネストされた」セクションにはありませんでした。おそらく、それがネストされていることに気付かなかったのですか?

(a,b), *c = 'XY', 2, 3               # a = 'X', b = 'Y', c = [2,3]

なる

((a, b), *c) = (('X', 'Y'), 2, 3)

トップレベルのタプルの最初の値が割り当てられ、トップレベルのタプル(2および3)の残りの値が割り当てられますc-期待どおりです。

(a,b),c = 1,2,3                      # ERROR -- too many values to unpack
*(a,b), c = 1,2,3                    # a = 1, b = 2, c = 3

最初の行でエラーがスローされる理由についてはすでに説明しました。2行目はばかげていますが、これが機能する理由は次のとおりです。

(*(a, b), c) = (1, 2, 3)

前述のように、私たちは端から働きます。3がに割り当てられc、残りの値はその*前にある変数に割り当てられます(a, b)。この場合はです。これ(a, b) = (1, 2)は、要素の数が正しいため、たまたま動作すると同等です。これが実際のコードに表示される理由は考えられません。同様に、

*(a, *b), c = 'this'                 # a = 't', b = ['h', 'i'], c = 's'

なる

(*(a, *b), c) = ('t', 'h', 'i', 's')

両端から作業し、に's'割り当てられc、に('t', 'h', 'i')割り当てられ(a, *b)ます。端から再び作業を、't'に割り当てられa、そして('h', 'i')リストとしてBに割り当てられています。これは、実際のコードでは決して表示されるべきではないもう1つのばかげた例です。


24
OPは長いリストの例を示したので、説明の長いリストを与えるのはあくまでも適切です。
ジョンY

7

Python 2タプルの解凍はかなり簡単です。左側の各名前は、シーケンス全体または右側のシーケンス内の単一のアイテムに対応しています。名前が任意のシーケンスの単一の項目に対応する場合、すべての項目をカバーするのに十分な名前が必要です。

ただし、拡張アンパックは非常に強力であるため、確かに混乱を招く可能性があります。現実には、最後の10以上の有効な例を決して実行してはなりません。データがその構造化されているdict場合、リストのような非構造化形式ではなく、またはクラスインスタンスにある必要があります。

明らかに、新しい構文は悪用される可能性があります。あなたの質問への答えは、あなたはそのような表現を読む必要はないということです-それらは悪い習慣であり、私はそれらが使われることを疑います。

任意に複雑な式を記述できるからといって、そうするべきではありません。あなたは次のようなコードを書くことができmap(map, iterable_of_transformations, map(map, iterable_of_transformations, iterable_of_iterables_of_iterables))ますません


注:いくつかのレベルがより複雑であることを除いて、私はそのようなコードを書きました。これは演習のみを目的としており、3か月後には私には意味がなく、他の人には理解できないことを十分に理解して行われました。私の記憶が正しければ、ポリゴンテストのポイントを実装し、いくつかの座標変換を行い、SVG、HTML、JavaScriptを作成しました。
agf 2011

3

私はあなたのコードがそれを表現するために他の形式を使用することを誤解させるかもしれないと思う。

これは、演算子の優先順位に関する質問を回避するために、式で余分な括弧を使用するようなものです。私はあなたのコードを読みやすくするために常に良い投資をするわけではありません。

私はスワップのような単純なタスクにのみ解凍を使用することを好みます。

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.