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

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

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

Python 3はPythonプログラミング言語の最新バージョンであり、2008年12月3日にリリースされました。

Python

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

Q&A

解決済

2回答

1790閲覧

Python 3.8.4で並列処理しようとすると、グローバルに存在するコードが4度呼ばれる。

GUWAGUWA

総合スコア20

Python 3.x

Python 3はPythonプログラミング言語の最新バージョンであり、2008年12月3日にリリースされました。

Python

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

0グッド

0クリップ

投稿2020/07/24 13:41

Python 3.8.4で並列処理をしようと考えております。
下記のようなコードを書くと、グローバルに存在する「print("aaaa")」が4度呼ばれます。
なぜでしょうか?また、これを起こさないためにはどうすればよいのでしょうか。

Python

1import concurrent.futures as confu 2import time 3 4print("aaaa") 5 6def task(v): 7 print("start", v) 8 time.sleep(1.0) 9 print("end", v) 10 11def main(): 12 print("main start") 13# - - - - 処理開始 - - - - # 14 with confu.ProcessPoolExecutor(max_workers=3) as executor: 15 executor.submit(task, 5) 16 executor.submit(task, 10) 17 print("submit end") 18 19# - - - - 並列処理が終わった後 - - - - # 20 print("main end") 21 22if __name__ == '__main__': 23 main()

Console

1aaaa 2main start 3submit end 4aaaa 5aaaa 6aaaa 7start 5 8start 10 9end 5 10end 10 11main end

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

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

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

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

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

guest

回答2

0

ベストアンサー

https://docs.python.org/ja/3/library/multiprocessing.html#contexts-and-start-methods

spawn
親プロセスは新たに python インタープリタープロセスを開始します。(略)

Unix と Windows で利用可能。Windows と macOS でのデフォルト。

ワーカー数の分だけ子プロセスを作り、それぞれがpythonを実行します。
task関数が実行できるようになるためにはそれぞれのPythonでこのモジュールをインポートしなければなりません。(そうしないとtask関数を定義するdef文が実行されません)
したがってモジュールに書いてあるトップレベルの文はそれぞれのPythonで実行されます。


子プロセスでインポートされるときのモジュール名が__main__でないことで、マルチプロセッシングによってインポートされたことが区別できます。

https://docs.python.org/ja/3/library/multiprocessing.html#the-spawn-and-forkserver-start-methods を参考にしてください。

投稿2020/07/24 14:30

quickquip

総合スコア11038

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

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

y_shinoda

2020/07/24 14:45

おお、ドキュメントに書いてありますね 最後のリンクは「メインモジュールの安全なインポート」ってタイトルのところに・・ > 代わりに、次のように if __name__ == '__main__': を使用して > プログラムの "エントリポイント" を保護すべきです
quickquip

2020/07/24 14:55

残念なことにそこにアンカーがないんです…
GUWAGUWA

2020/07/25 03:21

ご回答ありがとうございます! ワーカ分モジュールをインポートするから、print("aaaa")もワーカ分+1回実行されるんですね! ありがとうございます!
guest

0

これを起こさないためにはどうすればよいのでしょうか。

こういう場合のために main() があると言えます:

python

1import concurrent.futures as confu 2import time 3 4 5def task(v): 6 print("start", v) 7 time.sleep(1.0) 8 print("end", v) 9 10def main(): 11 print("aaaa") 12 print("main start") 13# - - - - 処理開始 - - - - # 14 with confu.ProcessPoolExecutor(max_workers=3) as executor: 15 executor.submit(task, 5) 16 executor.submit(task, 10) 17 print("submit end") 18 19# - - - - 並列処理が終わった後 - - - - # 20 print("main end") 21 22if __name__ == '__main__': 23 main()

なぜでしょうか?

おそらくですが、
関数が実行される際に max_workers で指定した回数だけ
このモジュールが評価されるコードになっているのだと思われます

詳細を調べる場合は、コードを追っていけば、いつかは該当の処理が見つかるものと思われます:
cpython/init.py at v3.8.4 · python/cpython

(かなり奥深いところのようなので、僕は調査を断念しました)

投稿2020/07/24 14:11

y_shinoda

総合スコア3272

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

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

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問