suchen
HeimBackend-EntwicklungPHP-TutorialPHP-Verschlüsse und -Generatoren können Zirkelverweise enthalten

PHP Closures and Generators can hold circular references

Zirkelverweise in PHP sind eine häufige Ursache für Speicherlecks. Zirkelverweise treten auf, wenn Objekte direkt oder indirekt aufeinander verweisen. Glücklicherweise verfügt PHP über einen Garbage Collector, der Zirkelverweise erkennen und bereinigen kann. Dies verbraucht jedoch CPU-Zyklen und kann die Anwendung verlangsamen.

Der Garbage Collector wird ausgelöst, wenn 10.000 mögliche Schleifenobjekte oder Arrays im Speicher vorhanden sind und eines davon den Gültigkeitsbereich verlässt.

Wenn Sie eine kleine Anzahl von Objekten haben, die viel Speicher verbrauchen, wird die Speicherbereinigung nie ausgelöst. Möglicherweise erreichen Sie das Speicherlimit, selbst wenn der Speicher von verwaisten Objekten verwendet wird, die der Garbage Collector sammeln soll.

Deshalb sollten Sie Situationen identifizieren, die Zirkelbezüge erzeugen, und diese vermeiden.

Idealerweise möchten Sie für Webanwendungen den Garbage Collector deaktivieren und PHP nach dem Senden der Antwort den gesamten Speicher freigeben lassen. Dies ist jedoch für lang laufende Skripte wie Daemons oder Worker-Prozesse gefährlich, da sich mit der Zeit Speicherlecks ansammeln und die Anwendung durch häufige Aufrufe des Garbage Collectors verlangsamen können.

In diesem Artikel werden wir untersuchen, wie Schließungen und Generatoren Zirkelverweise speichern und wie man sie verhindert.

  • Über Zirkelbezüge
    • Typisches Beispiel für einen Zirkelverweis
    • Verwenden Sie schwache Referenzen, um Zirkelverweise zu verhindern
  • Abschlüsse und Zirkelverweise
  • Generatoren und Zirkelverweise
  • Fazit

Über Zirkelbezüge

Typisches Beispiel für einen Zirkelverweis

class A {
    public B $b;

    public function __construct()
    {
        $this->b = new B($this);
    }
}

class B {
    public function __construct(public A $a) {}
}

In diesem Beispiel beziehen sich A und B aufeinander. Wenn Sie eine Instanz von A erstellen, wird eine Instanz von B erstellt, die auf A verweist. Dadurch entsteht ein Zirkelverweis.

Um Zirkelverweise zu erkennen, können wir den Garbage Collector manuell mit gc_collect_cycles() auslösen und die Anzahl der gesammelten Verweise mit gc_status() auslesen.

// 创建的对象但未分配给变量
new A();

gc_collect_cycles();
print_r(gc_status());

Dies wird ausgegeben:

<code>Array
(
    ...
    [collected] => 2
    ...
)</code>

Dieses Beispiel zeigt, dass der Garbage Collector zwei Objekte mit Zirkelverweisen erkannt und gelöscht hat.

Sie können auch die Funktion xdebug_debug_zval() verwenden, um die Anzahl der Verweise auf ein Objekt anzuzeigen.

Verwenden Sie schwache Referenzen, um Zirkelverweise zu verhindern

Bei Zirkelbezügen besteht eine einfache Lösung darin, schwache Bezüge zu verwenden. Eine schwache Referenz ist ein Objekt, das eine Referenz enthält, die den Garbage Collector nicht daran hindert, das Objekt, auf das es verweist, zu sammeln. In PHP können Sie mit der Klasse WeakReference schwache Referenzen erstellen.

Dies erfordert einige Änderungen am Code. Klasse B speichert jetzt WeakReference-Objekte anstelle von A-Objekten. Sie müssen mit der WeakReference-Methode des get()-Objekts auf das A-Objekt zugreifen.

class A {
    public B $b;

    public function __construct()
    {
        $this->b = new B($this);
    }
}

class B {
    /** @var WeakReference<a> $a */
    public WeakReference $a;

    public function __construct(A $a)
    {
        $this->a = WeakReference::create($a);    
    }
}
// 创建的对象但未分配给变量
new A();

gc_collect_cycles();
print_r(gc_status());
// [collected] => 0

In der Ausgabe sehen Sie, dass die Anzahl der gesammelten Zitate jetzt 0 beträgt.

Tipp 1: Verwenden Sie schwache Referenzen nur bei Bedarf, um Zirkelverweise zu vermeiden.

Abschlüsse und Zirkelverweise

Das Konzept des Abschlusses in PHP besteht darin, eine Funktion zu erstellen, die auf Variablen im übergeordneten Bereich zugreifen kann. Dies kann zu Zirkelverweisen führen, wenn Sie nicht vorsichtig sind.

class A {
    public B $b;

    public function __construct()
    {
        $this->b = new B($this);
    }
}

class B {
    public function __construct(public A $a) {}
}

In diesem Beispiel bezieht sich der Abschluss $a->b auf eine Variable $a im übergeordneten Bereich. Zirkelbezüge sind leicht zu erkennen, da die Bezüge eindeutig sind.

Das gleiche Problem kann jedoch auf subtilere Weise auftreten, wenn Sie die Kurzsyntax von Abschlüssen verwenden. Bei Pfeilfunktionen wird die Variable $a im Abschluss nicht explizit referenziert, sie wird aber dennoch per Referenz erfasst.

// 创建的对象但未分配给变量
new A();

gc_collect_cycles();
print_r(gc_status());

In diesem Beispiel beträgt die Anzahl der gesammelten Referenzen 2, was auf einen Zirkelverweis hinweist.

Verweis auf $this im Abschluss

Jeder nicht statische Abschluss, der innerhalb einer Klassenmethode erstellt wird, hat einen Verweis auf die Objektinstanz ($this), auch wenn nicht auf $this zugegriffen wird.

<code>Array
(
    ...
    [collected] => 2
    ...
)</code>

Das liegt daran, dass $this Referenzen in Abschlüssen immer durch Referenz erfasst werden. Der Zugriff erfolgt über Reflection::getClosureThis().

class A {
    public B $b;

    public function __construct()
    {
        $this->b = new B($this);
    }
}

class B {
    /** @var WeakReference<a> $a */
    public WeakReference $a;

    public function __construct(A $a)
    {
        $this->a = WeakReference::create($a);    
    }
}

Wenn der Abschluss aus dem globalen Bereich oder einer statischen Methode erstellt wird, ist die $this-Referenz null.

Tipp 2: Wenn Sie $this nicht benötigen, verwenden Sie immer static function () {} oder static fn () =>, um einen Abschluss zu erstellen.

Generatoren und Zirkelverweise

Lassen Sie uns über den Grund für diesen Artikel sprechen. Ich habe kürzlich etwas entdeckt: Generatoren behalten ihre Referenzen, solange sie nicht erschöpft sind.

In diesem Beispiel speichert die Klasse den Generator in einer Eigenschaft, aber der Generator hat einen $this Verweis auf die Objektinstanz. Ein Generator verhält sich wie ein Abschluss und enthält eine Referenz auf die Objektinstanz.

// 创建的对象但未分配给变量
new A();

gc_collect_cycles();
print_r(gc_status());
// [collected] => 0

Die Klasseninstanz wird vom Garbage Collector gesammelt, da sie einen Verweis auf den Generator hat, der wiederum einen Verweis auf die Objektinstanz hat.

Sobald der Generator erschöpft ist, wird die Referenz freigegeben und die Objektinstanz aus dem Speicher entfernt.

function createCircularReference()
{
    $a = new stdClass();
    $a->b = function () use ($a) {
        return $a;
    };

    return $a;
}

Tipp 3: Erschöpfen Sie den Generator immer durch Iteration.

Tipp 4: Verwenden Sie statische Methoden oder Abschlüsse, um Generatoren zu erstellen, um die Beibehaltung von Verweisen auf Objektinstanzen zu vermeiden.

Fazit

Zirkelverweise sind eine häufige Ursache für Speicherlecks in PHP. Selbst wenn der Garbage Collector Zirkelverweise erkennen und bereinigen kann, verbraucht er CPU-Zyklen und kann die Anwendung verlangsamen. Sie müssen Situationen erkennen, die solche Zirkelverweise erzeugen, und Ihren Code anpassen, um dies zu verhindern. Die Verwendung schwacher Referenzen kann Referenzzyklen verhindern, aber einige einfache Tipps können Ihnen dabei helfen, sie von vornherein zu verhindern:

  • Wenn $this nicht erforderlich ist, verwenden Sie static function () {} oder static fn () =>, um einen Abschluss zu erstellen.
  • Erschöpfen Sie den Generator immer durch Iteration.
  • Verwenden Sie statische Methoden oder Abschlüsse, um Generatoren zu erstellen, um das Beibehalten von Verweisen auf Objektinstanzen zu vermeiden.

Weiterlesen

  • PHP Garbage Collection – Leistungsüberlegungen
  • Was ist Garbage Collection in PHP? Wie kann man das Beste daraus machen?
  • memprof – Speicheranalysator für PHP. Helfen Sie dabei, Speicherlecks in PHP-Skripten zu finden.
  • Der integrierte Analysator von Xdebug

Das obige ist der detaillierte Inhalt vonPHP-Verschlüsse und -Generatoren können Zirkelverweise enthalten. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Stellungnahme
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn
Wie ändern Sie Daten, die in einer PHP -Sitzung gespeichert sind?Wie ändern Sie Daten, die in einer PHP -Sitzung gespeichert sind?Apr 27, 2025 am 12:23 AM

TomodifyDatainaphpSession, startTheSessionwithSession_Start (), dann $ _SessionToSet, modify, orremovevariables.1) startTheSession.2) setOrmodifySessionvariabling $ _Session.3) removeVariables mit ()

Geben Sie ein Beispiel für die Speicherung eines Arrays in einer PHP -Sitzung.Geben Sie ein Beispiel für die Speicherung eines Arrays in einer PHP -Sitzung.Apr 27, 2025 am 12:20 AM

Arrays können in PHP -Sitzungen gespeichert werden. 1. Starten Sie die Sitzung und verwenden Sie Session_Start (). 2. Erstellen Sie ein Array und speichern Sie es in $ _Session. 3. Abrufen Sie das Array durch $ _Session ab. 4. Optimieren Sie Sitzungsdaten, um die Leistung zu verbessern.

Wie funktioniert die Müllsammlung für PHP -Sitzungen?Wie funktioniert die Müllsammlung für PHP -Sitzungen?Apr 27, 2025 am 12:19 AM

Die PHP -Sitzungsmüllsammlung wird durch einen Wahrscheinlichkeitsmechanismus ausgelöst, um abgelaufene Sitzungsdaten zu beseitigen. 1) Legen Sie die Auslöserwahrscheinlichkeit und die Sitzungslebenszyklus in der Konfigurationsdatei ein. 2) Sie können Cron-Aufgaben verwenden, um Hochlastanwendungen zu optimieren. 3) Sie müssen die Häufigkeit und Leistung von Müllsammlungen ausgleichen, um Datenverlust zu vermeiden.

Wie können Sie die Sitzungsaktivität in PHP verfolgen?Wie können Sie die Sitzungsaktivität in PHP verfolgen?Apr 27, 2025 am 12:10 AM

Die Verfolgung von Benutzersitzungsaktivitäten in PHP wird durch Sitzungsverwaltung implementiert. 1) Verwenden Sie Session_start (), um die Sitzung zu starten. 2) Speichern Sie Daten über das $ _Session -Array. 3) Call Session_Destroy (), um die Sitzung zu beenden. Die Sitzungsverfolgung wird für die Analyse der Benutzerverhalten, die Sicherheitsüberwachung und die Leistungsoptimierung verwendet.

Wie können Sie eine Datenbank verwenden, um PHP -Sitzungsdaten zu speichern?Wie können Sie eine Datenbank verwenden, um PHP -Sitzungsdaten zu speichern?Apr 27, 2025 am 12:02 AM

Die Verwendung von Datenbanken zum Speichern von PHP -Sitzungsdaten kann die Leistung und Skalierbarkeit verbessern. 1) Konfigurieren Sie MySQL, um Sitzungsdaten zu speichern: Richten Sie den Sitzungsprozessor in Php.ini oder PHP -Code ein. 2) Benutzerdefinierte Sitzungsprozessor implementieren: Definieren Sie Öffnung, Schließen, Lesen, Schreiben und andere Funktionen, um mit der Datenbank zu interagieren. 3) Optimierung und Best Practices: Verwenden Sie Indexierung, Zwischenspeicherung, Datenkomprimierung und verteilter Speicher, um die Leistung zu verbessern.

Erläutern Sie das Konzept einer PHP -Sitzung in einfachen Worten.Erläutern Sie das Konzept einer PHP -Sitzung in einfachen Worten.Apr 26, 2025 am 12:09 AM

PhpSessionStrackUserDataacrossMultiplePageRequestsusesuseiquiTIdStoredInacookie.her'ShowtomagetheFectiv: 1) StartaSessionswithSession_start () und storateatain $ _Session.2) regeneratethessionSessionInoginWithSession_IDENT_IDENTE_IDENTE_IDENTE_IDENTE_IDENTE_IDENTE_IDENTE_IDENTE_IDENTE_IDENTE_IDENTE_IDENTE_IDENTE_IDENTE_IDENTE_IDENTE_IDENTE_IDENTE_IDENTE_IDENTEL

Wie schleifen Sie alle in einer PHP -Sitzung gespeicherten Werte durch?Wie schleifen Sie alle in einer PHP -Sitzung gespeicherten Werte durch?Apr 26, 2025 am 12:06 AM

In PHP können durch Sitzungsdaten in den folgenden Schritten iteriert werden: 1. Starten Sie die Sitzung mit Session_Start (). 2. Iterieren Sie durch die Foreach-Schleife durch alle Schlüsselwertpaare im $ _Session-Array. 3. Wenn Sie komplexe Datenstrukturen verarbeiten, verwenden Sie is_array () oder is_object () Funktionen und verwenden Sie print_r (), um detaillierte Informationen auszugeben. 4. Bei der Optimierung von Traversal kann Paging verwendet werden, um eine gleichzeitige Verarbeitung großer Datenmengen zu vermeiden. Auf diese Weise können Sie PHP -Sitzungsdaten in Ihrem tatsächlichen Projekt effizienter verwalten und verwenden.

Erklären Sie, wie Sie Sitzungen für die Benutzerauthentifizierung verwenden.Erklären Sie, wie Sie Sitzungen für die Benutzerauthentifizierung verwenden.Apr 26, 2025 am 12:04 AM

Die Sitzung realisiert die Benutzerauthentifizierung über den serverseitigen Statusverwaltungsmechanismus. 1) Erstellung der Sitzung und Erzeugung eindeutiger IDs, 2) IDs werden durch Cookies weitergeleitet, 3) Server speichert und greift auf Sitzungsdaten über IDs, 4) Benutzerauthentifizierung und Statusverwaltung zugeordnet und verbessert die Sicherheit und die Benutzererfahrung von Anwendungen.

See all articles

Heiße KI -Werkzeuge

Undresser.AI Undress

Undresser.AI Undress

KI-gestützte App zum Erstellen realistischer Aktfotos

AI Clothes Remover

AI Clothes Remover

Online-KI-Tool zum Entfernen von Kleidung aus Fotos.

Undress AI Tool

Undress AI Tool

Ausziehbilder kostenlos

Clothoff.io

Clothoff.io

KI-Kleiderentferner

Video Face Swap

Video Face Swap

Tauschen Sie Gesichter in jedem Video mühelos mit unserem völlig kostenlosen KI-Gesichtstausch-Tool aus!

Heiße Werkzeuge

Herunterladen der Mac-Version des Atom-Editors

Herunterladen der Mac-Version des Atom-Editors

Der beliebteste Open-Source-Editor

MantisBT

MantisBT

Mantis ist ein einfach zu implementierendes webbasiertes Tool zur Fehlerverfolgung, das die Fehlerverfolgung von Produkten unterstützen soll. Es erfordert PHP, MySQL und einen Webserver. Schauen Sie sich unsere Demo- und Hosting-Services an.

DVWA

DVWA

Damn Vulnerable Web App (DVWA) ist eine PHP/MySQL-Webanwendung, die sehr anfällig ist. Seine Hauptziele bestehen darin, Sicherheitsexperten dabei zu helfen, ihre Fähigkeiten und Tools in einem rechtlichen Umfeld zu testen, Webentwicklern dabei zu helfen, den Prozess der Sicherung von Webanwendungen besser zu verstehen, und Lehrern/Schülern dabei zu helfen, in einer Unterrichtsumgebung Webanwendungen zu lehren/lernen Sicherheit. Das Ziel von DVWA besteht darin, einige der häufigsten Web-Schwachstellen über eine einfache und unkomplizierte Benutzeroberfläche mit unterschiedlichen Schwierigkeitsgraden zu üben. Bitte beachten Sie, dass diese Software

WebStorm-Mac-Version

WebStorm-Mac-Version

Nützliche JavaScript-Entwicklungstools

SAP NetWeaver Server-Adapter für Eclipse

SAP NetWeaver Server-Adapter für Eclipse

Integrieren Sie Eclipse mit dem SAP NetWeaver-Anwendungsserver.