瀏覽器退出之後php還會繼續執行麼?
瀏覽器退出之後php還會繼續執行麼?
前提:這裡說的是典型的lnmp結構,nginx+php-fpm的模式
如果我有個php程式執行地非常慢,甚至於在程式碼中sleep(),然後瀏覽器連線上服務的時候,會啟動一個php-fpm程式,但是這個時候,如果瀏覽器關閉了,那麼請問,這個時候服務端的這個php-fpm程式是否還會繼續執行呢?
今天就是要解決這個問題。
最簡單的實驗
最簡單的方法就是做實驗,我們寫一個程式:在sleep之前和之後都用file_put_contents來寫入日誌:
<?php
file_put_contents(`/tmp/test.log`, `11111` . PHP_EOL, FILE_APPEND | LOCK_EX);
sleep(3);
file_put_contents(`/tmp/test.log`, `2222` . PHP_EOL, FILE_APPEND | LOCK_EX);
實際操作的結果是,我們在伺服器sleep的過程中,關閉客戶端瀏覽器,2222是會被寫入日誌中。
那麼就意味著瀏覽器關閉以後,服務端的php還是會繼續執行的?
ignore_user_abort
老王和diogin提醒,這個可能是和php的ignore_user_abort函式相關。
於是我就把程式碼稍微改成這樣的:
<?php
ignore_user_abort(false);
file_put_contents(`/tmp/test.log`, `11111` . PHP_EOL, FILE_APPEND | LOCK_EX);
sleep(3);
file_put_contents(`/tmp/test.log`, `2222` . PHP_EOL, FILE_APPEND | LOCK_EX);
發現並沒有任何軟用,不管設定ignore_user_abort為何值,都是會繼續執行的。
但是這裡有一個疑問: user_abort是什麼?
文件對cli模式的abort說的很清楚,當php指令碼執行的時候,使用者終止了這個指令碼的時候,就會觸發abort了。然後指令碼根據ignore_user_abort來判斷是否要繼續執行。
但是官方文件對cgi模式的abort並沒有描述清楚。感覺即使客戶端斷開連線了,在cgi模式的php是不會收到abort的。
難道ignore_user_abort在cgi模式下是沒有任何作用的?
是不是心跳問題呢?
首先想到的是不是心跳問題呢?我們斷開瀏覽器客戶端,等於在客戶端沒有close而斷開了連線,服務端是需要等待tcp的keepalive到達時長之後才會檢測出來的。
好,需要先排除瀏覽器設定的keepalive問題。
拋棄瀏覽器,簡單寫一個client程式:程式連線上http服務之後,傳送一個header頭,sleep1秒就主動close連線,而這個程式並沒有帶http的keepalive頭。
程式如下:
package main
import "net"
import "fmt"
import "time"
func main() {
conn, _ := net.Dial("tcp", "192.168.33.10:10011")
fmt.Fprintf(conn, "GET /index.php HTTP/1.0
")
time.Sleep(1 * time.Second)
conn.Close()
return
}
服務端程式:
<?php
ignore_user_abort(false);
file_put_contents(`/tmp/test.log`, `11111` . PHP_EOL, FILE_APPEND | LOCK_EX);
sleep(3);
file_put_contents(`/tmp/test.log`, `2222` . PHP_EOL, FILE_APPEND | LOCK_EX);
發現仍然還是一樣,php還是不管是否設定ignore_user_abort,會繼續執行完成整個指令碼。看來ignore_user_abort還是沒有生效。
如何觸發ignore_user_abort
那該怎麼觸發ignore_user_abort呢?服務端這邊怎麼知曉這個socket不能使用了呢?老王和diogin說是不是需要服務端主動和socket進行互動,才會判斷出這個socket是否可以使用?
另外,我們還發現,php提供了connection_status和connection_aborted兩個方法,這兩個方法都能檢測出當前的連線狀態。於是我們的打日誌的那行程式碼就可以改成:
file_put_contents(`/tmp/test.log`, `1 connection status: ` . connection_status() . "abort:" . connection_aborted() . PHP_EOL, FILE_APPEND | LOCK_EX);
根據手冊連線處理顯示我們可以列印出當前連線的狀態了。
下面還缺少一個和socket互動的程式,我們使用echo,後面也順帶記得帶上flush,排除了flush的影響。
程式就改成:
<?php
ignore_user_abort(true);
file_put_contents(`/tmp/test.log`, `1 connection status: ` . connection_status() . "abort:" . connection_aborted() . PHP_EOL, FILE_APPEND | LOCK_EX);
sleep(3);
for($i = 0; $i < 10; $i++) {
echo "22222";
flush();
sleep(1);
file_put_contents(`/tmp/test.log`, `2 connection status: ` . connection_status() . "abort:" . connection_aborted(). PHP_EOL, FILE_APPEND | LOCK_EX);
}
很好,執行我們前面寫的client。觀察日誌:
1 connection status: 0abort:0
2 connection status: 0abort:0
2 connection status: 1abort:1
2 connection status: 1abort:1
2 connection status: 1abort:1
2 connection status: 1abort:1
2 connection status: 1abort:1
2 connection status: 1abort:1
2 connection status: 1abort:1
2 connection status: 1abort:1
2 connection status: 1abort:1
終於製造出了abort。日誌也顯示後面幾次的abort狀態都是1。
但是這裡有個奇怪的地方,為什麼第一個2 connection status的狀態還是0呢(NORMAL)。
RST
我們使用wireshark抓包看整個客戶端和服務端互動的過程
這整個過程只有傳送14個包,我們看下服務端第一次傳送22222的時候,客戶端返回的是RST。後面就沒有進行後續的包請求了。
於是理解了,客戶端和服務端大概的互動流程是:
當服務端在迴圈中第一次傳送2222的時候,客戶端由於已經斷開連線了,返回的是一個RST,但是這個傳送過程算是請求成功了。直到第二次服務端再次想往這個socket中進行write操作的時候,這個socket就不進行網路傳輸了,直接返回說connection的狀態已經為abort。所以就出現了上面的情況,第一次222是status為0,第二次的時候才出現abort。
strace進行驗證
我們也可以使用strace php -S XXX來進行驗證
整個過程strace的日誌如下:
。。。
close(5) = 0
lstat("/tmp/test.log", {st_mode=S_IFREG|0644, st_size=49873651, ...}) = 0
open("/tmp/test.log", O_WRONLY|O_CREAT|O_APPEND, 0666) = 5
fstat(5, {st_mode=S_IFREG|0644, st_size=49873651, ...}) = 0
lseek(5, 0, SEEK_CUR) = 0
lseek(5, 0, SEEK_CUR) = 0
flock(5, LOCK_EX) = 0
write(5, "1 connection status: 0abort:0
", 30) = 30
close(5) = 0
sendto(4, "HTTP/1.0 200 OK
Connection: clo"..., 89, 0, NULL, 0) = 89
sendto(4, "111111111", 9, 0, NULL, 0) = 9
rt_sigprocmask(SIG_BLOCK, [CHLD], [], 8) = 0
rt_sigaction(SIGCHLD, NULL, {SIG_DFL, [], 0}, 8) = 0
rt_sigprocmask(SIG_SETMASK, [], NULL, 8) = 0
nanosleep({3, 0}, 0x7fff60a40290) = 0
sendto(4, "22222", 5, 0, NULL, 0) = 5
open("/tmp/test.log", O_WRONLY|O_CREAT|O_APPEND, 0666) = 5
fstat(5, {st_mode=S_IFREG|0644, st_size=49873681, ...}) = 0
lseek(5, 0, SEEK_CUR) = 0
lseek(5, 0, SEEK_CUR) = 0
flock(5, LOCK_EX) = 0
write(5, "2 connection status: 0abort:0
", 30) = 30
close(5) = 0
rt_sigprocmask(SIG_BLOCK, [CHLD], [], 8) = 0
rt_sigaction(SIGCHLD, NULL, {SIG_DFL, [], 0}, 8) = 0
rt_sigprocmask(SIG_SETMASK, [], NULL, 8) = 0
nanosleep({1, 0}, 0x7fff60a40290) = 0
sendto(4, "22222", 5, 0, NULL, 0) = -1 EPIPE (Broken pipe)
--- SIGPIPE {si_signo=SIGPIPE, si_code=SI_USER, si_pid=2819, si_uid=0} ---
open("/tmp/test.log", O_WRONLY|O_CREAT|O_APPEND, 0666) = 5
fstat(5, {st_mode=S_IFREG|0644, st_size=49873711, ...}) = 0
lseek(5, 0, SEEK_CUR) = 0
lseek(5, 0, SEEK_CUR) = 0
flock(5, LOCK_EX) = 0
write(5, "2 connection status: 1abort:1
", 30) = 30
close(5) = 0
rt_sigprocmask(SIG_BLOCK, [CHLD], [], 8) = 0
rt_sigaction(SIGCHLD, NULL, {SIG_DFL, [], 0}, 8) = 0
rt_sigprocmask(SIG_SETMASK, [], NULL, 8) = 0
nanosleep({1, 0}, 0x7fff60a40290) = 0
open("/tmp/test.log", O_WRONLY|O_CREAT|O_APPEND, 0666) = 5
fstat(5, {st_mode=S_IFREG|0644, st_size=49873741, ...}) = 0
lseek(5, 0, SEEK_CUR) = 0
lseek(5, 0, SEEK_CUR) = 0
flock(5, LOCK_EX) = 0
write(5, "2 connection status: 1abort:1
", 30) = 30
close(5)
。。。
我們照中看status從0到1轉變的地方。
...
sendto(4, "22222", 5, 0, NULL, 0) = 5
...
write(5, "2 connection status: 0abort:0
", 30) = 30
close(5) = 0
rt_sigprocmask(SIG_BLOCK, [CHLD], [], 8) = 0
rt_sigaction(SIGCHLD, NULL, {SIG_DFL, [], 0}, 8) = 0
rt_sigprocmask(SIG_SETMASK, [], NULL, 8) = 0
nanosleep({1, 0}, 0x7fff60a40290) = 0
sendto(4, "22222", 5, 0, NULL, 0) = -1 EPIPE (Broken pipe)
--- SIGPIPE {si_signo=SIGPIPE, si_code=SI_USER, si_pid=2819, si_uid=0} ---
open("/tmp/test.log", O_WRONLY|O_CREAT|O_APPEND, 0666) = 5
fstat(5, {st_mode=S_IFREG|0644, st_size=49873711, ...}) = 0
lseek(5, 0, SEEK_CUR) = 0
lseek(5, 0, SEEK_CUR) = 0
flock(5, LOCK_EX) = 0
write(5, "2 connection status: 1abort:1
", 30) = 30
close(5) = 0
第二次往socket中傳送2222的時候顯示了Broken pipe。這就是程式告訴我們,這個socket已經不能使用了,順便php中的connection_status就會被設定為1了。後續的寫操作也都不會再執行了。
總結
正常情況下,如果客戶端client異常推出了,服務端的程式還是會繼續執行,直到與IO進行了兩次互動操作。服務端發現客戶端已經斷開連線,這個時候會觸發一個user_abort,如果這個沒有設定ignore_user_abort,那麼這個php-fpm的程式才會被中斷。
至此,問題結了。
本文轉自軒脈刃部落格園部落格,原文連結:http://www.cnblogs.com/yjf512/p/5362025.html,如需轉載請自行聯絡原作者
相關文章
- java中異常丟擲後程式碼還會繼續執行嗎Java
- 2019 為什麼我們還會繼續使用 PHP ?PHP
- 通過 WebAssembly 在瀏覽器執行 PHPWeb瀏覽器PHP
- 瀏覽器執行緒瀏覽器執行緒
- 瀏覽器執行原理瀏覽器
- ie瀏覽器退役後還能用嗎 ie瀏覽器關閉停用以後怎麼辦瀏覽器
- php返回資料後如何讓程式繼續執行其它操作PHP
- 重學瀏覽器(1)-多程式多執行緒的瀏覽器瀏覽器執行緒
- 兄弟們還在繼續寫php嗎?PHP
- python:return之後的語句還會執行嗎Python
- 瀏覽器渲染程式多執行緒瀏覽器執行緒
- 瀏覽器EventLoop執行過程解析瀏覽器OOP
- 在瀏覽器中執行vscode -DEV瀏覽器VSCodedev
- 瀏覽器多執行緒和js單執行緒瀏覽器執行緒JS
- ie瀏覽器退役後還能用嗎 ie瀏覽器停止更新服務以後有影響嗎瀏覽器
- 如何在瀏覽器中執行 VS Code?瀏覽器
- 瀏覽器執行javaScript程式碼基礎瀏覽器JavaScript
- Rust在瀏覽器Wasm和後端伺服器中執行效能比較Rust瀏覽器ASM後端伺服器
- 疫情過後,遠端辦公軟體還會繼續普及嗎?
- 瀏覽器執行緒執行順序,瞭解一下瀏覽器執行緒
- JavaScript 執行機制-瀏覽器事件迴圈JavaScript瀏覽器事件
- JS在瀏覽器中的執行機制JS瀏覽器
- 【總結】瀏覽器的執行緒與程式瀏覽器執行緒
- Ooui:在瀏覽器中執行.NET應用UI瀏覽器
- 雲伺服器客戶端斷開後,如何保持程式繼續執行呢?伺服器客戶端
- 關於 SAP Spartacus CSR fallback 之後,是否仍然會繼續進行 SSR 的處理
- 瀏覽器之渲染引擎瀏覽器
- 瀏覽器之我見瀏覽器
- 通過瀏覽器執行cmd命令、啟動steam瀏覽器
- 在 vscode.dev 中直接執行 Python !純瀏覽器環境,無後端!VSCodedevPython瀏覽器後端
- Thread 中的 join() 方法的作用是呼叫執行緒等待該執行緒執行完後,再繼續執行thread執行緒
- 瀏覽器位址列輸入url回車之後發生了些什麼瀏覽器
- 從在瀏覽器中輸 URL 網址之後到底發生了什麼?瀏覽器
- 【譯】Go和WebAssembly:在瀏覽器中執行Go程式GoWeb瀏覽器
- Chrome、Edge瀏覽器內建多執行緒下載Chrome瀏覽器執行緒
- 當你在瀏覽器中輸入URL回車後會發生什麼?瀏覽器
- 從一道執行題,瞭解瀏覽器中JS執行機制瀏覽器JS
- 未來的瀏覽器會怎麼發展呢?瀏覽器
- 快取策略之瀏覽器快取瀏覽器