淺析PHP中Session可能會(huì)引起并發(fā)問(wèn)題
在進(jìn)行Web應(yīng)用程序開(kāi)發(fā)的時(shí)候,人們經(jīng)常會(huì)用Session存儲(chǔ)數(shù)據(jù)。但可能有人不知道,在PHP中,Session使用不當(dāng)可能會(huì)引起并發(fā)問(wèn)題。印度醫(yī)療行業(yè)軟件解決方案提供商Plus91 Technologies高級(jí)工程師Kishan Gor在個(gè)人博客上對(duì)這個(gè)問(wèn)題進(jìn)行了闡釋。
如果同一個(gè)客戶端并發(fā)發(fā)送多個(gè)請(qǐng)求,而每個(gè)請(qǐng)求都使用了Session,那么PHP Session鎖的存在會(huì)導(dǎo)致服務(wù)器串行響應(yīng)這些請(qǐng)求,而不是并行。這是因?yàn)樵谀J(rèn)情況下,PHP使用文件存儲(chǔ)Session數(shù)據(jù)。對(duì)于每一個(gè)新的Session,PHP會(huì)創(chuàng)建一個(gè)文件,并持續(xù)向其中寫(xiě)入數(shù)據(jù)。所以,每次調(diào)用session_start()方法,就會(huì)打開(kāi)Session文件,并取得文件的獨(dú)占鎖。這樣,如果服務(wù)器腳本正在處理一個(gè)請(qǐng)求,而客戶端又發(fā)送了一個(gè)同樣需要使用Session的請(qǐng)求,那么后一個(gè)請(qǐng)求會(huì)阻塞,直至前一個(gè)請(qǐng)求處理完成釋放了文件上的獨(dú)占鎖。不過(guò),這只限于來(lái)自同一個(gè)客戶端的多個(gè)請(qǐng)求,也就是說(shuō),來(lái)自一個(gè)客戶端的請(qǐng)求并不會(huì)阻塞另一個(gè)客戶端的請(qǐng)求。
如果腳本很短,這通常沒(méi)有問(wèn)題。但如果腳本運(yùn)行時(shí)間比較長(zhǎng),那就可能會(huì)產(chǎn)生問(wèn)題。在現(xiàn)代Web應(yīng)用程序開(kāi)發(fā)中,有一個(gè)非常常見(jiàn)的情況,就是使用AJAX技術(shù)在同一個(gè)頁(yè)面內(nèi)發(fā)送多個(gè)請(qǐng)求獲取數(shù)據(jù)。如果這些請(qǐng)求都需要使用Session,那么第一個(gè)請(qǐng)求到達(dá)服務(wù)器后會(huì)取得Session鎖,其它請(qǐng)求就必須等待,所有請(qǐng)求將串行處理,即使它們彼此之間并沒(méi)有依賴關(guān)系。這將大大增加頁(yè)面的響應(yīng)時(shí)間。
有一個(gè)方法可以避免這個(gè)問(wèn)題,就是在使用完Session以后立即調(diào)用session_write_close()方法關(guān)閉Session。這樣Session鎖就會(huì)釋放,即使當(dāng)前腳本還在等在處理。需要注意的是,調(diào)用該方法后,當(dāng)前腳本就不能進(jìn)一步操作Session了。
需要特別指出的是,本文所陳述的問(wèn)題和觀點(diǎn)只適用于使用session_start()方法的PHP默認(rèn)Session管理模式。比如,有用戶就指出,如果將應(yīng)用程序托管在AWS EC2上,并正確配置DynamoDB,Session鎖定問(wèn)題就不會(huì)出現(xiàn)。
附上一份實(shí)例代碼:
Session.php
<?php
final class SessionController extends YafController_Abstract
{
public function setUserFileAction()
{
session_start();
$_SESSION['user_name'] = 'xudianyang';
$_SESSION['user_id'] = '123';
sleep(3);
echo json_encode($_SESSION);
return false;
}
public function setLoginFileAction()
{
session_start();
$_SESSION['last_time'] = time();
echo json_encode($_SESSION);
return false;
}
public function indexFileAction()
{
// Auto Rend View
}
public function getSessionFileAction()
{
session_start();
var_dump($_SESSION);
return false;
}
public function setUserRedisAction()
{
$session = CoreFactory::session();
$session->set('user_name', 'xudianyang');
$session->set('user_id', '123');
sleep(3);
echo json_encode($_SESSION);
return false;
}
public function setLoginRedisAction()
{
$session = CoreFactory::session();
$session->set('last_time', time());
echo json_encode($_SESSION);
return false;
}
public function indexRedisAction()
{
// Auto Rend View
}
public function getSessionRedisAction()
{
$session = CoreFactory::session();
var_dump($_SESSION);
return false;
}
}
indexfile.phtml
<!DOCTYPE html>
<html>
<head>
<title>測(cè)試session并發(fā)鎖問(wèn)題</title>
<meta charset="utf-8">
<script type="text/javascript" src="/assets/js/jquery-1.10.2.min.js"></script>
<script type="text/javascript">
$.ajax({
url: "/session/setUserFile",
type: "get",
dataType: "json",
success: function(response){
console.info(response.last_time);
}
});
setTimeout(function(){
$.ajax({
url: "/session/setLoginFile",
type: "get",
dataType: "json",
success: function(response){
console.info(response.last_time);
}
});
}, 300);
</script>
</head>
<body>
同時(shí)發(fā)起2兩個(gè)ajax請(qǐng)求
</body>
</html>
indexredis.phtml
<!DOCTYPE html>
<html>
<head>
<title>測(cè)試session并發(fā)鎖問(wèn)題</title>
<meta charset="utf-8">
<script type="text/javascript" src="/assets/js/jquery-1.10.2.min.js"></script>
<script type="text/javascript">
$.ajax({
url: "/session/setUserRedis",
type: "get",
dataType: "json",
success: function(response){
console.info(response.last_time);
}
});
setTimeout(function(){
$.ajax({
url: "/session/setLoginRedis",
type: "get",
dataType: "json",
success: function(response){
console.info(response.last_time);
}
});
}, 300);
</script>
</head>
<body>
同時(shí)發(fā)起2兩個(gè)ajax請(qǐng)求
</body>
</html>
以上所述就是本文的全部?jī)?nèi)容了,希望大家能夠喜歡。
- PHP如何解決網(wǎng)站大流量與高并發(fā)的問(wèn)題
- php解決搶購(gòu)秒殺抽獎(jiǎng)等大流量并發(fā)入庫(kù)導(dǎo)致的庫(kù)存負(fù)數(shù)的問(wèn)題
- php并發(fā)對(duì)MYSQL造成壓力的解決方法
- php中并發(fā)讀寫(xiě)文件沖突的解決方案
- php并發(fā)加鎖示例
- php多線程并發(fā)實(shí)現(xiàn)方法
- php session的鎖和并發(fā)
- PHP編程中嘗試程序并發(fā)的幾種方式總結(jié)
- PHP接口并發(fā)測(cè)試的方法(推薦)
- php使用curl并發(fā)減少后端訪問(wèn)時(shí)間的方法分析
- PHP開(kāi)發(fā)中解決并發(fā)問(wèn)題的幾種實(shí)現(xiàn)方法分析
相關(guān)文章
php自定義函數(shù)call_user_func和call_user_func_array詳解
看UCenter的時(shí)候有一個(gè)函數(shù)call_user_func,百思不得其解,因?yàn)槲乙詾槭亲约憾x的函數(shù),結(jié)果到處都找不到,后來(lái)百度了一下才知道call_user_func是內(nèi)置函數(shù)2011-07-07
PHP 常用函數(shù)庫(kù)和一些實(shí)用小技巧
包括文件讀取函式,文件寫(xiě)入函式,靜態(tài)頁(yè)面生成函式,目錄刪除函式等2009-01-01
php使用redis的有序集合zset實(shí)現(xiàn)延遲隊(duì)列應(yīng)用示例
這篇文章主要介紹了php使用redis的有序集合zset實(shí)現(xiàn)延遲隊(duì)列,結(jié)合具體實(shí)例形式分析了PHP基于redis的有序集合zset實(shí)現(xiàn)延遲隊(duì)列的具體原理、應(yīng)用場(chǎng)景及相關(guān)操作技巧,需要的朋友可以參考下2020-02-02
php調(diào)用mysql存儲(chǔ)過(guò)程
php調(diào)用mysql存儲(chǔ)過(guò)程...2007-02-02
php基于socket實(shí)現(xiàn)SMTP發(fā)送郵件的方法
這篇文章主要介紹了php基于socket實(shí)現(xiàn)SMTP發(fā)送郵件的方法,實(shí)例分析了php采用socket實(shí)現(xiàn)smtp發(fā)送郵件的原理與技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-03-03
php中理解print EOT分界符和echo EOT的用法區(qū)別小結(jié)
這樣可以輸出大段的HTML 而且不用把里面的引號(hào)轉(zhuǎn)義 就是不用 \" 這樣自動(dòng)替換里面的變量。2010-02-02

