php性能優(yōu)化之不要在for循環(huán)中操作DB
前言
如何提高程序運行速度,減輕服務(wù)器壓力是服務(wù)端開發(fā)必須面對的一個問題。
簡單且樸素的原則:不要在for循環(huán)中操作DB,包括關(guān)系型數(shù)據(jù)庫和NoSql。
我們應(yīng)該根據(jù)自己的業(yè)務(wù)場景,在for循環(huán)之前批量拿到數(shù)據(jù),用盡量少的sql查詢批量查到結(jié)果。 在for循環(huán)中進行數(shù)據(jù)的匹配組裝。
場景說明
- 業(yè)務(wù)在多個情景下需要獲得用戶的詳細(xì)信息,有點可以通過查詢用戶表直接獲取到,有的需要查詢關(guān)聯(lián)關(guān)系表獲取到,有的只保存了關(guān)聯(lián)的id,并沒有單獨創(chuàng)建關(guān)聯(lián)關(guān)系表,需要單獨寫獲取函數(shù)取值。
- 既然多個場景下需要調(diào)用,那么封裝成一個公共方法,讓多個場景統(tǒng)一調(diào)用公共方法是基本的優(yōu)化思路。
- 上面提到了復(fù)雜的存取值關(guān)系,我們需要分析一下,哪些操作是耗時的,耗時的操作如何優(yōu)化,能否減少sql查詢的次數(shù)。
舉例說明
- 下面的代碼示例,我們封裝了 CommonRender 的類,所有可以統(tǒng)一輸出的方法都在這里
- 下面代碼標(biāo)注了優(yōu)化之前和優(yōu)化之后
- 優(yōu)化之前:在每次查詢都需要根據(jù)保存的id,去數(shù)據(jù)庫查詢;如果列表頁每次返回30條數(shù)據(jù),那這部分就需要30次sql查詢。
- 優(yōu)化之后:采用的是提前批量取值,又寫了一個函數(shù) _renderHobby ,只需要1次sql。
- 這樣就極大的減少了sql查詢,提高了程序響應(yīng)的速度。
<?php
namespace App\Render;
.
.
.
class CommonRender extends BaseRender
{
public static function renderUserinfo($data, $hobbyInfo = [])
{
if (!is_array($data)) {
return [];
}
$ret = [
'uid' => !isset($data['id']) ? 0 : $data['id'],
'userid' => !isset($data['userid']) ? '' : $data['userid'],
'username' => !isset($data['username']) ? '' : $data['username'],
'usericon' => !isset($data['usericon']) ? [] : $data['usericon'],
.
.
.
//優(yōu)化之前
// 'hobby' => !isset($data['hobby']) ? [] : HobbyInfo::getByIds($data['hobby']),
//優(yōu)化之后
'hobby' => !isset($data['hobby']) ? [] : self::_renderHobby($data['hobby'], $hobbyInfo),
.
.
.
if (!empty($ret['birth'])) {
$ret['zodiacSign'] = Utility::getZodiacSign($ret['birth']);
} else {
$ret['zodiacSign'] = '';
}
return $ret;
}
protected static function _renderHobby($userHobby, $hobbyInfo)
{
$ret = [];
if ($userHobby) {
$userHobbyIds = explode(',', $userHobby);
foreach ($userHobbyIds as $key => $userHobbyId) {
$ret[$key] = $hobbyInfo[$userHobbyId];
}
}
return $ret;
}
//用戶列表卡片常用字段
public static function renderListCardUserinfo($data)
{
.
.
.
}
}
進一步優(yōu)化
上面的代碼已經(jīng)優(yōu)化了性能,但是還不夠優(yōu)雅。
獲取單用戶信息場景比較多,比如編輯,登錄,查看單人信息等,這種情況下我還每次都提前批量查詢嗎?這樣的話需要改造的地方太多了。
下面做進一步優(yōu)化:
在render方法內(nèi)部封裝了一層,如果外部沒有傳入或傳入空數(shù)組,自己再查詢db獲得一次需要的數(shù)據(jù)源。
<?php
namespace App\Render;
.
.
.
class CommonRender extends BaseRender
{
public static function renderUserinfo($data, $hobbyInfo = [])
{
//區(qū)別在這里:批量查詢外部傳入,減少sql查詢次數(shù); 單次查詢在render內(nèi)查一次
$hobbyInfo = !empty($hobbyInfo) ? $hobbyInfo : HobbyInfo::getAllInfo();
if (!is_array($data)) {
return [];
}
$ret = [
'uid' => !isset($data['id']) ? 0 : $data['id'],
'userid' => !isset($data['userid']) ? '' : $data['userid'],
'username' => !isset($data['username']) ? '' : $data['username'],
'usericon' => !isset($data['usericon']) ? [] : $data['usericon'],
.
.
.
//優(yōu)化之前
// 'hobby' => !isset($data['hobby']) ? [] : HobbyInfo::getByIds($data['hobby']),
//優(yōu)化之后
'hobby' => !isset($data['hobby']) ? [] : self::_renderHobby($data['hobby'], $hobbyInfo),
.
.
.
if (!empty($ret['birth'])) {
$ret['zodiacSign'] = Utility::getZodiacSign($ret['birth']);
} else {
$ret['zodiacSign'] = '';
}
return $ret;
}
protected static function _renderHobby($userHobby, $hobbyInfo)
{
$ret = [];
if ($userHobby) {
$userHobbyIds = explode(',', $userHobby);
foreach ($userHobbyIds as $key => $userHobbyId) {
$ret[$key] = $hobbyInfo[$userHobbyId];
}
}
return $ret;
}
//用戶列表卡片常用字段
public static function renderListCardUserinfo($data)
{
.
.
.
}
}
這樣,那些獲得單個用戶資料的方法就不需要修改了。
//編輯用戶資料
public function editUserInfo(Request $request)
{
$userInfo = UserInfo::editUserById($this->_userid, $request);
return [
'user' =>
CommonRender::renderUserinfo($userInfo)
+ UserInfo::formatCoverAndPickedFootprint($userInfo)
];
}
性能對比
批量獲得用戶信息對比:性能提升立竿見影。
- 比如每次取30個用戶數(shù)據(jù),之前獲得愛好,職業(yè),期望部分要查詢30次db。
- 優(yōu)化之后只需要查詢3次db。
public static function getBatchUserIntro($userid, $userList)
{
$retData = [];
if (empty($userList)) {
return $retData;
}
.
.
.
//批量獲得愛好、職業(yè)、期望遇到 在foreach中計算取值,不重復(fù)請求DB取值
$hobbyInfo = HobbyInfo::getAllInfo();
$professionInfo = ProfessionInfo::getAllInfo();
$expectInfo = ExpectInfo::getAllInfo();
foreach ($batchUserInfo as $item) {
$retData[$item['userid']] = array_merge(
['wxnumber' => Utility::maskWxnumber($item['wxnumber'], $batchExchangeStatus[$item['userid']] == UserUserWeixinExchange::TYPE_TRUE)]
+ CommonRender::renderUserinfo($item, $hobbyInfo, $professionInfo, $expectInfo);
}
.
.
.
return $retData;
}
注意,為了行文緊湊,代碼段中省略了和文章無關(guān)的代碼,用豎著的三個.省略。
以上就是php性能優(yōu)化之不要在for循環(huán)中操作DB的詳細(xì)內(nèi)容,更多關(guān)于php性能優(yōu)化for循環(huán)DB操作的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
PHP實現(xiàn)使用DOM將XML數(shù)據(jù)存入數(shù)組的方法示例
這篇文章主要介紹了PHP實現(xiàn)使用DOM將XML數(shù)據(jù)存入數(shù)組的方法,結(jié)合具體實例形式分析了php基于DOM實現(xiàn)xml數(shù)據(jù)讀取與解析相關(guān)操作技巧,需要的朋友可以參考下2017-09-09
Can''t create/write to file ''C:\WINDOWS\TEMP\...MYSQL報錯解決方法
Can't create/write to file 'C:\WINDOWS\TEMP\...MYSQL報錯解決方法,參考下面的方法即可。2011-06-06
PHP基于數(shù)組實現(xiàn)的堆棧和隊列功能示例
這篇文章主要介紹了PHP基于數(shù)組實現(xiàn)的堆棧和隊列功能,結(jié)合實例形式分析了php基于數(shù)組的array_push()、array_pop()、array_shift()等函數(shù)實現(xiàn)堆棧與隊列的入棧、出棧以及隊列的添加、刪除等相關(guān)操作技巧,需要的朋友可以參考下2018-01-01
Could not load type System.ServiceModel.Activation.HttpModul
本文章來詳細(xì)介紹關(guān)于Could not load type System.ServiceModel.Activation.HttpModule from assembly System.ServiceModel解決辦法,有需要的朋友可參考2012-12-12
使用ThinkPHP框架(thinkphp8.0)創(chuàng)建定時任的操作步驟
這篇文章給大家介紹了使用ThinkPHP框架(thinkphp8.0)創(chuàng)建定時任的操作步驟,文中通過代碼示例給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2024-01-01
使用Memcached緩存實現(xiàn)Session共享技巧
這篇文章主要為大家介紹了使用Memcached緩存實現(xiàn)Session共享技巧,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-11-11

