現代瀏覽器限制到一個host並發連接的數量一般為4或6。這意味着,如果您的web頁面加載幾十個來自同一個host的assert file(js、圖像、css)時,由於並發數的限制,會產生排隊。同樣甚至更糟的是,這個問題也會發生在使用了session的php腳本中。
問題:
php的session缺省用文件存儲,當請求一個需要操作session的php文件(session_start())時,這個文件是會被第一個操作session的進程鎖定,導致其他請求阻塞。其他請求會掛起在session_start()直到session文件解鎖。
解決:
由於鎖定的session文件直到腳本執行結束或者session正常關閉才會解鎖,為了防止大量的php請求(需要使用$_SESSION數據)被鎖定,可以在寫完session后馬上關閉,這樣就釋放了鎖。
關閉session:
session_write_close();
這個技巧非常管用,尤其對於一個處理時間長的腳本。並且這個函數只是關閉了寫session,讀還是可以的。
// session_start(); //可以讀寫session $_SESSION['latestRequestTime'] = time(); //關閉session session_write_close(); //讀取session $twitterId = $_SESSION['twitterId'];
譯注:
在php5.4以后,session_set_save_handler 支持了傳遞SessionHandlerInterface 的方式,第二個參數就是指定session_write_close() 作為回調方法的(缺省就是true),將函數
session_write_close() 注冊為 register_shutdown_function() 函數。
用Memcache或者Redis做session的存儲,是能解決“鎖定”的問題,但處理不好會導致連接數標高(在session操作后如果有耗時操作,連接是不回收的,可以主動在session寫操作完成后做session_write_close() 操作)