へっぽこプログラマーの備忘録
プログラムを中心とした個人的なメモ用のブログです。 タイトルは迷走中。
内容の保証はできませんのであしからずご了承ください。
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
マージ元ブランチとマージ先ブランチ
TortoiseGit でブランチ間の差分を見る
[Python] 文字列の判定で、None と空文字を同時に判定する
[Python] matplotlib で plot する際に "Tcl_AsyncDelete: async handler deleted by the wrong thread" というエラーがでる
[Python] Tesseract で OCR を行ったら `UnicodeEncodeError: 'ascii' codec can't encode characters` のエラーが表示された
image
NO IMAGE
GitLab Runner でクローンする URL を変更する
image
NO IMAGE
Visual Studio で文字がにじむ(ぼやける)
Labels
.NET Core
31
.NET Framework
17
.NET Standard
2
AdminLTE
1
AI
1
Apache
3
AppVeyor
2
AsciiDoc
3
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
79
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
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
18
Python
86
PyTorch
1
RabbitVCS
1
Razor
3
Redmine
33
Redoc
1
remark.js
2
rocketchat
10
Ruby
3
scikit-learn
1
SignalR
1
Slack
1
Socket.IO
1
SonarQube
5
Sphinx
10
SQL Server
5
SQLite
1
StableDiffusion
1
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
Blog Archive
►
2024
(9)
►
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)