如何实现.Net MVC长轮询功能

来源:AI视频音频作者:永濑头衔:网络博主
导读:本期聚焦于小伙伴创作的《如何实现.Net MVC长轮询功能》,敬请观看详情,探索知识的价值。以下视频、文章将为您系统阐述其核心内容与价值。如果您觉得《如何实现.Net MVC长轮询功能》有用,将其分享出去将是对创作者最好的鼓励。

在.Net MVC开发中,长轮询是实现服务端主动向客户端推送实时数据的常用方案,它基于HTTP协议,通过客户端发起请求后服务端 hold 住连接,直到有数据更新或超时再返回响应的机制,实现类实时的通信效果,相比WebSocket兼容性更好,适合不需要高频双向通信的场景。

如何实现.Net MVC长轮询功能

长轮询的核心实现原理

长轮询的基本流程分为三步:

  • 客户端向服务端发起HTTP请求,设置较长的超时时间
  • 服务端接收到请求后,检查是否有待推送的数据,若暂时没有则挂起当前请求,等待数据更新或超时
  • 当服务端有数据更新时,立即返回响应给客户端,客户端处理完响应后立即再次发起新的请求,形成循环

.Net MVC服务端实现步骤

1. 定义异步控制器方法

长轮询需要服务端能够挂起请求,因此需要使用异步控制器方法,避免占用线程池资源。首先在控制器中定义异步的轮询接口:

using System.Threading.Tasks;
using System.Web.Mvc;

namespace LongPollingDemo.Controllers
{
    public class MessageController : AsyncController
    {
        // 模拟消息存储,实际项目中可替换为数据库、缓存等存储
        private static string _latestMessage = "";
        private static object _lockObj = new object();

        // 异步轮询入口方法
        public void PollAsync(string lastMessageId)
        {
            // 开启异步操作
            AsyncManager.OutstandingOperations.Increment();
            
            // 模拟检查是否有新消息,实际项目中可监听数据变更事件
            Task.Run(() =>
            {
                // 最多等待30秒
                int waitTime = 0;
                while (waitTime < 30)
                {
                    lock (_lockObj)
                    {
                        // 若消息有更新则直接返回
                        if (!string.IsNullOrEmpty(_latestMessage))
                        {
                            AsyncManager.Parameters["message"] = _latestMessage;
                            AsyncManager.OutstandingOperations.Decrement();
                            return;
                        }
                    }
                    // 每隔1秒检查一次
                    System.Threading.Thread.Sleep(1000);
                    waitTime++;
                }
                // 超时无新消息,返回空
                AsyncManager.Parameters["message"] = "";
                AsyncManager.OutstandingOperations.Decrement();
            });
        }

        // 异步操作完成后的回调方法
        public ActionResult PollCompleted(string message)
        {
            return Json(new { hasNew = !string.IsNullOrEmpty(message), data = message }, JsonRequestBehavior.AllowGet);
        }

        // 模拟服务端更新消息的接口
        [HttpPost]
        public ActionResult UpdateMessage(string newMessage)
        {
            lock (_lockObj)
            {
                _latestMessage = newMessage;
            }
            return Json(new { success = true });
        }
    }
}

2. 配置异步请求超时时间

默认的HTTP请求超时时间较短,需要修改Web.config中的请求执行超时配置,确保长轮询的请求不会被提前中断:

<configuration>
  <system.web>
    <!-- 设置执行超时时间为60秒,适配长轮询的等待需求 -->
    <httpRuntime targetFramework="4.8" executionTimeout="60" />
  </system.web>
</configuration>

客户端实现逻辑

客户端使用JavaScript发起轮询请求,处理响应后循环发起新请求:

// 记录最后获取到的消息ID,实际项目中可用来做增量校验
let lastMessageId = "";

// 轮询函数
function startPolling() {
    fetch(`/Message/PollAsync?lastMessageId=${lastMessageId}`)
        .then(response => response.json())
        .then(data => {
            if (data.hasNew) {
                // 处理新消息
                console.log("收到新消息:", data.data);
                // 更新最后消息ID
                lastMessageId = new Date().getTime().toString();
            }
            // 无论是否有新消息,都立即发起下一次轮询
            startPolling();
        })
        .catch(error => {
            console.error("轮询请求失败:", error);
            // 失败后延迟1秒重试,避免频繁请求
            setTimeout(startPolling, 1000);
        });
}

// 页面加载后启动轮询
window.onload = function() {
    startPolling();
};

长轮询的注意事项

  • 服务端挂起请求时需要使用异步操作,避免占用IIS线程池资源,防止高并发时服务不可用
  • 客户端请求超时时间需要和与服务端等待时间匹配,避免客户端提前断开连接
  • 若项目需要更高性能的实时通信,可考虑使用SignalR框架,它内部会根据环境自动选择长轮询、WebSocket等最优传输方式
  • 长轮询不适合高频数据更新场景,高频场景下连接频繁断开重连会带来额外开销

方案对比

以下是长轮询和其他常见实时通信方案的对比:

方案兼容性实时性服务端开销
长轮询所有支持HTTP的浏览器中等中等
WebSocket现代浏览器,旧浏览器不支持
短轮询所有支持HTTP的浏览器

.Net_MVC长轮询异步请求SignalR修改时间:2026-06-16 22:36:30

免责声明:​ 已尽一切努力确保本网站所含信息的准确性。网站内容多为原创整理与精心编撰,观点力求客观中立。本站旨在免费分享,内容仅供个人学习、研究或参考使用。若引用了第三方作品,版权归原作者所有。如内容涉及您的权益,请联系我们处理。
内容垂直聚焦
专注技术核心技术栏目,确保每篇文章深度聚焦于实用技能。从代码技巧到架构设计,为用户提供无干扰的纯技术知识沉淀,精准满足专业提升需求。
知识结构清晰
覆盖从开发到部署的全链路。AI、前端、编程、数据库、服务器、建站、系统层层递进,构建清晰学习路径,帮助用户系统化掌握开发与运维所需的核心技术。
深度技术解析
拒绝泛泛而谈,深入技术细节与实践难点。无论是数据库优化还是服务器配置,均结合真实场景与代码示例进行剖析,致力于提供可直接应用于工作的解决方案。
专业领域覆盖
精准对应开发生命周期。从前端界面到后端编程,从数据库操作到服务器运维,形成完整闭环,一站式满足全栈工程师和运维人员的技术需求。
即学即用高效
内容强调实操性,步骤清晰、代码完整。用户可根据教程直接复现和应用于自身项目,显著缩短从学习到实践的距离,快速解决开发中的具体问题。
持续更新保障
专注既定技术方向进行长期、稳定的内容输出。确保各栏目技术文章持续更新迭代,紧跟主流技术发展趋势,为用户提供经久不衰的学习价值。