- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我正在开发一种解决方案,当服务器(MVC Core 网络应用程序)上发生某些事件时,使用网络套接字协议(protocol)通知客户端(网络浏览器)。我使用 Microsoft.AspNetCore.WebSockets nuget。
这是我的客户端代码:
$(function () {
var socket = new WebSocket("ws://localhost:61019/data/openSocket");
socket.onopen = function () {
$(".socket-status").css("color", "green");
}
socket.onmessage = function (message) {
$("body").append(document.createTextNode(message.data));
}
socket.onclose = function () {
$(".socket-status").css("color", "red");
}
});
加载此 View 时,套接字请求会立即发送到 MVC 核心应用程序。这是 Controller 操作:
[Route("data")]
public class DataController : Controller
{
[Route("openSocket")]
[HttpGet]
public ActionResult OpenSocket()
{
if (HttpContext.WebSockets.IsWebSocketRequest)
{
WebSocket socket = HttpContext.WebSockets.AcceptWebSocketAsync().Result;
if (socket != null && socket.State == WebSocketState.Open)
{
while (!HttpContext.RequestAborted.IsCancellationRequested)
{
var response = string.Format("Hello! Time {0}", System.DateTime.Now.ToString());
var bytes = System.Text.Encoding.UTF8.GetBytes(response);
Task.Run(() => socket.SendAsync(new System.ArraySegment<byte>(bytes),
WebSocketMessageType.Text, true, CancellationToken.None));
Thread.Sleep(3000);
}
}
}
return new StatusCodeResult(101);
}
}
此代码运行良好。这里的 WebSocket 专门用于发送,不接收任何东西。然而,问题是 while 循环一直保持 DataController 线程直到检测到取消请求。
这里的Web socket是绑定(bind)到HttpContext对象上的。一旦 Web 请求的 HttpContext 被销毁,套接字连接就会立即关闭。
问题 1:有什么方法可以在 Controller 线程之外保留套接字?我尝试将它放入一个单例中,该单例位于在主应用程序线程上运行的 MVC Core Startup 类中。有什么方法可以让套接字保持打开状态或从主应用程序线程中再次建立连接,而不是用 while 循环保持 Controller 线程?即使保留套接字连接的 Controller 线程保持打开状态被认为是可以的,我也想不出有什么好的代码可以放在 OpenSocket 的 while 循环中。您如何看待 Controller 中的手动重置事件并等待它在 OpenSocket 操作的 while 循环内设置?
问题2:如果在MVC中不能分离HttpContext和WebSocket对象,还有什么替代技术或开发模式可以实现socket连接重用?如果有人认为 SignalR 或类似的库有一些代码允许套接字独立于 HttpContext,请分享一些示例代码。如果有人认为对于这种特定情况有更好的 MVC 替代方案,请提供示例,如果 MVC 没有处理独立套接字通信的能力,我不介意切换到纯 ASP.NET 或 Web API。
问题 3:要求是保持套接字连接有效或能够重新连接,直到明确超时或用户取消请求。这个想法是一些独立的事件发生在服务器上,触发已建立的套接字发送数据。如果您认为 Web 套接字以外的某些技术对这种情况(例如 HTML/2 或流式传输)更有用,能否请您描述一下您将使用的模式和框架?
附言可能的解决方案是每秒发送一次 AJAX 请求,询问服务器上是否有新数据。这是最后的手段。
最佳答案
经过长时间的研究,我最终选择了自定义中间件解决方案。这是我的中间件类:
public class SocketMiddleware
{
private static ConcurrentDictionary<string, SocketMiddleware> _activeConnections = new ConcurrentDictionary<string, SocketMiddleware>();
private string _packet;
private ManualResetEvent _send = new ManualResetEvent(false);
private ManualResetEvent _exit = new ManualResetEvent(false);
private readonly RequestDelegate _next;
public SocketMiddleware(RequestDelegate next)
{
_next = next;
}
public void Send(string data)
{
_packet = data;
_send.Set();
}
public async Task Invoke(HttpContext context)
{
if (context.WebSockets.IsWebSocketRequest)
{
string connectionName = context.Request.Query["connectionName"]);
if (!_activeConnections.Any(ac => ac.Key == connectionName))
{
WebSocket socket = await context.WebSockets.AcceptWebSocketAsync();
if (socket == null || socket.State != WebSocketState.Open)
{
await _next.Invoke(context);
return;
}
Thread sender = new Thread(() => StartSending(socket));
sender.Start();
if (!_activeConnections.TryAdd(connectionName, this))
{
_exit.Set();
await _next.Invoke(context);
return;
}
while (true)
{
WebSocketReceiveResult result = socket.ReceiveAsync(new ArraySegment<byte>(new byte[1]), CancellationToken.None).Result;
if (result.CloseStatus.HasValue)
{
_exit.Set();
break;
}
}
SocketHandler dummy;
_activeConnections.TryRemove(key, out dummy);
}
}
await _next.Invoke(context);
string data = context.Items["Data"] as string;
if (!string.IsNullOrEmpty(data))
{
string name = context.Items["ConnectionName"] as string;
SocketMiddleware connection = _activeConnections.Where(ac => ac.Key == name)?.Single().Value;
if (connection != null)
{
connection.Send(data);
}
}
}
private void StartSending(WebSocket socket)
{
WaitHandle[] events = new WaitHandle[] { _send, _exit };
while (true)
{
if (WaitHandle.WaitAny(events) == 1)
{
break;
}
if (!string.IsNullOrEmpty(_packet))
{
SendPacket(socket, _packet);
}
_send.Reset();
}
}
private void SendPacket(WebSocket socket, string packet)
{
byte[] buffer = Encoding.UTF8.GetBytes(packet);
ArraySegment<byte> segment = new ArraySegment<byte>(buffer);
Task.Run(() => socket.SendAsync(segment, WebSocketMessageType.Text, true, CancellationToken.None));
}
}
这个中间件将在每个请求上运行。调用 Invoke 时,它会检查它是否是 Web 套接字请求。如果是,中间件检查这样的连接是否已经打开,如果没有,则接受握手并将其添加到连接字典中。字典是静态的很重要,这样它在应用程序生命周期内只创建一次。
现在如果我们停在这里并向上移动管道,HttpContext 最终将被销毁,并且由于套接字未正确封装,它也将被关闭。所以我们必须保持中间件线程运行。这是通过要求套接字接收一些数据来完成的。
你可能会问,如果需求只是发送,为什么还要接收?答案是它是可靠检测客户端断开连接的唯一方法。 HttpContext.RequestAborted.IsCancellationRequested 仅当您在 while 循环中不断发送时才有效。如果您需要在 WaitHandle 上等待某些服务器事件,则取消标志永远不会为真。我试图等待 HttpContext.RequestAborted.WaitHandle 作为我的退出事件,但它也从未设置过。所以我们要求套接字接收一些东西,如果那东西将 CloseStatus.HasValue 设置为 true,我们就知道客户端已断开连接。如果我们收到其他东西(客户端代码不安全),我们将忽略它并重新开始接收。
发送是在一个单独的线程中完成的。原因是一样的,如果我们在主中间件线程上等待,就不可能检测到断开连接。为了通知发送方线程客户端断开连接,我们使用 _exit 同步变量。请记住,这里有私有(private)成员很好,因为 SocketMiddleware 实例保存在静态容器中。
现在,我们如何使用此设置实际发送任何内容?假设服务器上发生了一个事件,一些数据变得可用。为简单起见,我们假设此数据到达某个 Controller 操作的正常 http 请求中。 SocketMiddleware 将为每个请求运行,但由于它不是 web 套接字请求,_next.Invoke(context) 被调用并且请求到达 Controller 操作,它可能看起来像这样:
[Route("ProvideData")]
[HttpGet]
public ActionResult ProvideData(string data, string connectionName)
{
if (!string.IsNullOrEmpty(data) && !string.IsNullOrEmpty(connectionName))
{
HttpContext.Items.Add("ConnectionName", connectionName);
HttpContext.Items.Add("Data", data);
}
return Ok();
}
Controller 填充用于在组件之间共享数据的项目集合。然后管道再次返回到 SocketMiddleware,我们检查 context.Items 中是否有任何有趣的东西。如果存在,我们从字典中选择相应的连接并调用其 Send() 方法,该方法设置数据字符串并设置 _send 事件并允许在发送方线程内单次运行 while 循环。
瞧,我们有发送服务器端事件的套接字连接。这个例子非常原始,只是为了说明这个概念。当然,要使用此中间件,您需要在添加 MVC 之前在 Startup 类中添加以下行:
app.UseWebSockets();
app.UseMiddleware<SocketMiddleware>();
代码非常奇怪,希望当 dotnetcore 的 SignalR 最终出来时,我们能够写出更好的东西。希望这个例子对某人有用。欢迎提出意见和建议。
关于c# - MVC 核心、Web 套接字和线程,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44511167/
我有一个我想暂时存储的对象。该对象现在在 Controller 中, Controller 将生成一个 View 。 AJAX 请求从 View 发送到下一个 Controller 。那一刻我需要先前
从MVC 2开始,我们可以轻松创建区域。现在,我的问题与嵌套区域(区域内部的区域)有关。 选择我的“father”区域文件夹,右键单击> Add> NO选项以获取new Area。 是否有可能以其他方
关闭。这个问题不满足Stack Overflow guidelines .它目前不接受答案。 想改善这个问题吗?更新问题,使其成为 on-topic对于堆栈溢出。 7年前关闭。 Improve thi
我已经尝试了一些谷歌搜索和堆栈流搜索,但事实证明这比我想象的要难找到。我需要为我们的商店迁移到 ASP.NET MVC 2 的管理提供理由。最大的帮助将是任何企业级站点或使用 ASP.NET MVC
关闭。这个问题是opinion-based 。目前不接受答案。 想要改进这个问题吗?更新问题,以便 editing this post 可以用事实和引文来回答它。 . 已关闭 6 年前。 Improv
我有一些常见的网页,它们将出现在多个 MVC 应用程序中。对于这些页面,我想在不同的 MVC 网站之间重用相同的源代码( Controller + View )。这样做的最佳方法是什么? ASP.NE
我正在使用 Spring MVC 来构建我的应用程序。 当用户在浏览器中运行应用程序时,我想显示一个默认的 jsp。我不想用 web.xml 中的标记。 我想我可以用 我已经创建了一个文件夹并添
我可能在这里分析过度了,但是根据我对 MVC 的阅读,似乎有很多关于如何做事情的观点。 是否有一个“最佳实践”网站或文档来定义 MVC 各个部分的职责? 请记住,我使用 EF/Repository&U
当杰里米和查德 posted about their FubuMvc project ,他们提到的差异化因素之一是他们的“雷霆穹顶校长”: The “Thunderdome Principle” –
我正在为 Spring MVC 应用程序实现缓存清除系统。 为了让这个系统正常工作,我必须从给定的 url 中删除“缓存破坏代码”。假设我生成的缓存破坏代码是“123”,我有一个 .css url:/
在调试 ASP.NET MVC 源时,我发现使用了“MVC-ControllerTypeCache.xml” 文件。但我无法理解这个文件的用途。我的意思是这个文件存储在哪里?asp.net MVc 如
我刚刚在我的本地机器上安装了 Visual Studio 11 和 MVC 4 beta。但是,每当我打开一个 MVC 3 项目(我想保留为 MVC 3)时,所有引用都已更新为版本 4 DLL。当然它
我有一个 MVC 3 应用程序,它具有一些核心功能(最重要的是自动化),但主要用作不同区域或模块的门户。我想将它组织到不同的模块中,只需稍作更改也可以部署为他们自己的网站。 该项目由论坛、博客引擎、用
我有自己的服务器,正在考虑将我的一个解决方案升级到 ASP.NET MVC 4,然后再升级其余的 (3+)。 作为其中的一部分,我下载了 the standalone installer对于 ASP.
构图 我有一个 MVC 项目,其中包含 C# 类,这些类最终通过 ajax 等进行序列化和使用。我使用 TypeLite 生成这些 C# 类的定义( here 讨论了 TypeLite 的替代方案),
我正在尝试了解现代 Web 应用程序架构。在 ASP.NET MVC 中,所有业务逻辑类都在 Model 中,Controller 接受并引导用户请求。如果我使用它,是否可以使用本身是 MVC 架构但
我有一个带有 OWIN 的 WebAPI 2 应用程序。现在我正在尝试向所有内容添加一个 MVC 5 Controller ,但没有找到我的 MVC 路由。我收到以下错误: No HTTP resou
在 MVC 3 中,他们添加了我一直在使用的依赖解析器。在回答某人对您发表评论的问题时,您应该使用 Ninject MVC 3 插件。 所以我的问题是为什么要使用它而不是内置的?如果这是要走的路,你如
我是 ASP.NET MVC 的新手,我正在寻找最不痛苦的方法来设置全局错误处理、日志记录和报告(通过电子邮件)。仅供引用,我的 ASP.NET MVC 应用程序在 Azure 中作为 Web 角色托
何时使用 MVC View 页面和 MVC View 内容页面?它们有什么区别? 最佳答案 **MVC View Page 用于创建页面,MVC VewP Content Page 用于创建页面并指定
我是一名优秀的程序员,十分优秀!