記一次對 Laravel-permission 項目的性能優化

file

我最近研究分析了在 SWIS上面建立的項目的性能。使人驚訝的是,最耗費性能的方法之一是優秀的  spatie/laravel-permission 包形成的。laravel

通過查閱更多資料和研究,發現一個可能明顯改善的性能問題 。既然解決方案已明確闡述,就很容易編寫代碼改善,提交請求。git

如今這個解決方案已被合併和發佈,下面是這個性能問題的分析和如何在本身的項目避免這類問題。github

TL;DR: 跳轉到結論部分.數據庫

性能瓶頸

若是咱們抽象的看 spatie/laravel-permission 它主要作兩件事:緩存

  1. 保持一個屬於某個模型的權限清單。
  2. 檢查某個模型是否具備權限。

第一點說是性能瓶頸有點牽強。這裏的權限數據存放在數據庫中,須要的時候將會被讀取出來。這個過程是有點慢但也只是執行一次。結果會被緩存下來,後續的請求能夠直接使用。性能

第二點在性能瓶頸的觀點上來看確實是一個瓶頸。 這個瓶頸取決於權限的性質和項目的大小, 由於權限會被頻繁的檢查。 在這個檢查的過程當中任何的遲鈍都會成爲整個項目的性能瓶頸。測試

過濾集合類

過濾權限集合的方法被認爲是形成低性能的緣由。 它作了以下事情:spa

$permission = $permissions
    ->where('id', $id)
    ->where('guard_name', $guardName)
    ->first();

修改後:code

$permission = $permissions
    ->filter(function ($permission) use ($id, $guardName) {
        return $permission->id === $id && $permission->guard_name === $guardName;
    })
    ->first();

這兩個代碼段實現了同一件事情,但第二個更快。orm

性能測試

我正在開發的應用中大約有 150 個不一樣的權限。 在一個普通的請求中, 大約有 50 個權限須要用  hasPermissionTo 這個方法去檢查,固然,有些頁面可能須要檢查大約 200 個權限。

如下是用來作性能測試的一些設置。

$users = factory(User::class, 150)->make();
$searchForTheseUsers = $users->shuffle()->take(50);

# 方法 1: where
foreach($searchForTheseUsers as $user) {
    $result = $users->where('id', '=', $user->id)->first();
}

# 方法 2: 過濾,傳遞一個模型做爲回調
foreach($searchForTheseUsers as $searchUser) {
    $result = $users->filter(function($user) use ($searchUser) {
        return $user->id === $searchUser->id;
    })->first();
}

# 方法 3: 過濾,傳遞屬性做爲回調
foreach($searchForTheseUsers as $user) {
    $searchId = $user->id;
    $result = $users->filter(function($user) use ($searchId) {
        return $user->id === $searchId;
    })->first();
}

以上三個方法都會被用來測試過濾 1 個屬性,2 個屬性,3 個屬性,因此,用方法 1 過濾三個屬性就會是這樣:

foreach($searchForTheseUsers as $user) {
    $result = $users
        ->where('id', '=', $user->id)
        ->where('firstname', '=', $user->firstname)
        ->where('lastname', '=', $user->lastname)->first();
}

結果

方法 #1 方法 #2 方法 #3
1個屬性 0.190 0.139 (-27%) 0.072 (-62%)
2個屬性 0.499 0.372 (-25%) 0.196 (-61%)
3個屬性 0.488 0.603 (+25%) 0.198 (-59%)

結論

咱們能夠得出結論:對一個項目而言,重複的過濾一個大集合會引起嚴重性能瓶頸。

多屬性的過濾明顯增長計算成本。

使用 Collection::filter() 代替 Collection::where() 能夠提升60%的性能。

警告:傳遞完整的模型給過濾器回調是很耗費性能的,最好是傳遞單獨的屬性。

致謝

感謝 Spatie 和 spatie/laravel-permissions 的貢獻者建立如此優秀的包,我很是喜歡使用!感謝 Andru Beldie 指出這些性能問題,我纔有機會對其進行調查和糾正。

連接

更多現代化 PHP 知識,請前往 Laravel / PHP 知識社區

相關文章
相關標籤/搜索