.NET框架之“小马过河”(小马过河国际教育咨询有限公司)
ccwgpt 2024-09-23 04:27 46 浏览 0 评论
.NET框架之“小马过河”
有许多流行的 .NET
框架,大家都觉得挺“重”,认为很麻烦,重量级,不如其它“轻量级”框架,从而不愿意使用。面对形形色色的框架发愁,笔者也曾发愁。但我发现只要敢于尝试,这些框架都是“纸老虎”。就像“小马过河”一样,自己尝试一下,就会发现“原来河水既不像老牛说的那样浅,也不像松鼠说的那样深。”
项目中的代码,都在 LINQPad6
中运行并测试通过,也可以复制到VisualStudio
中执行。
做简单的 Http
服务器很“重”
有些非常简单的 Http
服务器,我看到有些.NET
开发居然也用Node.js
、Python
等语言,一问,他们会回答说“这种简单的东西,用.NET
,太重了”。殊不知其实用.NET
做起来,也很轻(甚至更轻):
// 代码不需要引入任何第三方包
var http = new HttpListener;
http.Prefixes.Add("http://localhost:8080/");
http.Start;
while (true)
{
var ctx = await http.GetContext;
using var writer = new StreamWriter(ctx.Response.OutputStream);
writer.Write(DateTime.Now);
}
运行效果:
可见,包括空行,仅10行代码即可完成一个简单的 HTTP
服务器。
使用 EntityFramework
很“重”
EntityFramework
,简称EF
,现在有两个版本,EFCore
和EF6
,其中EFCore
可以同时运行在.NETFramework
和.NETCore
中,但EF6
只能在.NETFramework
中运行。本文中只测试了EFCore
,但EF6
代码也一样简单。
EntityFramework
是.NET
下常用的数据访问框架,以代码简单、功能强大而著名。但不少人却嗤之以鼻、不以为意。询问时,回答说EntityFramework
很“重”。
这个“重”字,我理解为它可能占用内存高,或者它可能代码极其麻烦,配置不方便(像iBatis
/Hibernate
那样),真的这样吗?
如图,假设我有一个 UserVoiceStatus
表:
下面,我们通过 EF
将数据取出来:
// 引用NuGet包:
// Microsoft.EntityFrameworkCore.SqlServer
void Main
{
var db = new MyDB(new DbContextOptionsBuilder
.UseSqlServer(Util.GetPassword("ConnectionString"))
.Options);
db.UserVoiceStatus.Dump;
}
public class UserVoiceStatus
{
public byte Id { get; set; }
public string Name { get; set; }
}
public class MyDB : DbContext
{
public MyDB(DbContextOptions options): base(options)
{
}
public DbSet<UserVoiceStatus> UserVoiceStatus { get; set; }
}
执行效果如图:
注意,如果使用
LINQPad
,事情还能更简单,只要一行代码即可,效果完全一样:UserVoiceStatuses
使用 ASP.NET MVC
很“重”
上文说到了如何做一个简单的 Http
服务器,如果想复杂一点,初始化ASP.NET MVC
也很简单,甚至只需要一个文件即可完成:
void Main
{
WebHost
.CreateDefaultBuilder
.UseStartup<UserQuery>
.UseUrls("https://localhost:55555")
.Build
.Run;
}
public void ConfigureServices(IServiceCollection services)
{
services.AddControllers;
}
public void Configure(IApplicationBuilder app)
{
app.UseRouting;
app.UseEndpoints(endpoints =>
{
endpoints.MapControllerRoute(
name: "default",
pattern: "{controller}/{action}/{id?}",
defaults: new { controller = "Home", action = "Index" });
});
}
namespace Controllers
{
public class HomeController : Controller
{
public DateTime Index
{
return DateTime.Now;
}
}
}
麻雀虽小,五脏俱全,这么简短的几千代码中,可以使用 Https
、包含了依赖注入,还能完整的路由功能,就构成了ASP.NET MVC
的基本代码。运行效果如图:
使用 WebSockets
很“重”
WebSockets
是个流行的Http
双向通信技术,以前在Node.js
中很流行(用socket.io
)。代码如下:
async Task Main
{
await WebHost
.CreateDefaultBuilder
.UseStartup<UserQuery>
.UseUrls("https://*:55555")
.Build
.RunAsync;
}
async Task Echo(HttpContext ctx, WebSocket webSocket, CancellationToken cancellationToken)
{
var buffer = new byte[4096];
ValueWebSocketReceiveResult result = await webSocket.ReceiveAsync(buffer.AsMemory, cancellationToken);
while (!result.EndOfMessage)
{
await webSocket.SendAsync(buffer.AsMemory(..result.Count), result.MessageType, result.EndOfMessage, cancellationToken);
result = await webSocket.ReceiveAsync(buffer.AsMemory, cancellationToken);
}
await webSocket.CloseAsync(WebSocketCloseStatus.NormalClosure, "NA", cancellationToken);
}
public void ConfigureServices(IServiceCollection services)
{
}
public void Configure(IApplicationBuilder app)
{
app.UseWebSockets;
app.Use(async (ctx, next) =>
{
if (ctx.Request.Path == "/ws")
{
if (ctx.WebSockets.IsWebSocketRequest)
{
WebSocket webSocket = await ctx.WebSockets.AcceptWebSocketAsync;
await Echo(ctx, webSocket, CancellationToken.None);
return;
}
}
await next;
});
app.Run(x => x.Response.WriteAsync("Please call /ws using WebSockets."));
}
该代码是个 Echo
服务器,它会将客户端发过来和内容,按原因返回给客户端。然后,.NET
也内置了WebSockets
的客户端:可以高效地访问刚刚创建并运行的WebSockets
服务器。
using (var ws = new ClientWebSocket)
{
await ws.ConnectAsync(new Uri("wss://localhost:55555/ws"), CancellationToken.None);
var completeEvent = new ManualResetEventSlim;
var cts = new CancellationTokenSource;
new Task( => SendMessage(ws, cts)).Start;
var buffer = new byte[4096];
do
{
var r = await ws.ReceiveAsync(buffer, cts.Token);
$"[{Util.ElapsedTime}] Received {Encoding.UTF8.GetString(buffer, 0, r.Count)}".Dump;
} while (ws.State != WebSocketState.Closed);
}
$"[{Util.ElapsedTime}] Closed.".Dump;
async void SendMessage(WebSocket ws, CancellationTokenSource cts)
{
for (var i = 0; i <3; ++i)
{
await ws.SendAsync(
Encoding.UTF8.GetBytes($"[{Util.ElapsedTime}] Send {DateTime.Now.ToString}".Dump),
WebSocketMessageType.Text,
endOfMessage: false, default);
await Task.Delay(1000);
}
await ws.CloseAsync(WebSocketCloseStatus.Empty, , default);
cts.Cancel;
}
最后,客户端与服务器双向通信效果如下:
使用 SignalR
很“重”
SignalR
是ASP.NET
推出的抽象式的Http
协议双向通信框架。SignalR
可以用相同的API
,支持像长轮询、ServerSentEvents
和WebSocket
的技术。SignalR
默认优先选择使用WebSocket
以达到最高性能,如果客户端或服务器不支持,则会回退至其它稍慢的技术。
SignalR
客户端还支持几乎所有语言、所有平台。它是如此好用,几乎可以取代传统的请求/响应,成为新的Http
开发模型。(事实上Blazor
正在尝试这样做)
但 SignalR
最为令人震撼的,还是它非常简单的使用方式,而恰恰是这一点给人误会最深。它的服务端API
,甚至比WebSocket
还要简单清晰简单:
async Task Main
{
await WebHost
.CreateDefaultBuilder
.UseStartup<UserQuery>
.UseUrls("https://localhost:55555")
.Build
.RunAsync;
}
public void ConfigureServices(IServiceCollection services)
{
services.AddSignalR;
}
public void Configure(IApplicationBuilder app)
{
app.UseRouting;
app.UseEndpoints(endpoints =>
{
endpoints.MapHub<Hubs.ChatHub>("/chat");
});
}
namespace Hubs
{
public class ChatHub : Hub
{
public async Task Broadcast(string id, string text)
{
await Clients.All.SendAsync("Broadcast", id, text);
}
}
}
前文提到, SignalR
提供了所有平台的SignalR
客户端,如js
、Android
等,其中当然(显然)也包括.NET
的。SignalR
的.NET
客户端使用起来也非常简单:
// 引入NuGet包:Microsoft.AspNetCore.SignalR.Client
// 代码在LINQPad中运行
var hub = new HubConnectionBuilder
.WithUrl("https://localhost:55555/chat")
.Build;
hub.On("Broadcast", (string id, string msg) =>
{
Console.WriteLine($"{id}: {msg}");
});
new Label("姓名: ").Dump;
var idBox = new TextBox(Guid.NewGuid.ToString).Dump;
await hub.StartAsync;
while (true)
{
var text = Console.ReadLine;
if (text == "Q") break;
await hub.SendAsync("Broadcast", idBox.Text, text);
}
这是一个非常简单的多人聊天室,运行效果如下:
总结
面对形形色色的框架发愁,笔者也曾发愁。但现在不了,什么框架拿过来,马上试试,也就几十秒钟的事。好用不好用,用用便知。
那么读者,你的“小马过河”的故事是怎样的呢?
相关推荐
- 十分钟让你学会LNMP架构负载均衡(impala负载均衡)
-
业务架构、应用架构、数据架构和技术架构一、几个基本概念1、pv值pv值(pageviews):页面的浏览量概念:一个网站的所有页面,在一天内,被浏览的总次数。(大型网站通常是上千万的级别)2、u...
- AGV仓储机器人调度系统架构(agv物流机器人)
-
系统架构层次划分采用分层模块化设计,分为以下五层:1.1用户接口层功能:提供人机交互界面(Web/桌面端),支持任务下发、实时监控、数据可视化和报警管理。模块:任务管理面板:接收订单(如拣货、...
- 远程热部署在美团的落地实践(远程热点是什么意思)
-
Sonic是美团内部研发设计的一款用于热部署的IDEA插件,本文其实现原理及落地的一些技术细节。在阅读本文之前,建议大家先熟悉一下Spring源码、SpringMVC源码、SpringBoot...
- springboot搭建xxl-job(分布式任务调度系统)
-
一、部署xxl-job服务端下载xxl-job源码:https://gitee.com/xuxueli0323/xxl-job二、导入项目、创建xxl_job数据库、修改配置文件为自己的数据库三、启动...
- 大模型:使用vLLM和Ray分布式部署推理应用
-
一、vLLM:面向大模型的高效推理框架1.核心特点专为推理优化:专注于大模型(如GPT-3、LLaMA)的高吞吐量、低延迟推理。关键技术:PagedAttention:类似操作系统内存分页管理,将K...
- 国产开源之光【分布式工作流调度系统】:DolphinScheduler
-
DolphinScheduler是一个开源的分布式工作流调度系统,旨在帮助用户以可靠、高效和可扩展的方式管理和调度大规模的数据处理工作流。它支持以图形化方式定义和管理工作流,提供了丰富的调度功能和监控...
- 简单可靠高效的分布式任务队列系统
-
#记录我的2024#大家好,又见面了,我是GitHub精选君!背景介绍在系统访问量逐渐增大,高并发、分布式系统成为了企业技术架构升级的必由之路。在这样的背景下,异步任务队列扮演着至关重要的角色,...
- 虚拟服务器之间如何分布式运行?(虚拟服务器部署)
-
在云计算和虚拟化技术快速发展的今天,传统“单机单任务”的服务器架构早已难以满足现代业务对高并发、高可用、弹性伸缩和容错容灾的严苛要求。分布式系统应运而生,并成为支撑各类互联网平台、企业信息系统和A...
- 一文掌握 XXL-Job 的 6 大核心组件
-
XXL-Job是一个分布式任务调度平台,其核心组件主要包括以下部分,各组件相互协作实现高效的任务调度与管理:1.调度注册中心(RegistryCenter)作用:负责管理调度器(Schedule...
- 京东大佬问我,SpringBoot中如何做延迟队列?单机与分布式如何做?
-
京东大佬问我,SpringBoot中如何做延迟队列?单机如何做?分布式如何做呢?并给出案例与代码分析。嗯,用户问的是在SpringBoot中如何实现延迟队列,单机和分布式环境下分别怎么做。这个问题其实...
- 企业级项目组件选型(一)分布式任务调度平台
-
官网地址:https://www.xuxueli.com/xxl-job/能力介绍架构图安全性为提升系统安全性,调度中心和执行器进行安全性校验,双方AccessToken匹配才允许通讯;调度中心和执...
- python多进程的分布式任务调度应用场景及示例
-
多进程的分布式任务调度可以应用于以下场景:分布式爬虫:importmultiprocessingimportrequestsdefcrawl(url):response=re...
- SpringBoot整合ElasticJob实现分布式任务调度
-
介绍ElasticJob是面向互联网生态和海量任务的分布式调度解决方案,由两个相互独立的子项目ElasticJob-Lite和ElasticJob-Cloud组成。它通过弹性调度、资源管控、...
- 分布式可视化 DAG 任务调度系统 Taier 的整体流程分析
-
Taier作为袋鼠云的开源项目之一,是一个分布式可视化的DAG任务调度系统。旨在降低ETL开发成本,提高大数据平台稳定性,让大数据开发人员可以在Taier直接进行业务逻辑的开发,而不用关...
- SpringBoot任务调度:@Scheduled与TaskExecutor全面解析
-
一、任务调度基础概念1.1什么是任务调度任务调度是指按照预定的时间计划或特定条件自动执行任务的过程。在现代应用开发中,任务调度扮演着至关重要的角色,它使得开发者能够自动化处理周期性任务、定时任务和异...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- MVC框架 (46)
- spring框架 (46)
- 框架图 (58)
- flask框架 (53)
- quartz框架 (51)
- abp框架 (47)
- jpa框架 (47)
- laravel框架 (46)
- springmvc框架 (49)
- 分布式事务框架 (65)
- scrapy框架 (56)
- shiro框架 (61)
- 定时任务框架 (56)
- java日志框架 (61)
- JAVA集合框架 (47)
- grpc框架 (55)
- ppt框架 (48)
- 内联框架 (52)
- winform框架 (46)
- gui框架 (44)
- cad怎么画框架 (58)
- ps怎么画框架 (47)
- ssm框架实现登录注册 (49)
- oracle字符串长度 (48)
- oracle提交事务 (47)