Requête asynchrone au serveur PHP sans renvoyer de données ? (Ci-joint la solution)

藏色散人
Libérer: 2023-04-10 21:34:02
avant
4863 Les gens l'ont consulté

J'ai récemment découvert un problème avec la requête asynchrone ajax, utilisez $.post, $.get, $.ajax code >Lors de la requête du serveur PHP, les données ne peuvent pas être renvoyées de manière asynchrone. <code>ajax异步请求的问题,用$.post$.get$.ajax请求PHP服务器时,总是无法异步返回数据。

经多次测试才发现:
– 不同浏览器,请求不同域名-不阻塞:无需实验
– 不同浏览器,请求同域名-不阻塞:session_id()返回不同
– 同一浏览器,请求不同域名-不阻塞:session_id返回不同
– 同一浏览器,请求同域名-阻塞:session_id()返回相同

发现问题所在:
1 关闭XDEBUG
2 SESSION
3 清除输出缓冲区

1 关闭XDEBUG
XDEBUG是实时调试。调试时,它将保持FPM以确保线程正在工作以避免数据污染。
典型的测试方法是,使用XDEBUG进行调试时,打开另一个浏览器并访问该站点,该站点这个时候是无法访问的。

这对并行响应有重大影响,即,即使前端发送多个请求,它也受XDEBUG控制,并且只能同时响应一个。
另外,由于XDEBUG依赖于SESSION,因此即使您使用session_write_close(),也要关闭会话锁(请参见下文)。XDEBUG仍会自动打开。

2 SESSION
session_write_close() 关闭SESSION的写锁,这适合SESSION保存为File的情况。如果SESSION保存在Redis,则不需要。

3 清除输出缓冲区
使用session_write_close()可能无法立即关闭SESSION锁,所以在这个方法之前加上:ob_end_flush()。让 session_write_close() 马上生效。

4 示例
有如下一个范例,当点击【提交】按钮时,前端会给后台服务器发送两种请求。
一种是get请求,每隔1秒请求一次。
一种是post

Après de nombreux tests, nous avons découvert :

– Différents navigateurs, demandant des noms de domaine différents - pas de blocage : pas besoin d'expérimenter

– Différents navigateurs, demandant le même nom de domaine - pas de blocage : session_id() renvoie différents

– le même navigateur, demande des noms de domaine différents - non bloquant : session_id renvoie différents

– le même navigateur, demande le même nom de domaine - bloquant : session_id() code>Renvoie la même choseProblème trouvé :1 Fermez XDEBUG2 Verrouillage SESSION
3 Effacer tampon de sortie

🎜 1 Désactivez XDEBUG🎜XDEBUG est un débogage en temps réel. Lors du débogage, il maintiendra FPM pour garantir que les threads fonctionnent pour éviter la pollution des données. 🎜La méthode de test typique consiste à ouvrir un autre navigateur et à accéder au site lors du débogage à l'aide de XDEBUG. Le site est inaccessible pour le moment. 🎜🎜Cela a un impact significatif sur les réponses parallèles, c'est-à-dire que même si le frontend envoie plusieurs requêtes, il est contrôlé par XDEBUG et ne peut répondre qu'à une seule à la fois. 🎜De plus, puisque XDEBUG dépend de SESSION, vous devez fermer le verrouillage de session même si vous utilisez session_write_close() (voir ci-dessous). XDEBUG s'ouvrira toujours automatiquement. 🎜🎜2 SESSIONLock🎜Utilisez session_write_close() pour fermer le verrou en écriture de SESSION, qui convient à SESSION Enregistrer sous Fichier. Non requis si SESSION est enregistré dans Redis. 🎜🎜3 Effacer le tampon de sortie🎜L'utilisation de session_write_close() peut ne pas fermer le verrou SESSION immédiatement, alors ajoutez : avant cette méthode >ob_end_flush(). Laissez session_write_close() prendre effet immédiatement. 🎜🎜4 Exemple🎜Il existe un exemple comme suit Lorsque vous cliquez sur le bouton [Soumettre], le front-end enverra deux requêtes au serveur backend. 🎜L'une est la requête get, qui est demandée toutes les secondes. 🎜L'une est la requête post, qui est envoyée une fois au début puis attend la fin correspondante. 🎜🎜Regardez le code HTML🎜
<form>
<input type="submit" value="提交" />
</form>

<script src="//cdn.bootcss.com/jquery/3.1.1/jquery.min.js"></script>

<script type="text/javascript">
    $(&#39;form&#39;).on(&#39;submit&#39;, function(e) {
        e.preventDefault();

        // 每隔一秒请求一次服务器
        var id = setInterval(function() {
            $.get(
                &#39;save.php?action=get&#39;,
                {},
                function(data) {
                    console.log(data);
                },
                &#39;json&#39;
            );
        }, 1000);

        $.post(
            &#39;save.php?action=post&#39;,
            {},
            function(data) {
                console.log(data);
                // 停止定时循环
                clearInterval(id);
            },
            &#39;json&#39;
        );
    });
</script>
Copier après la connexion
🎜code php🎜
<?php
session_start();$action = $_GET[&#39;action&#39;];
if ($action == &#39;post&#39;) {

   $_SESSION[&#39;time&#39;] = 0;
   session_write_close();

    while ($_SESSION[&#39;time&#39;] < 5) {
      session_start();
      $_SESSION[&#39;time&#39;] = $_SESSION[&#39;time&#39;] + 1;

      // 将SESSION数据写入文件中,并关闭写锁
      session_write_close();

      // sleep()模拟花费时间较长的程序,这样在关闭写锁之后,
      // 服务器就能够相应别的请求,如下的$action=get,
      sleep(1);
    }

    echo json_encode([session_id() => $_SESSION[&#39;time&#39;]]);
    exit();}if ($action == &#39;get&#39;) {
    echo json_encode([session_id() => $_SESSION[&#39;time&#39;]]);
    exit();}
Copier après la connexion
🎜🎜Apprentissage recommandé : "🎜Tutoriel vidéo PHP🎜"

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Étiquettes associées:
php
source:learnku.com
Déclaration de ce site Web
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn
Tutoriels populaires
Plus>
Derniers téléchargements
Plus>
effets Web
Code source du site Web
Matériel du site Web
Modèle frontal