へっぽこプログラマーの備忘録
プログラムを中心とした個人的なメモ用のブログです。 タイトルは迷走中。
内容の保証はできませんのであしからずご了承ください。
menu
keyboard_arrow_up
Top
search
close
home
ホーム
computer
PC一般
construction
開発環境・ツール
code
プログラミング
home
ホーム
computer
PC一般
construction
開発環境・ツール
code
プログラミング
Home
›
Python
›
[Python] アドレス (id) は変更せずに辞書 (dict) をコピーしたい
2022/08/13
[Python] アドレス (id) は変更せずに辞書 (dict) をコピーしたい
update
event_note
label
Python
わかってしまえば当たり前なのですが、ちょっとハマったのでメモ。
結論から先に言うと、`clear()` `update()` を使うことで実現できました(ただし排他には要注意)。 ## 環境 - Python 3.8.10 ## 前提知識 以下の話はググれば解説されているページがいくらでも見つかりますが・・・。 `dict` はミュータブルな型なので、代入する場合には注意が必要です。 ```py dict_a = {"key_a": "value_a"} dict_b = {"key_b": "value_b"} # 値と id の確認 print(f'dict_a={dict_a}, id:{id(dict_a)}') print(f'dict_b={dict_b}, id:{id(dict_b)}') print(f'----------') dict_a = dict_b # 代入 # 値と id の確認 print(f'dict_a={dict_a}, id:{id(dict_a)}') print(f'dict_b={dict_b}, id:{id(dict_b)}') ``` 代入した場合には同じオブジェクトを指すようになるので、以下のように `id` が同じになります。 ```sh dict_a={'key_a': 'value_a'}, id:140351608696704 dict_b={'key_b': 'value_b'}, id:140351608696768 ---------- dict_a={'key_b': 'value_b'}, id:140351608696768 dict_b={'key_b': 'value_b'}, id:140351608696768 ``` なので、`dic_a` を変更したら `dict_b` の中身も変わります。 これを回避するには(つまり `id` は別で中身だけをコピーしたい場合は) `copy()` を使います。 ``` dict_a = {"key_a": "value_a"} dict_b = {"key_b": "value_b"} # 値と id の確認 print(f'dict_a={dict_a}, id:{id(dict_a)}') print(f'dict_b={dict_b}, id:{id(dict_b)}') print(f'----------') dict_a = dict_b.copy() # 値と id の確認 print(f'dict_a={dict_a}, id:{id(dict_a)}') print(f'dict_b={dict_b}, id:{id(dict_b)}') ``` この場合は新たにオブジェクトが作成されるので、`id` は別で中身は同じになります。 ``` dict_a={'key_a': 'value_a'}, id:139764478058368 dict_b={'key_b': 'value_b'}, id:139764478058432 ---------- dict_a={'key_b': 'value_b'}, id:139764476783232 dict_b={'key_b': 'value_b'}, id:139764478058432 ``` ### じゃあ id を変えずに中身だけをコピーするには? `clear()` と `update()` を使えばできます。 ```py dict_a = {"key_a": "value_a"} dict_b = {"key_b": "value_b"} # 値と id の確認 print(f'dict_a={dict_a}, id:{id(dict_a)}') print(f'dict_b={dict_b}, id:{id(dict_b)}') print(f'----------') dict_a.clear() dict_a.update(dict_b) # 値と id の確認 print(f'dict_a={dict_a}, id:{id(dict_a)}') print(f'dict_b={dict_b}, id:{id(dict_b)}') ``` ``` dict_a={'key_a': 'value_a'}, id:139790269009792 dict_b={'key_b': 'value_b'}, id:139790269009856 ---------- dict_a={'key_b': 'value_b'}, id:139790269009792 dict_b={'key_b': 'value_b'}, id:139790269009856 ``` `dict_a` `dict_b` の中身は同じですが、`dict_a` の `id` は変わっていません。 ### 関数に引数で渡す場合 Python では全て参照渡しなので、`dict` を関数に渡して関数内で変更した場合、その関数の外にも影響を与えます。 ```py def change_dict(target_dict): target_dict['hoge'] = 'piyo' dict_a = {"key_a": "value_a"} dict_b = {"key_b": "value_b"} # 値と id の確認 print(f'dict_a={dict_a}, id:{id(dict_a)}') print(f'dict_b={dict_b}, id:{id(dict_b)}') print(f'----------') change_dict(dict_a) # 値と id の確認 print(f'dict_a={dict_a}, id:{id(dict_a)}') print(f'dict_b={dict_b}, id:{id(dict_b)}') ``` ``` dict_a={'key_a': 'value_a'}, id:139844338024320 dict_b={'key_b': 'value_b'}, id:139844338024384 ---------- dict_a={'key_a': 'value_a', 'hoge': 'piyo'}, id:139844338024320 dict_b={'key_b': 'value_b'}, id:139844338024384 ``` ## 本題 じゃあ、関数内で `dict` を丸ごと差し替えたい場合はどうすればよいか? 引数で渡した値に丸ごと新しい `dict` を代入すると、その引数の参照先が変わるだけなので、参照元には反映されません。 ```py def change_dict(target_dict): target_dict = {'hoge': 'piyo'} dict_a = {"key_a": "value_a"} dict_b = {"key_b": "value_b"} # 値と id の確認 print(f'dict_a={dict_a}, id:{id(dict_a)}') print(f'dict_b={dict_b}, id:{id(dict_b)}') print(f'----------') change_dict(dict_a) # 値と id の確認 print(f'dict_a={dict_a}, id:{id(dict_a)}') print(f'dict_b={dict_b}, id:{id(dict_b)}') ``` ``` dict_a={'key_a': 'value_a'}, id:139712684732288 dict_b={'key_b': 'value_b'}, id:139712684732352 ---------- dict_a={'key_a': 'value_a'}, id:139712684732288 dict_b={'key_b': 'value_b'}, id:139712684732352 ``` かと言って、`copy()` を使うと新たにオブジェクトが作成されるので、この場合も参照元には反映されません(コピー前のオブジェクトを参照しているため)。 ```py def change_dict(target_dict): target_dict = {'hoge': 'piyo'}.copy() dict_a = {"key_a": "value_a"} dict_b = {"key_b": "value_b"} # 値と id の確認 print(f'dict_a={dict_a}, id:{id(dict_a)}') print(f'dict_b={dict_b}, id:{id(dict_b)}') print(f'----------') change_dict(dict_a) # 値と id の確認 print(f'dict_a={dict_a}, id:{id(dict_a)}') print(f'dict_b={dict_b}, id:{id(dict_b)}') ``` ``` dict_a={'key_a': 'value_a'}, id:2642357453888 dict_b={'key_b': 'value_b'}, id:2642357181760 ---------- dict_a={'key_a': 'value_a'}, id:2642357453888 dict_b={'key_b': 'value_b'}, id:2642357181760 ``` なので、参照元にも変更を反映したい場合には、前述したように `clear()` `update()` を使います。 ```py def change_dict(target_dict): target_dict.clear() target_dict.update({'hoge': 'piyo'}) dict_a = {"key_a": "value_a"} dict_b = {"key_b": "value_b"} # 値と id の確認 print(f'dict_a={dict_a}, id:{id(dict_a)}') print(f'dict_b={dict_b}, id:{id(dict_b)}') print(f'----------') change_dict(dict_a) # 値と id の確認 print(f'dict_a={dict_a}, id:{id(dict_a)}') print(f'dict_b={dict_b}, id:{id(dict_b)}') ``` ```py dict_a={'key_a': 'value_a'}, id:2371211616000 dict_b={'key_b': 'value_b'}, id:2371211612864 ---------- dict_a={'hoge': 'piyo'}, id:2371211616000 dict_b={'key_b': 'value_b'}, id:2371211612864 ``` ### 排他には注意 `clear()` `update()` という2つの関数を使って `dict` を更新するので、並列処理を行っている場合には排他が必要です。 `clear()` と `update()` の間で値が読み込まれるということが発生しうるからです。 私の場合、マルチプロセスで動いているアプリで実際にそれが発生しました。 (使っていたのは普通の `dict` ではなく `multiprocessing.Manager().dict()` ですが)
## 参考 URL
tweet
facebook
Pocket
B!
はてブ
LINE
chevron_left
chevron_right
Translate
Popular Posts
TortoiseGit でコミットメッセージを変更する
image
NO IMAGE
smbclient で session setup failed: NT_STATUS_LOGON_FAILURE が表示される
Docker for Windows の設定
image
NO IMAGE
TortoiseSVN ロック状態のチェック
image
NO IMAGE
マージ元ブランチとマージ先ブランチ
image
NO IMAGE
Visual Studio で文字がにじむ(ぼやける)
TortoiseGit でブランチ間の差分を見る
image
NO IMAGE
AsciidocFX をビルドする
image
NO IMAGE
PowerShellでブレークポイントが設定できない場合
[Python] 文字列の判定で、None と空文字を同時に判定する
Labels
.NET Core
31
.NET Framework
17
.NET Standard
2
AdminLTE
1
AI
1
Apache
3
AppVeyor
2
AsciiDoc
7
ASP.NET Core
55
Atom
4
AWS
5
AWS Cloud9
4
blockdiag
1
Blogger
13
Bootstrap
3
C/C++
6
C#
106
CentOS
3
Chrome
1
Chronograf
3
chrony
1
Codecov
1
CSS
1
Docker
82
DokuWiki
4
Doxygen
1
draw.io
1
EasyTag
1
Electron
1
Electron.NET
2
Entity Framework Core
9
Excel
2
FFmpeg
3
Firefox
6
Flask
1
Git
19
GitBook
4
GitBucket
7
GitHub
7
GitLab
39
Go
1
Google
1
Google Cloud Platform
1
Grafana
13
GStreamer
2
HTML
5
IIS
8
InfluxDB
14
JavaScript
15
Jekyll
3
Jenkins
7
Linux
34
Log4View
1
MahApps.Metro
3
MaterialDesignInXamlToolkit
1
MkDocs
2
MongoDB
5
MVC
1
MVVM
6
nginx
3
NLog
3
Node.js
8
npm
1
NVIDIA
3
onvif
1
OpenAPI
2
OpenCV
4
OpenSSL
3
OpenVINO
2
ownCloud
2
pandas
1
Pine Script
1
PlantUML
5
Portainer
3
PowerShell
8
Prism
2
PySide
19
Python
88
PyTorch
1
RabbitVCS
1
Razor
3
redis
1
Redmine
33
Redoc
1
remark.js
2
rocketchat
10
Ruby
3
scikit-learn
1
shotcut
1
SignalR
1
Slack
1
Socket.IO
1
SonarQube
5
Sphinx
10
SQL Server
5
SQLite
1
StableDiffusion
2
Subversion
2
Swagger
1
Swarmpit
1
Syslog
3
Telegraf
6
Tesseract
3
TestLink
2
Tomcat
2
TortoiseGit
11
TortoiseSVN
2
Trading View
1
Traefik
3
Travis CI
1
Ubuntu
31
Visual Studio
39
Visual Studio Code
10
VSCode
8
Vue.js
8
Windows
62
Windows 10
5
Windows ADK
1
Windows API
2
Windows Embedded
4
wkhtmltopdf
2
Word
3
WPF
12
WSL
5
WSL2
5
Xamarin
1
xUnit
5
yaml
1
yolo
1
アプリケーション
1
デザインパターン
1
テスト
1
バッチファイル
2
プログラミング
4
ライセンス
1
暗号資産(仮想通貨)
1
英語
2
確定申告
1
機械学習
1
強化学習
1
雑記
1
書籍
1
数学
1
正規表現
1
動画編集
1
Blog Archive
►
2024
(21)
►
12月
(3)
►
9月
(5)
►
8月
(1)
►
7月
(2)
►
6月
(1)
►
4月
(2)
►
3月
(1)
►
2月
(5)
►
1月
(1)
►
2023
(30)
►
12月
(3)
►
11月
(5)
►
10月
(2)
►
9月
(1)
►
8月
(2)
►
7月
(4)
►
6月
(2)
►
5月
(3)
►
4月
(2)
►
3月
(2)
►
2月
(3)
►
1月
(1)
▼
2022
(106)
►
12月
(5)
►
11月
(1)
►
10月
(3)
►
9月
(6)
▼
8月
(7)
[Docker] コンテナ内の実行ユーザーをホストのユーザーと合わせる
Telegraf で CPU の温度を取得する
overlayroot 環境内で docker を使う方法
Rocket.Chat の REST API でダイレクトメッセージを送る
[Python] アドレス (id) は変更せずに辞書 (dict) をコピーしたい
InfluxDB のコンテナ作成時にデータベースとリテンションポリシーの作成を行う
Ubuntu に nginx を入れてリバースプロキシとして使用する
►
7月
(6)
►
6月
(13)
►
5月
(9)
►
4月
(15)
►
3月
(11)
►
2月
(14)
►
1月
(16)
►
2021
(85)
►
12月
(11)
►
11月
(6)
►
10月
(4)
►
9月
(10)
►
8月
(8)
►
7月
(4)
►
6月
(18)
►
5月
(7)
►
4月
(8)
►
3月
(2)
►
2月
(2)
►
1月
(5)
►
2020
(56)
►
12月
(1)
►
11月
(3)
►
10月
(3)
►
9月
(3)
►
8月
(3)
►
7月
(7)
►
6月
(7)
►
5月
(2)
►
4月
(6)
►
3月
(6)
►
2月
(3)
►
1月
(12)
►
2019
(92)
►
12月
(13)
►
11月
(9)
►
10月
(3)
►
9月
(2)
►
8月
(3)
►
7月
(5)
►
6月
(11)
►
5月
(6)
►
4月
(17)
►
3月
(9)
►
2月
(6)
►
1月
(8)
►
2018
(100)
►
12月
(1)
►
11月
(11)
►
10月
(8)
►
9月
(6)
►
8月
(10)
►
7月
(10)
►
6月
(8)
►
5月
(9)
►
4月
(8)
►
3月
(14)
►
2月
(4)
►
1月
(11)
►
2017
(117)
►
12月
(14)
►
11月
(20)
►
10月
(17)
►
9月
(19)
►
8月
(10)
►
7月
(8)
►
6月
(3)
►
5月
(6)
►
4月
(5)
►
3月
(2)
►
2月
(8)
►
1月
(5)
►
2016
(91)
►
12月
(5)
►
11月
(9)
►
10月
(11)
►
9月
(9)
►
8月
(6)
►
7月
(14)
►
6月
(14)
►
5月
(11)
►
4月
(10)
►
3月
(2)
►
2015
(23)
►
12月
(4)
►
11月
(2)
►
10月
(8)
►
9月
(8)
►
7月
(1)
►
2013
(3)
►
11月
(1)
►
9月
(1)
►
7月
(1)
►
2012
(2)
►
7月
(1)
►
6月
(1)
►
2011
(1)
►
9月
(1)
►
2009
(1)
►
7月
(1)
►
2008
(2)
►
11月
(1)
►
7月
(1)
►
2007
(3)
►
10月
(3)