AJAX / Comet Chat Tutorial

Diese Seite verwendet Cookies. Durch die Nutzung unserer Seite erklären Sie sich damit einverstanden, dass wir Cookies setzen. Weitere Informationen

  • In diesem Tutorial wird erläutert wie die Techniker hinter einem professionell PHP + MySQL realisiert ist.
    Dabei werden unter anderem die Techniken "AJAX" und "Comet" genutzt, was den Webserver sehr entlasten wird, weil es das s.g. Polling umgeht.
    == Server zu Client Kommunikation ==
    Alles was über das HTTP Protokoll funktioniert geht immer vom Client aus. Der Client fragt etwas an, der Server antwortet. Anders funktioniert es nicht. Man kann nur versuchen einer Server zu Client Kommunikation möglichst nahe zu kommen.

    == AJAX ==
    Wer bereits etwas Erfahrung mit AJAX hat, der würde einen Chat vermutlich mit einem setInterval programmieren, der kontinuierlich in Abständen von 2-3 Sekunden beim Server nachfragt ob es Änderungen gibt.

    Wenn der Client den Server kontinuierlich nach Informationen abfragt, dann nennt man das Pollen. Es ist eine Art Ping Pong und die Technik ist sehr einfach zu implementeren. Beispiele fürs Polling findet ihr hier: [wiki]Mehrere DIV Container mit AJAX aktualisieren[/wiki]

    == Problem Polling ==
    Das Problem am Polling ist, dass bei 100 gleichzeitigen Besuchern jede Sekunde Sekunde 100 Anfragen gemacht werden müssen, auch wenn gar nichts passiert.
    Das erhöht die Last für den Server und auch beim Client vergeht unnötig Latenzzeit für die ganzen erzeugten Verbindungen.

    == Comet ==
    Für Comet gibt es keine Voraussetzungen, es ist keine Erweiterung oder eine spezielle Programmiersprache. Comet ist das Vorgehen, das es dem Server erlaubt Stück für Stück Informationen an den Client freizugeben währen eine einzige HTTP Verbindung offen gehalten wird.

    Der Browser fragt also den Client: "gib mir alle Chat Nachrichten".
    Und der Server antwortet: "Hier hast diese..." und nach gewisser Zeit "Hier hast du jene ...".
    Er lässt sich mit der Antwort also Zeit und pumpt nach und nach mehr Nachrichten zurück an den Client, die sogar viel neuer sein können, als der Zeitpunkt zu dem der Client zum ersten mal angefragt hat.

    === Beispiel ===

    Quellcode

    1. <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.1//EN" "http://www.w3.org/TR/xhtml11/DTD/xhtml11.dtd">
    2. <html xmlns="http://www.w3.org/1999/xhtml">
    3. <head>
    4. <title>Comet example 1</title>
    5. <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
    6. <script type="text/javascript">
    7. function chat(txt) {
    8. document.getElementById("chat").innerHTML += txt;
    9. }
    10. </script>
    11. </head>
    12. <body>
    13. <div id="chat"></div>
    14. <?php
    15. for($i=0; $i<10; $i++) {
    16. echo '<script type="text/javascript">';
    17. echo 'chat("<p>server is still alive at '.date('Y-m-d H:i:s').'</p>");';
    18. echo '</script>';
    19. // sende die Ausgabe zum Browser
    20. flush();
    21. // warte 0.2 Sekunden um den Server zu entlasten
    22. usleep(200000);
    23. }
    24. ?>
    25. <script type="text/javascript">
    26. chat("<p>server is done</p>");
    27. </script>
    28. </body>
    29. </html>
    Alles anzeigen


    Erweiterung mit Datenbankabfrage
    In Zeile #18 könnt ihr nun eure Datenbankabfrage einbauen, die immer wieder prüft ob neue Nachrichten da sind. Wenn keine neuen Nachrichten vorhanden sind, müsst ihr natürlich auch keine Nachricht senden.

    === Demo ===
    Ihr findet die Online Demo unter: demo.easy-coding.de/ajax/comet-chat-tutorial/example1.php

    === Einschränkungen #1 ===
    Das Beispielssript beendet sich nach 10 Schleifendurchläufen. Ihr könntet es aber mit einer while Schleife unendlich laufen lassen, dann würdet ihr irgendwann einen Fehler bekommen, dass die "max execution time" - die maximale Ausführzeit für ein Script überschritten wurde.

    Außerdem habt ihr potentiell immer noch 100 offene Verbindungen während einer Sekunde. Die Anzahl an Verbindungsaufbauten habt ihr aber schon deutlich reduziert.

    == AJAX & Comet ==
    Ihr könnt AJAX und Comet zusammen benutzen. Das wird so ablaufen, dass der Client nach neuen Nachrichten fragt und der Server immer antwortet, wenn eine neue Nachricht verfügbar ist. Danach schließt er die Verbindung und ein neuer AJAX Request wird gestartet.

    Für die AJAX Funktionalität hängen wir die ajax.js ein, die wir hier schon in mehreren Beispielen verwendet haben.

    Der Klasse die wir gleich erstellen müssen wir einen DIV Container und eine AJAX Url geben. Außerdem geben wir den Default Zeitstempel an. mit dem wir beginnen.

    Quellcode

    1. var chat = new Chat('chat', 'example2-backend.php', new Date().getTime());


    Die eigentlich update Funktion muss also, immer wenn sie eine Antwort bekommt (es also neue Nachrichten gibt), einen neuen Request senden. Immer wenn es zu Fehlern kommt, geben wir dem Script 5 Sekunden Aufschub und starten den Request dann erneut.

    Die gesamte Klasse und Client Logik sieht damit wie folgt aus:

    Quellcode

    1. function Chat(div, url, lastupdate) {
    2. this.div = div;
    3. this.url = url;
    4. this.lastupdate = lastupdate || null;
    5. this.start = function(lastupdate) {
    6. this.lastupdate = lastupdate || this.lastupdate;
    7. ajaxPost(this.url, 'lastupdate='+ this.lastupdate, function(up) {
    8. return function() {
    9. if (this.readyState == 4) {
    10. // success
    11. if(this.status == 200) {
    12. var data = eval('(' + this.responseText + ')');
    13. // start next timer, if response is empty, reuse the last known update index
    14. up.start(data.lastupdate || up.lastupdate);
    15. // fill chat
    16. if(data.html) {
    17. document.getElementById(up.div).innerHTML += data.html;
    18. }
    19. }
    20. // connection error
    21. else {
    22. // try again in 5 seconds
    23. setTimeout(function(){
    24. up.start(up.lastupdate);
    25. }, 5000);
    26. }
    27. }
    28. };
    29. }(this));
    30. };
    31. }
    Alles anzeigen


    Serverseitig in der example2-backend.php werden wir nur Antworten wenn wir eine Antwort haben. Wir werden dann HTML Code und den neuen Zeitstempel zurückgeben:

    Quellcode

    1. <?php
    2. function getNewMessagesSince($timestamp) {
    3. if(rand(1,20) == 5) {
    4. return array(
    5. 'lastupdate' => time(),
    6. 'html' => '<p>server is still alive at '.date('Y-m-d H:i:s').'</p>'
    7. );
    8. }
    9. return false;
    10. }
    11. $timeout = time() + 10;
    12. while(!($row = getNewMessagesSince($_POST['lastupdate'])) && time() < $timeout) {
    13. // warte 0.2 Sekunden um den Server zu entlasten
    14. usleep(200000);
    15. }
    16. // liefere den Inhalt
    17. echo json_encode($row);
    Alles anzeigen


    === Demo ===
    Ihr findet die Online Demo unter: demo.easy-coding.de/ajax/comet-chat-tutorial/example2.php

    == Fazit ==
    Das zweite Script antwortet immer, wenn es eine neue Antwort gibt. Bei einem häufig genutzten Chat wäre aber Beispiel eins besser, weil man hier mehrere Nachrichten über eine Verbindung schicken kann.
    Auch denkbar wäre den readyState 3 als Stream zu benutzen. Allerdings kann der Internet Explorer dies nicht. Mit einer JavaScript Bibliothek, die für die unterstützen Browser readyState 3 nutzt, und für die Internet Explorers dieser Welt auf eine Iframe/Ajax Kombination zurück greift, wäre dem AJAX/Comet Chat nichts mehr im Wege :)

    Die beste Lösung ist also die Kombination aller Verfahren. Ich gebe an dieser Stelle keine Referenzimplementierung vor. Ihr solltet mit den hier erworbenen Kenntnissen aber loslegen können um die perfekte Lösung zu implementieren.

    Den Download aller Scripte findet ihr hier: demo.easy-coding.de/ajax/comet-chat-tutorial/download.zip

    == Weiterführende Wikis ==
    • [wiki]Comet Chat Beispiel mit PHP + MySQL[/wiki]

    16.362 mal gelesen