Python 的模塊一旦加載就會(huì )常駐內存,直到程序結束。再碰到 import 語(yǔ)句式只是修改名字空間,而不需要重新加載。這種機制是出于運行時(shí)的效率考慮,每遇到 import 的時(shí)候重新加載顯然很低效。它也不會(huì )檢查源文件的修改時(shí)間以確定是否重新加載,Python 有那么多的模塊,每次調用時(shí)都檢查一遍時(shí)間也是不行的。
這種機制下,開(kāi)發(fā)長(cháng)時(shí)間運行的守護程序就會(huì )很麻煩,修改源代碼后要重新啟動(dòng)程序才能讓新的代碼生效。比如用 mod_python 做 web 開(kāi)發(fā),Apache 會(huì )啟動(dòng)多個(gè)守護進(jìn)程來(lái)應答客戶(hù)請求,里面有 python 的解釋引擎和加載的模塊,若要讓修改后的代碼生效只能重起 apache,這會(huì )影響到其它服務(wù)的正常運行,非常不方便。mod_python 有一個(gè)PythonAutoReload 參數,它只是針對 PythonHandler 而言的,能夠對設定的 PythonHandler 實(shí)現自動(dòng)重新加載,而該 Handler 中所用到的模塊卻不能自動(dòng) reload。
這種修改源代碼然后重起 apache 的調試方式實(shí)在讓我無(wú)法忍受了,決定實(shí)現一種自動(dòng)重新加載機制?;镜乃悸肪褪敲總€(gè)用戶(hù)請求到來(lái)時(shí),檢查我所關(guān)心的那些模塊源文件的修改時(shí)間,如果比加載時(shí)的修改時(shí)間新,則重新加載。
編寫(xiě)一個(gè)檢測時(shí)間和重新加載的函數,讓它在每個(gè)請求到來(lái)時(shí)執行:
這段代碼不長(cháng),但是改了好多個(gè)版本,最開(kāi)始用 has_key() 的方式來(lái)檢測是否存在某個(gè)模塊,檢測該模塊是否有 loadtime 屬性( 用 module.__dict__ ),現在這種方式應該效率高一些,曾經(jīng)在一個(gè) blog 上看到過(guò)對比測試數據。起初還在每個(gè)關(guān)心的模塊里面加上一句loadtime = os.path.getmtime( __file__ ),這是不必要的,因為 Python 用的是動(dòng)態(tài)類(lèi)型,可以在運行時(shí)追加屬性,第一次檢測時(shí)設置初始狀態(tài)即可。
有了這段代碼,開(kāi)發(fā) BlogXP 方便多了,改了源碼之后立馬就能生效,而且它在正常運行時(shí)的消耗也很小。另外,由于mod_python 能夠實(shí)現指定的 Handler 的自動(dòng)重新加載,將這段代碼放在該 Handler 中,可以方便地改變所關(guān)心的模塊列表,也不需重起 apache。
聯(lián)系客服