使用PHP靜態(tài)變量當緩存的方法
更新時間:2013年11月13日 10:39:52 作者:
本文介紹如何使用PHP靜態(tài)變量當緩存,減少代碼重復(fù)執(zhí)行,優(yōu)化程序性能
下面這個PHP的代碼實例,功能是幫助用戶重置密碼,requestResetPassword是接收用戶重置密碼的請求并且做了相應(yīng)的檢查。為了更好的復(fù)用性,我將重置密碼的操作單獨分配到一個新的resetPassword的函數(shù),更改完密碼的后再調(diào)用sendEmail向用戶發(fā)送一封通知郵件。
/**
* 用戶請求重置密碼的接收器
*/
function requestResetPassword() {
//檢查用戶是否存在
if( !checkUserExists( $_GET['userid'] ) ) {
exit('抱歉,用戶不存在,請確認用戶帳號。');
}
resetPassword( $_GET['userid'] );
//最后向用戶發(fā)送一封郵件
sendEmail( $_GET['userid'], '重置密碼成功', '新的密碼是xxxx' );
exit('新密碼已經(jīng)發(fā)送到你的郵箱。');
}
/**
* 幫助用戶重置密碼
*/
function resetPassword( $userid ) {
//檢查用戶是否存在
if( !checkUserExists( $userid ) ) {
return false;
}
//進行重置用戶密碼的操作
//略...
return true;
}
/**
* 向用戶發(fā)送一封郵件
*/
function sendEmail( $userid, $title, $content ) {
//檢查用戶是否存在
if( !checkUserExists( $userid ) ) {
return false;
}
//發(fā)送郵件操作
//略...
return true;
}
/**
* 檢查某個用戶是否存在
*/
function checkUserExists( $userid ) {
$user = getUserInfo( $userid );
return !empty( $user );
}
/**
* 獲取某個用戶的數(shù)據(jù)
*/
function getUserInfo( $userid ) {
//假設(shè)我有一個query的函數(shù),它用來查詢數(shù)據(jù)庫并返回數(shù)據(jù)
$user = query( "SELECT * FROM `user` WHERE `uid`=" . intval( $userid ) );
return is_array( $user ) ? $user : array() ;
}
現(xiàn)在問題是,這三個函數(shù)都同時使用checkUserExists這個函數(shù)來檢查用戶不存在,數(shù)據(jù)庫查詢了三次,這樣帶來了一些額外的開銷。
如果要去掉三者之間任意一個checkUserExists,看上去是可能的。但是如果之后有某些功能要調(diào)用resetPassword或者sendEmail,用戶不存在時,系統(tǒng)可能會發(fā)生錯誤。
還有一個解決方法是,將resetPassword的邏輯寫到requestResetPassword里,再過一點,把sendEmail的邏輯也寫進去。這樣函數(shù)調(diào)用減少,數(shù)據(jù)庫查詢也變成一次了,性能得到了提高。但是重置密碼和發(fā)送郵件的功能將不能得到復(fù)用,并且違背了單一責任的原則,代碼復(fù)雜度也提高了。
不過,因為函數(shù)分離和復(fù)用性都很好,如果實際性能受到影響,可能考慮用緩存的方法減少數(shù)據(jù)庫查詢,我改動了它們共用的checkUserExists函數(shù):
/**
* 檢查某個用戶是否存在
*/
function checkUserExists( $userid ) {
//增加一個緩存,用以記錄檢查用戶的結(jié)果
static $cache = array();
//檢查當前用戶是否已經(jīng)檢查過一次
if( isset( $cache[ $userid ] ) ) {
return $cache[ $userid ];
}
$user = getUserInfo( $userid );
//把結(jié)果記錄到緩存中
$cache[ $userid ] = !empty( $user );
return $cache[ $userid ];
}
也可以用同樣的方法改動getUserInfo函數(shù)。
這里可以看到,當代碼的復(fù)用性提高時,想提高性能是很簡單的,性能的瓶頸也很容易被發(fā)現(xiàn)和修改。
盡管這個例子對性能影響還不夠大,還有一些影響更大的,比如說遍歷,我可能為了復(fù)用而將遍歷封裝到一個函數(shù)中,并且多次使用它。這些開銷對我的項目根本沒有預(yù)想中那樣有太大的影響,或者說是微乎其微的。所以我更愿意把時間花在如何提高代碼的復(fù)用性和維護性方面,而不是糾結(jié)于浪費多這一點性能。實際性能如果真的達不到要求,也可以權(quán)衡增加硬件配置。
復(fù)制代碼 代碼如下:
/**
* 用戶請求重置密碼的接收器
*/
function requestResetPassword() {
//檢查用戶是否存在
if( !checkUserExists( $_GET['userid'] ) ) {
exit('抱歉,用戶不存在,請確認用戶帳號。');
}
resetPassword( $_GET['userid'] );
//最后向用戶發(fā)送一封郵件
sendEmail( $_GET['userid'], '重置密碼成功', '新的密碼是xxxx' );
exit('新密碼已經(jīng)發(fā)送到你的郵箱。');
}
/**
* 幫助用戶重置密碼
*/
function resetPassword( $userid ) {
//檢查用戶是否存在
if( !checkUserExists( $userid ) ) {
return false;
}
//進行重置用戶密碼的操作
//略...
return true;
}
/**
* 向用戶發(fā)送一封郵件
*/
function sendEmail( $userid, $title, $content ) {
//檢查用戶是否存在
if( !checkUserExists( $userid ) ) {
return false;
}
//發(fā)送郵件操作
//略...
return true;
}
/**
* 檢查某個用戶是否存在
*/
function checkUserExists( $userid ) {
$user = getUserInfo( $userid );
return !empty( $user );
}
/**
* 獲取某個用戶的數(shù)據(jù)
*/
function getUserInfo( $userid ) {
//假設(shè)我有一個query的函數(shù),它用來查詢數(shù)據(jù)庫并返回數(shù)據(jù)
$user = query( "SELECT * FROM `user` WHERE `uid`=" . intval( $userid ) );
return is_array( $user ) ? $user : array() ;
}
現(xiàn)在問題是,這三個函數(shù)都同時使用checkUserExists這個函數(shù)來檢查用戶不存在,數(shù)據(jù)庫查詢了三次,這樣帶來了一些額外的開銷。
如果要去掉三者之間任意一個checkUserExists,看上去是可能的。但是如果之后有某些功能要調(diào)用resetPassword或者sendEmail,用戶不存在時,系統(tǒng)可能會發(fā)生錯誤。
還有一個解決方法是,將resetPassword的邏輯寫到requestResetPassword里,再過一點,把sendEmail的邏輯也寫進去。這樣函數(shù)調(diào)用減少,數(shù)據(jù)庫查詢也變成一次了,性能得到了提高。但是重置密碼和發(fā)送郵件的功能將不能得到復(fù)用,并且違背了單一責任的原則,代碼復(fù)雜度也提高了。
不過,因為函數(shù)分離和復(fù)用性都很好,如果實際性能受到影響,可能考慮用緩存的方法減少數(shù)據(jù)庫查詢,我改動了它們共用的checkUserExists函數(shù):
復(fù)制代碼 代碼如下:
/**
* 檢查某個用戶是否存在
*/
function checkUserExists( $userid ) {
//增加一個緩存,用以記錄檢查用戶的結(jié)果
static $cache = array();
//檢查當前用戶是否已經(jīng)檢查過一次
if( isset( $cache[ $userid ] ) ) {
return $cache[ $userid ];
}
$user = getUserInfo( $userid );
//把結(jié)果記錄到緩存中
$cache[ $userid ] = !empty( $user );
return $cache[ $userid ];
}
也可以用同樣的方法改動getUserInfo函數(shù)。
這里可以看到,當代碼的復(fù)用性提高時,想提高性能是很簡單的,性能的瓶頸也很容易被發(fā)現(xiàn)和修改。
盡管這個例子對性能影響還不夠大,還有一些影響更大的,比如說遍歷,我可能為了復(fù)用而將遍歷封裝到一個函數(shù)中,并且多次使用它。這些開銷對我的項目根本沒有預(yù)想中那樣有太大的影響,或者說是微乎其微的。所以我更愿意把時間花在如何提高代碼的復(fù)用性和維護性方面,而不是糾結(jié)于浪費多這一點性能。實際性能如果真的達不到要求,也可以權(quán)衡增加硬件配置。
您可能感興趣的文章:
相關(guān)文章
Laravel統(tǒng)一封裝接口返回狀態(tài)實例講解
這篇文章主要介紹了Laravel統(tǒng)一封裝接口返回狀態(tài)實例講解,封裝接口返回狀態(tài)有利于前后端分離項目的合作開發(fā),有正好需要的同學(xué)可以研究下2021-03-03PHP獲取IP地址所在地信息的實例(使用純真IP數(shù)據(jù)庫qqwry.dat)
下面小編就為大家?guī)硪黄狿HP獲取IP地址所在地信息的實例(使用純真IP數(shù)據(jù)庫qqwry.dat)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2016-11-11ajax完美實現(xiàn)兩個網(wǎng)頁 分頁功能的實例代碼
ajax完美實現(xiàn)兩個網(wǎng)頁 分頁功能的實例代碼,需要的朋友可以參考一下2013-04-04