Heim  >  Artikel  >  Backend-Entwicklung  >  Warum hat PHP keine Timing-Funktion?

Warum hat PHP keine Timing-Funktion?

青灯夜游
青灯夜游Original
2023-02-24 09:59:122899Durchsuche

Der wesentliche Grund, warum PHP in der Webentwicklung keine Timer verwenden kann, ist das Fehlen einer kontrollierbaren Betriebsumgebung für den residenten Speicher: Der erste ist der residente Speicher und der zweite ist kontrollierbar. Im CGI-Modus wird der Prozess direkt nach der Ausführung des Skripts beendet und es kann nicht erwartet werden, dass er die Aufgabe zu einem bestimmten Zeitpunkt ausführt. Im PHP-FPM-Modus befindet sich der Prozess im Speicher, kann aber nicht gesteuert werden.

Warum hat PHP keine Timing-Funktion?

Die Betriebsumgebung dieses Tutorials: Windows 7-System, PHP 8-Version, DELL G3-Computer

Warum verfügt PHP nicht über eine Timing-Funktion?

Es gibt zwei gängige Timer: Einer ist die periodische Timing-Ausführung , wie z. B. jeden Tag am frühen Morgen Der Bericht wird um drei Uhr ausgegeben, der andere wird nach einer bestimmten Zeit (einmal) ausgeführt, beispielsweise werden tägliche Anmeldeprämien fünf Minuten nach der Anmeldung des Mitglieds beim System ausgegeben. Entspricht den Funktionen setInterval und setTimeout in JavaScript (streng genommen wird setInterval regelmäßig ausgeführt und die Ausführung zu einem bestimmten Zeitpunkt muss selbst erledigt werden).

PHP-Programmierer, die Webentwicklung betreiben, sollten mit den beiden Timer-Funktionen in JavaScript vertraut sein. Zurück zur PHP-Ebene sind sie etwas verblüfft:

In PHP gibt es Ruhe, aber keinen (eingebauten) Timer Funktion verfügbar. Die Schlaffunktion kann dies kaum tun, führt jedoch dazu, dass der Prozess blockiert wird und in diesem Zeitraum keine anderen Dinge ausgeführt werden können (oder nicht mehr reagiert). Warum stellt PHP die Timer-Funktion nicht zur Verfügung?

Gründe

Der wesentliche Grund, warum PHP in der Webentwicklung keine Timer verwenden kann, ist das Fehlen einer kontrollierbaren Betriebsumgebung mit residentem Speicher. Zwei wichtige Punkte: erstens residenter Speicher und zweitens steuerbar. Im CGI-Modus wird der Prozess direkt nach der Ausführung des Skripts beendet und es kann nicht erwartet werden, dass er die Aufgabe zu einem bestimmten Zeitpunkt ausführt. Im PHP-FPM-Modus befindet sich der Prozess (meistens) im Speicher, ist jedoch nicht kontrollierbar.

Unkontrollierbar bedeutet, dass der Prozess, der PHP ausführt, nicht vom PHP-Code beeinflusst wird und der Eintrittspunkt und der Austrittszeitpunkt des Prozesses durch zusätzliche Programme gesteuert werden. Im FPM-Modus unterbrechen beispielsweise die Exit- und Die-Funktionen im PHP-Skript nur die Ausführung des Skripts und haben keine besonderen Auswirkungen auf den Ausführungsprozess des Skripts (mit Ausnahme von Speicherlecks). Das von PHP-Entwicklern geschriebene Skript ist der Ausführungskörper des Prozesses. Nach der Ausführung wird es aus dem Ausführungskontext des Prozesses entladen. In diesem Fall wird der Zeitpunkt der Ausführung des PHP-Skripts immer noch von außen bestimmt. Wenn keine externe Anforderung vorliegt, liegt der PHP-Code still auf der Festplatte, ohne etwas zu tun, und es handelt sich um eine geplante Aufgabe.

Da PHP hauptsächlich auf die Webentwicklung ausgerichtet ist, ist der Ausführungsmodus von PHP stabil und zuverlässig und die Entwicklungseffizienz ist schnell. Durch das Weglassen des Ressourcenfreigabeschritts können beispielsweise viel Arbeitsaufwand und Fallstricke bei der Entwicklung vermieden werden. Denken Sie an einige Bibliothekscodes von Drittanbietern, die Zeitzonen, Zeichenkodierungen usw. ändern und diese nicht wiederherstellen. In einer Umgebung mit residentem Speicher führt dies mit ziemlicher Sicherheit zu Problemen bei nachfolgenden Anforderungen. Im FPM-Modus wird dieser Fallstrick jedoch unbeabsichtigt geglättet, was viel Debugging-Zeit spart und einen erheblichen Beitrag dazu leistet, dass Programmierer ihren Haaransatz behalten können.

Das Problem wurde verstanden. Wie kann man also einen Timer verwenden, um geplante Aufgaben in PHP auszuführen?

Gefährliche Praxis

In einer Webumgebung haben PHP-Skripte standardmäßig eine Zeitüberschreitung. Entfernen Sie die Timeout-Einstellung und Sie können das Programm im Hintergrund weiterlaufen lassen (wenn der Prozess nicht beendet wird). Beispielsweise läuft der folgende Code nach der Beantwortung der Anfrage im Hintergrund weiter und gibt die Uhrzeit alle fünf Sekunden in eine Datei aus:

# test.php
set_time_limit(0); # 取消超时设置,让脚本可一直运行

echo 'This is a background run forever script. Now you can leave me alone.';

fastcgi_finish_request();   # 结束当前请求

do{
   file_put_contents("/tmp/out.dat", "test script, now:" . date("Y-m-d H:i:s") . "\n", FILE_APPEND);
   sleep(5);
}while(true);

Nach der Anforderung des http://localhost:8080/test.php-Datei, überwachen Sie die Datei <code>/tmp/out.dat. Sie werden feststellen, dass der Inhalt kontinuierlich ausgegeben wird, unabhängig davon, ob der Client die Verbindung trennt, den Browser schließt oder den Computer neu startet (der Server kann dies nicht tun). neu gestartet). Dies zeigt, dass das Programm ausgeführt wurde und die gewünschte Timer-Funktion erreicht wurde. Wenn Sie sleep in usleep oder time_nanosleep ändern, können Sie auch Mikrosekunden- und Nanosekunden-Timer implementieren. Wäre das nicht schön? http://localhost:8080/test.php文件后,监测/tmp/out.dat文件,会发现不断有内容输出,无论客户端是否断开连接、关闭浏览器或者重启电脑(不能重启服务器)。这说明程序一直在执行,并且也实现了我们想要的定时器功能。如果把sleep改成usleeptime_nanosleep,还能实现微秒、纳秒级定时器,岂不美哉?

实践中应当尽量避免用这种方式实现定时器,不仅因为低效,还略有危险。原因之一是每次请求会占用一个进程,请求十万次需要十万个进程,基本上会导致系统崩溃或后续请求无响应;另外如果打开了session,但是忘记调用session_write_close

In der Praxis sollten Sie versuchen, die Implementierung von Timern auf diese Weise zu vermeiden, da dies nicht nur ineffizient, sondern auch leicht gefährlich ist. Einer der Gründe dafür ist, dass jede Anfrage einen Prozess belegt und einhunderttausend Anfragen einhunderttausend Prozesse erfordern, was im Grunde dazu führt, dass das System abstürzt oder nachfolgende Anfragen nicht mehr reagieren, wenn die Sitzung außer Ihnen geöffnet ist Vergessen Sie, session_write_close aufzurufen, was dazu führt, dass nachfolgende Anfragen desselben Benutzers hängen bleiben (die Sitzung befindet sich in einem gesperrten Zustand, wenn sie aktiv ist, und wenn die Sitzung nicht geschlossen wird, können nachfolgende Prozesse nicht ausgeführt werden um die Sitzung zu eröffnen).

Die Webentwicklung sollte so schnell wie möglich auf Benutzeranfragen reagieren. Die erzwungene Implementierung von Timern in der Webentwicklung führt dazu, dass die gesamte Webanwendung instabil, unzuverlässig oder unvorhersehbar wird. Menzius sagte: Wisse und handle umsichtig, ein Gentleman steht nicht vor einer gefährlichen Mauer. Unzuverlässige Praktiken sollten so weit wie möglich vermieden werden, und ganz nebenbei sollten auch Schuldzuweisungen und Schuldzuweisungen vermieden werden.

Als nächstes werfen wir einen Blick auf die richtige Haltung für die Verwendung von Timern in PHP.

Richtige Haltung

Die Methoden zur Implementierung von Timern in PHP lassen sich einfach wie folgt zusammenfassen:
  • Verwenden Sie Cron, Jenkins und andere Planungstools, um regelmäßig geplante Aufgaben auszuführen (entweder um Skripte auszuführen oder eine bestimmte URL anzufordern). ;
  • Einmalige Ausführungsaufgaben werden über Nachrichtenwarteschlangen, Datenbanken usw. an Programme von Drittanbietern übermittelt.
  • Simulieren Sie geplante Aufgaben wie WordPress, aber denken Sie daran, dass diese Methode auf Clientanforderungen und -bedürfnissen basiert um Probleme mit der Prozessparallelität selbst zu lösen;
  • Verwenden Sie den residenten Speichermodus, um PHP-Programme auszuführen, d. h. den CLI-Modus. 🎜

Mit Ausnahme der dritten Methode werden alle anderen Methoden empfohlen. Bitte berücksichtigen Sie den spezifischen Plan basierend auf den tatsächlichen Bedürfnissen. Als PHP-Programmierer ist die erste Wahl natürlich die Verwendung von PHP, also dem CLI-Modus.

CLI-Modus

Um ehrlich zu sein: Der CLI-Modus ermöglicht es PHP, seinen Speicherplatz erheblich zu erweitern. Im CLI-Modus ist der Einstiegspunkt des Programms das Skript, und der Code kann im Speicher liegen, und der Prozess wird vollständig vom PHP-Code gesteuert. In dieser Form gibt es viele Möglichkeiten, den Timer zu implementieren. In diesem Artikel werden mehrere Methoden aufgeführt, um andere zu inspirieren:

  • Verwenden Sie Frameworks wie swoole und workerman mit integrierten (hochpräzisen) Timern swooleworkerman等框架,内置(高精度)定时器;

  • 使用多进程(池)/多线程(池)技术(pcntlpthreads拓展在CLI模式下才可用);

  • 处理tick或者alarm等信号;

  • 使用libeventlibev等事件驱动库;

  • sleep加循环或自己实现事件循环。

想折腾的话自己用2-5方案,不想折腾swooleworkerman

Verwenden Sie Multiprozess- (Pool)/Multithreading- (Pool-)Technologie (pcntl, pthreads-Erweiterungen sind nur im CLI-Modus verfügbar). tick oder alarm;

Verwenden Sie ereignisgesteuerte Bibliotheken wie libevent und libev;


sleep, um Schleifen hinzuzufügen oder zu implementieren Ereignisschleifen selbst.

🎜🎜🎜Wenn Sie sich damit anlegen möchten, verwenden Sie selbst den 2-5-Plan. Wenn Sie sich nicht mit Frameworks wie swoole und workerman anlegen möchten, Sie sind die erste Wahl. Sie sind stabil und zuverlässig. 🎜🎜🎜Zusammenfassung🎜🎜🎜 Unterscheiden Sie die Beziehung zwischen HTTP-Anfragen und Aufgaben, und es wird einfacher, geplante Aufgaben zu implementieren. Ob PHP zur Implementierung verwendet werden soll, ist eine andere Sache. Natürlich kann PHP als bevorzugte Sprache für die Webentwicklung geplante Aufgaben problemlos implementieren. 🎜🎜Empfohlenes Lernen: „🎜PHP-Video-Tutorial🎜“🎜🎜

Das obige ist der detaillierte Inhalt vonWarum hat PHP keine Timing-Funktion?. 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