我最近研究分析了在 SWIS上面建立的项目的性能。使人惊讶的是,最耗费性能的方法之一是优秀的 spatie/laravel-permission
包形成的。laravel
通过查阅更多资料和研究,发现一个可能明显改善的性能问题 。既然解决方案已明确阐述,就很容易编写代码改善,提交请求。git
如今这个解决方案已被合并和发布,下面是这个性能问题的分析和如何在本身的项目避免这类问题。github
TL;DR: 跳转到结论部分.数据库
若是咱们抽象的看 spatie/laravel-permission
它主要作两件事:缓存
第一点说是性能瓶颈有点牵强。这里的权限数据存放在数据库中,须要的时候将会被读取出来。这个过程是有点慢但也只是执行一次。结果会被缓存下来,后续的请求能够直接使用。性能
第二点在性能瓶颈的观点上来看确实是一个瓶颈。 这个瓶颈取决于权限的性质和项目的大小, 由于权限会被频繁的检查。 在这个检查的过程当中任何的迟钝都会成为整个项目的性能瓶颈。测试
过滤权限集合的方法被认为是形成低性能的缘由。 它作了以下事情: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 知识社区