php 查看文件锁定状态_php文件锁死锁怎么办
什么是死鎖
學(xué)過操作系統(tǒng)的通同學(xué),都了解多線程的概念。在多線程中訪問公共資源,需要對資源加鎖。訪問結(jié)束后,釋放鎖。如果沒有釋放鎖,那么下一個線程來獲取資源的時候就會永遠都無法獲取資源的鎖,于是這個線程死鎖了。那么CGI是多線程的公共資源訪問導(dǎo)致的死鎖嗎? 答案是NO。
1. CGI 是單線程進程,通過ps 就能看到。(進程狀態(tài) Sl的才是多線程進程)。
2. 即使是多線程的,死鎖發(fā)生在PHP的shutdown過程中調(diào)用glibc 中time 函數(shù)的位置,不是php模塊造成的。而glibc 中的time相關(guān)函數(shù)是線程安全的,不會產(chǎn)生死鎖。
什么導(dǎo)致的死鎖呢?
通過分析linux中死鎖產(chǎn)生的機制,發(fā)現(xiàn)除了多線程會產(chǎn)生死鎖外,信號處理函數(shù)同樣會產(chǎn)生死鎖。那么cgi是由于信號處理導(dǎo)致的死鎖嗎?在這之前介紹一個感念。
函數(shù)的可重入性與信號安全
函數(shù)可重入是指,無論第幾次進入該函數(shù),函數(shù)都能正常執(zhí)行并返回結(jié)果。那么線程安全函數(shù)是可重入的嗎?答案是NO。 線程安全函數(shù),在第一次訪問公共資源時,會獲取全局鎖。如果函數(shù)沒有執(zhí)行完成,鎖還沒釋放,此時進程被中斷。那么在中斷處理函數(shù)中,再次訪問該函數(shù),就會產(chǎn)生死鎖。
那么什么樣的函數(shù)才可以在中斷處理函數(shù)中訪問呢?
除了沒有使用全局鎖的函數(shù),還有一些signal safe的系統(tǒng)調(diào)用可以使用。調(diào)用任何其他的非signal safe的函數(shù)都會產(chǎn)生不可預(yù)知的后果(比如 死鎖)。詳見 man signal。在分析死鎖的原因前,我們先看看cgi執(zhí)行的流程,分析其中有沒有產(chǎn)生死鎖的可能。
PHP-CGI的執(zhí)行流程
Glibc中的時間函數(shù)使用到了全局鎖,保證函數(shù)的線程安全,但沒有保證信號安全(signal safe)。經(jīng)過之前的分析,我們初步懷疑死鎖是由于PHP-CGI進程接收到了一個信號,然后在signal handle中執(zhí)行了非signal safe的函數(shù)。主流程在中斷前,正在執(zhí)行g(shù)libc中的時間函數(shù)。在函數(shù)獲取的鎖沒釋放前,進入中斷流程。而中斷過程中又訪問了glibc中的時間函數(shù)。于是導(dǎo)致了死鎖。
PHP-CGI的執(zhí)行流程,如下圖所示:
解決辦法:
去掉或簡化qalarm注冊到shutdown中的鉤子函數(shù)。避免不安全的函數(shù)調(diào)用。
以上內(nèi)容僅供參考!
總結(jié)
以上是生活随笔為你收集整理的php 查看文件锁定状态_php文件锁死锁怎么办的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 让机器人学会咖啡拉花,得从流体力学搞起,
- 下一篇: 为减少打嗝、放屁!英国计划给奶牛喂甲烷抑