Heim > Artikel > Backend-Entwicklung > Leistungsoptimierung für Laravel-Berechtigungsprojekt
Dieser Artikel stellt hauptsächlich die Leistungsoptimierung des Laravel-Berechtigungsprojekts vor, das einen gewissen Referenzwert hat. Jetzt kann ich es mit allen teilen, die es brauchen
Ich habe kürzlich die Leistung von auf SWIS erstellten Projekten recherchiert und analysiert. Überraschenderweise wird eine der leistungsintensivsten Methoden durch das hervorragende Paket spatie/laravel-permission
verursacht.
Nach Durchsicht weiterer Informationen und Recherchen habe ich ein Leistungsproblem festgestellt, das möglicherweise erheblich verbessert werden kann. Nachdem die Lösung nun klar dargelegt ist, ist es einfach, Verbesserungen zu programmieren und Pull-Anfragen einzureichen.
Nachdem diese Lösung zusammengeführt und veröffentlicht wurde, finden Sie hier eine Analyse dieses Leistungsproblems und wie Sie solche Probleme in Ihren eigenen Projekten vermeiden können.
TL;DR: Springe zum Schlussteil.
Wenn wir es abstrakt betrachten spatie/laravel-permission
bewirkt es hauptsächlich zwei Dinge:
Führen Sie eine Liste der Berechtigungen, die zu einem Modell gehören.
Überprüfen Sie, ob ein Modell über Berechtigungen verfügt.
Der erste Punkt ist etwas weit hergeholt zu sagen, dass es sich um einen Leistungsengpass handelt. Die Berechtigungsdaten werden hier in der Datenbank gespeichert und bei Bedarf gelesen. Dieser Vorgang ist etwas langsam, wird aber nur einmal durchgeführt. Die Ergebnisse werden zwischengespeichert und können direkt in nachfolgenden Anfragen verwendet werden.
Der zweite Punkt ist tatsächlich ein Engpass aus Sicht des Leistungsengpasses. Dieser Engpass hängt von der Art der Berechtigungen und der Größe des Projekts ab, da Berechtigungen häufig überprüft werden. Jede Trägheit während dieser Prüfung führt zu einem Leistungsengpass für das gesamte Projekt.
Die Methode zum Filtern von Berechtigungssammlungen wird als Ursache für geringe Leistung angesehen. Es bewirkt Folgendes:
$permission = $permissions ->where('id', $id) ->where('guard_name', $guardName) ->first();
Geändert:
$permission = $permissions ->filter(function ($permission) use ($id, $guardName) { return $permission->id === $id && $permission->guard_name === $guardName; }) ->first();
Diese beiden Codeausschnitte erreichen dasselbe, aber der zweite ist schneller.
In der App, die ich entwickle, gibt es etwa 150 verschiedene Berechtigungen. Bei einer normalen Anfrage müssen etwa 50 Berechtigungen mit der Methode hasPermissionTo
überprüft werden. Natürlich müssen auf einigen Seiten möglicherweise etwa 200 Berechtigungen überprüft werden.
Im Folgenden sind einige Einstellungen aufgeführt, die für Leistungstests verwendet werden.
$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(); }
Die oben genannten drei Methoden werden verwendet, um die Filterung von 1 Attribut, 2 Attributen und 3 Attributen zu testen. Daher sieht die Verwendung von Methode 1 zum Filtern von drei Attributen wie folgt aus:
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%) |
Wir können daraus schließen, dass bei einem Projekt das wiederholte Filtern einer großen Sammlung zu ernsthaften Leistungsengpässen führt.
Multi-Attribut-Filterung erhöht den Rechenaufwand erheblich.
Die Verwendung von Collection::filter()
anstelle von Collection::where()
kann die Leistung um 60 % verbessern.
Warnung: Die Übergabe des kompletten Modells an den Filter-Callback ist sehr leistungsintensiv, es ist besser, einzelne Attribute zu übergeben.
Das Obige ist der gesamte Inhalt dieses Artikels. Ich hoffe, dass er für das Studium aller hilfreich ist. Weitere verwandte Inhalte finden Sie auf der chinesischen PHP-Website.
Verwandte Empfehlungen:
Verwendung der Coroutine-Datenbankabfrage von Swoole in Laravel 5.6
Laravels Fassadendarstellungssystemanalyse
Das obige ist der detaillierte Inhalt vonLeistungsoptimierung für Laravel-Berechtigungsprojekt. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!