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

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

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

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

Q&A

解決済

1回答

415閲覧

pythonの辞書型のappendで、copyした変数にも値が追加されてしまう

oboa

総合スコア1

Python

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

0グッド

0クリップ

投稿2022/07/28 06:45

編集2022/07/28 06:54

前提

環境:VSCode、Python3.8.3
下記のソースコードで、変数 data2 には変数 newData の初期値を持っていてほしいのですが、**append() **で newData を変更した段階で data2 も同じ値を持ってしまいます。

python

1newData = dict() 2 3newData = [ 4 { 5 "title": [ 6 { 7 "text": "aaa", 8 "type": "Noun", 9 "property": "TenseNoun" 10 } 11 ], 12 "synopsis": [ 13 { 14 "text": "bbb", 15 "type": "Noun", 16 "property": "ConcreteNoun" 17 } 18 ] 19 } 20] 21 22data2 = newData.copy() 23 24addTitle = dict() 25addTitle = { 26 "text": "test", 27 "type": "titleType", 28 "property": "propertyType" 29} 30 31newData[0]["title"].append(addTitle) 32 33print(newData) 34print(data2) 35print(id(newData)) 36print(id(data2)) 37 38#実行結果 39#[{'title': [{'text': 'aaa', 'type': 'Noun', 'property': 'TenseNoun'}, {'text': 'test', 'type': 'titleType', 'property': 'propertyType'}], 'synopsis': [{'text': 'bbb', 'type': 'Noun', 'property': 'ConcreteNoun'}]}] 40#[{'title': [{'text': 'aaa', 'type': 'Noun', 'property': 'TenseNoun'}, {'text': 'test', 'type': 'titleType', 'property': 'propertyType'}], 'synopsis': [{'text': 'bbb', 'type': 'Noun', 'property': 'ConcreteNoun'}]}] 41#2086704527040 42#2086704526976

このように copy() を用いてidの違う変数になっているにも関わらず、同じ値を持ってしまいます。

また、以下のように newData 全体に対して append() する場合は data2 の値が期待通りになってくれます

python

1newData = dict() 2 3newData = [ 4 { 5 "title": [ 6 { 7 "text": "aaa", 8 "type": "Noun", 9 "property": "TenseNoun" 10 } 11 ], 12 "synopsis": [ 13 { 14 "text": "bbb", 15 "type": "Noun", 16 "property": "ConcreteNoun" 17 } 18 ] 19 } 20] 21 22data2 = newData.copy() 23 24addTitle = dict() 25addTitle = { 26 "title": [ 27 { 28 "text": "testA", 29 "type": "Noun", 30 "property": "TenseNoun" 31 } 32 ], 33 "synopsis": [ 34 { 35 "text": "testB", 36 "type": "Noun", 37 "property": "ConcreteNoun" 38 } 39 ] 40 } 41 42newData.append(addTitle) 43 44print(newData) 45print(data2) 46print(id(newData)) 47print(id(data2)) 48 49#実行結果 50#[{'title': [{'text': 'aaa', 'type': 'Noun', 'property': 'TenseNoun'}], 'synopsis': [{'text': 'bbb', 'type': 'Noun', 'property': 'ConcreteNoun'}]}, {'title': [{'text': 'testA', 'type': 'Noun', 'property': 'TenseNoun'}], 'synopsis': [{'text': 'testB', 'type': 'Noun', 'property': 'ConcreteNoun'}]}] 51#[{'title': [{'text': 'aaa', 'type': 'Noun', 'property': 'TenseNoun'}], 'synopsis': [{'text': 'bbb', 'type': 'Noun', 'property': 'ConcreteNoun'}]}] 52#2420668184768 53#2420668184704

なぜこのように2つのソースの実行結果に差異ができてしまうのでしょうか
また、1つ目のソースの実行で newDatadata2 の値が同じにならないようにするにはどうすれば良いのでしょうか

回答よろしくお願いいたします。

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

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

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

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

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

guest

回答1

0

ベストアンサー

提示コードではnewDataのリストは浅いコピーが行われます。
つまりリストの各要素は同じもの(この場合は辞書)を指しているためそのような結果になります。
copy.deepcopyを使うと意図通りの結果になります。
参考:Pythonの浅いコピーと深いコピー: copy(), deepcopy()

Python

1import copy 2 3newData = dict() 4 5newData = [ 6 { 7 "title": [ 8 { 9 "text": "aaa", 10 "type": "Noun", 11 "property": "TenseNoun" 12 } 13 ], 14 "synopsis": [ 15 { 16 "text": "bbb", 17 "type": "Noun", 18 "property": "ConcreteNoun" 19 } 20 ] 21 } 22] 23 24data2 = newData.copy() 25 26print(id(newData[0])==id(data2[0])) # True 27 28data2 = copy.deepcopy(newData) 29print(id(newData[0])==id(data2[0])) # False

投稿2022/07/28 07:05

can110

総合スコア38262

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

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

oboa

2022/07/28 07:16

ガワが参照じゃないけど中身は参照なんですね… ありがとうございました!!
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問