推 LiloHuang: 我想先說明,終止線程跟 OS 本身有極高的相關性 08/22 20:45
→ LiloHuang: 我印象中目前並無某種方法,或某個內建的 API 08/22 20:45
→ LiloHuang: 可以保證的在所有的系統上,終止某個特定線程。 08/22 20:46
→ LiloHuang: 至少 Win32 得用 TerminateThread,不同於 pthread 08/22 20:46
→ LiloHuang: 我先假設你所使用的作業系統所操作的是 pthread 08/22 20:46
→ LiloHuang: 無論是透過 get_ident 或用 ident 都是拿到 pthread_t 08/22 20:46
→ LiloHuang: 也就是 pthread_self() 會拿到的內容 (可查源碼參照) 08/22 20:46
→ LiloHuang: 你拿到的數值會有點大,我猜你的系統應該是 64 bit 08/22 20:46
→ LiloHuang: StackOverflow 那則被打勾的回應是常見的好方法 08/22 20:47
→ LiloHuang: 至少這個方式不是真正強制硬砍掉 Thread 運行 08/22 20:47
→ LiloHuang: 然而之所以你會拿到 invalid thread ID 的原因是 08/22 20:47
→ LiloHuang: 你得改所有用到 PyThreadState_SetAsyncExc 的部分 08/22 20:47
→ LiloHuang: foobar = ctypes.pythonapi.PyThreadState_SetAsyncExc 08/22 20:47
→ LiloHuang: foobar.argtypes = [ctypes.c_long, ctypes.py_object] 08/22 20:47
→ LiloHuang: res = foobar(tid, ctypes.py_object(exctype)) 08/22 20:48
→ LiloHuang: 請把原文章中的 _async_raise 前幾行,適度修改如上 08/22 20:48
→ LiloHuang: 如此一來就應該不會出現 invalid thread ID 的狀況 08/22 20:49
→ LiloHuang: 原因是在 64 bit 的環境中,得寫清楚帶進去的資料型態 08/22 20:50
→ LiloHuang: 再透過 _async_raise(thr.ident, RuntimeError) 來終止 08/22 20:51
推 LiloHuang: 至於是否有更好的做法? 可以開 child process 也是一種 08/22 20:53
→ LiloHuang: 可以把傷害擺到另外一個進程,儘管還是會有機率出狀況 08/22 20:55
→ LiloHuang: 最好的方式是改 3rd party lib 使其 graceful exit :D 08/22 20:55
→ LiloHuang: 行有餘力還可以順便 merge 回 master/trunk 之類的 XD 08/22 20:57
感謝你的回覆,後來發現裏面推文就有了 忘記來這邊改XD
果然發文之前要好好爬文啊~
擺到另外一個 process 的方式下面推文也有,不過感覺其實是差不多的~
至於改 code... trace 了一下裏面有使用 requests 這個 lib
用 session.get() 經過 verification 抓 json 會造成 block
比較好的方式是改用 async 的方法再去 busy wait
要中斷就 set flag 結束 while loop 這樣 XD?
可惜不能 merge 回去,因爲這段 code 最近因爲被某 L 社要求撤掉了 Orz
※ 編輯: carylorrk (1.34.244.41), 08/23/2014 00:07:49
推 LiloHuang: 其實我也沒仔細看所有回應,只是要說 ctypes 在 64bit 08/23 00:14
→ LiloHuang: 的環境得多設定 argtypes 或者直接類似該討論串中某篇 08/23 00:15
→ LiloHuang: 直接生出 c_long(tid) 再進行帶入的動作... 08/23 00:15
→ LiloHuang: 最好的方式就是用 event / mutex 實作 graceful exit 08/23 00:15
→ LiloHuang: 或者用一個 atomic variable 來當作 stop flag 之類的 08/23 00:16
→ LiloHuang: 畢竟沒看到 3rd party lib 的原始碼,我只能猜到這 XD 08/23 00:18
推 LiloHuang: 至於 Python 還有個 GIL,如果用一些強制力砍掉 thread 08/23 00:21
→ LiloHuang: 如果剛好被砍掉的 thread 是持有 GIL 的話,那就有趣了 08/23 00:21
→ LiloHuang: 如果是 mission critical 的環境,就得考量更多的細節 08/23 00:23
→ LiloHuang: 總之恭喜你有把問題解決囉 :D 08/23 00:24
→ LiloHuang: 畢竟提及 child process 只是想避掉 GIL 這個老問題 08/23 00:27
瞭解~
不過如果用 async raise 還會有 GIL 的問題嗎?
畢竟不是 kill thread,應該會 release GIL 吧~
※ 編輯: carylorrk (1.34.244.41), 08/23/2014 01:16:53
推 LiloHuang: 不應該有GIL的問題,誠如我一開始提到這不是強制砍掉它 08/23 01:35
→ LiloHuang: 所謂的強制力是指 pthread_kill or TerminateThread 08/23 01:37
→ LiloHuang: 然而如果能實作 graceful exit 還是最好的方向就是 :) 08/23 01:38
推 LiloHuang: 因為 PyThreadState_SetAsyncExc 是請求觸發 exception 08/23 01:58
→ LiloHuang: 並不會真的立刻把該 thread 給終止掉,聽起來雖然不錯 08/23 01:59
→ LiloHuang: 然而如果該 thread 不幸卡在一個很久的 blocking I/O 08/23 02:01
→ LiloHuang: 或者該 thread 剛好操作某個會卡很久的 C-API 08/23 02:02
→ LiloHuang: 就不一定能如同原本預期,順利的把 thread 給終止掉了 08/23 02:03
→ LiloHuang: 畢竟這個的原理是在 PyEval_EvalFrameEx 還沒跑 opcode 08/23 02:05
→ LiloHuang: 之前進行檢查 tstate->async_exc 的值是否有被設定 :D 08/23 02:05
原來如此!如果沒有解說我還真把它當神奇的黑魔法 XDD
看來還需要研究一下細節,說不定哪天手上沒有 source code 就用的到了XD
感謝~
※ 編輯: carylorrk (1.34.244.41), 08/23/2014 03:10:49