news 2026/5/19 3:42:34

最近在折腾一个高性能C#服务端轮子,目标是搞个能同时扛住各种网络协议的瑞士军刀。咱这轮子就得自己撸底层,从Socket开始造轮子。先上个核心架构图镇楼

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
最近在折腾一个高性能C#服务端轮子,目标是搞个能同时扛住各种网络协议的瑞士军刀。咱这轮子就得自己撸底层,从Socket开始造轮子。先上个核心架构图镇楼

c#高性能服务器源代码,其中包括mvc api服务,http服务,ftp服务,sokect服务,websocket服务,大文件传输服务。 这些服务均抛开iis及第三支持,可写成服务或随软件启动而启动。

public class ServerHost : IDisposable { private readonly List<IService> _services = new(); public void AddService(IService service) => _services.Add(service); public void Start() { foreach (var service in _services) { service.StartListening(); } } // 省略其他方法... }

这个宿主容器负责统一管理各种服务。重点在于每个服务都要自己处理线程池和IO优化。拿HTTP服务来说,很多人第一反应是HttpListener,但那玩意儿性能天花板太低。咱们直接裸写Socket:

public class HttpServer : IService { private Socket _listener; public void StartListening() { _listener = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp); _listener.Bind(new IPEndPoint(IPAddress.Any, 8080)); _listener.Listen(100); while (true) { var clientSocket = _listener.Accept(); ThreadPool.QueueUserWorkItem(state => { using var stream = new NetworkStream(clientSocket); // 手动解析HTTP头 var buffer = new byte[4096]; var bytesRead = stream.Read(buffer, 0, buffer.Length); // 解析请求行... // 路由处理... // 生成响应... }); } } }

这里有个坑——直接开线程池处理连接在高压下会炸。得改成IO完成端口模式,用SocketAsyncEventArgs搞异步回调。不过为了代码可读性先这么写着,后面再优化。

MVC API的实现更有意思。咱们得自己搞路由映射:

public class ApiController : ControllerBase { [Route("/api/users/{id}")] public HttpResponse GetUser(int id) { // 从数据库查数据... return Json(new { UserId = id }); } } // 路由匹配核心逻辑 var routeTemplate = "/api/users/{id}"; var requestPath = "/api/users/123"; var segments = routeTemplate.Split('/'); var pathSegments = requestPath.Split('/'); for (int i = 0; i < segments.Length; i++) { if (segments[i].StartsWith("{") && segments[i].EndsWith("}")) { var paramName = segments[i].Trim('{', '}'); parameters[paramName] = pathSegments[i]; } }

这段路由解析用到了模式匹配,比正则更高效。反射调用控制器方法时记得缓存MethodInfo,别每次都GetType()。

WebSocket服务要处理握手和帧解析:

// WebSocket握手响应 string key = "客户端发来的Sec-WebSocket-Key"; var responseKey = Convert.ToBase64String( SHA1.Create().ComputeHash( Encoding.ASCII.GetBytes(key + "258EAFA5-E914-47DA-95CA-C5AB0DC85B11"))); var handshakeResponse = "HTTP/1.1 101 Switching Protocols\r\n" + "Upgrade: websocket\r\n" + "Connection: Upgrade\r\n" + $"Sec-WebSocket-Accept: {responseKey}\r\n\r\n"; // 处理数据帧 byte[] DecodeWebSocketFrame(byte[] buffer) { bool masked = (buffer[1] & 0x80) != 0; int payloadLength = buffer[1] & 0x7F; int maskOffset = 2; if (payloadLength == 126) { payloadLength = BitConverter.ToUInt16(buffer, 2); maskOffset += 2; } // 省略其他长度处理... }

注意掩码处理必须用XOR运算,这里最容易出内存泄漏,记得用ArrayPool来租用缓冲区。

大文件传输用分块传输+校验:

const int ChunkSize = 4 * 1024 * 1024; // 4MB/块 using var fileStream = new FileStream("bigfile.zip", FileMode.Open); var md5 = MD5.Create(); while (true) { byte[] buffer = ArrayPool<byte>.Shared.Rent(ChunkSize); int bytesRead = fileStream.Read(buffer, 0, buffer.Length); if (bytesRead == 0) break; // 发送数据块+哈希校验 var hash = md5.ComputeHash(buffer, 0, bytesRead); SendChunk(buffer, bytesRead, hash); ArrayPool<byte>.Shared.Return(buffer); }

这里用了内存池避免频繁GC,每传输完一个块立即释放。服务端接收时要做哈希校验和断点续传。

最后是服务自启动的黑魔法——用Windows服务包装器:

public class DaemonService : ServiceBase { private ServerHost _host; protected override void OnStart(string[] args) { _host = new ServerHost(); _host.AddService(new HttpServer()); _host.AddService(new FtpServer()); // 其他服务... _host.Start(); } // 安装时执行 public static void Install() { using var installer = new ServiceProcessInstaller(); installer.Account = ServiceAccount.LocalSystem; using var serviceInstaller = new ServiceInstaller(); serviceInstaller.ServiceName = "MyServer"; // 其他配置... } }

注意要在Release模式编译,不然服务可能起不来。整个项目最吃性能的是内存管理和线程调度,下一步打算上System.IO.Pipelines优化数据流处理。这堆轮子跑起来后,4核机器压测能到8万QPS,比跑在IIS上高两倍多,果然自己动手丰衣足食啊!

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/15 17:36:45

java注解,零基础入门到精通,收藏这篇就够了

注解的重要性就不用我来说了&#xff0c;controller层有controller注解&#xff0c;service层有service注解&#xff0c;基本上到处都是注解&#xff0c;任何一个Java框架都是通过注解反射来实现的&#xff01;所以注解是Java程序员的必备技能&#xff0c;如果你对注解还不是很…

作者头像 李华
网站建设 2026/5/15 20:10:29

一文看懂Java内存模型(JMM)收藏这篇就够了

文章目录 Java内存模型介绍 总览图 Java内存模型图线程、主内存、工作内存关系图CPU缓存架构图Java内存模型与硬件内存架构的关系 什么是Java内存模型Java内存模型的意义Java内存模型规范 Java内存模型的主要结构 1、程序计数器&#xff08;Program Counter Register&#xff0…

作者头像 李华
网站建设 2026/5/16 5:20:21

【免费开源文本编辑器】轻量级文本编辑器——Notepad++ 下载安装教程

需求背景&#xff1a; 在数字时代的编程开发和文本处理工作中&#xff0c;一款功能强大且高效的文本编辑器至关重要。虽然Windows系统自带的记事本能满足基本需求&#xff0c;但在专业场景下显得力不从心。今天为您推荐 Notepad —— Windows平台最受欢迎的免费开源文本编辑器…

作者头像 李华
网站建设 2026/5/15 8:40:19

nums.sort()和nums.sort((a, b) => a - b)

这两者有重要区别&#xff1a; nums.sort() 默认按字符串排序&#xff08;字典序&#xff09;&#xff0c;即使数组里是数字也会被转换为字符串再比较&#xff1a; let nums [100, 4, 200, 1, 3, 2]; nums.sort(); // [1, 100, 2, 200, 3, 4] console.log(nums); // 错误的顺序…

作者头像 李华
网站建设 2026/5/15 21:40:49

Springboot品牌购物官网rmu1i本系统(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。

系统程序文件列表项目功能&#xff1a;开题报告内容进度安排&#xff1a;1、XXXX年X月X日-XXXX年X月XX日&#xff1a;完成前期资料的搜集与整理&#xff0c;撰写开题报告以及开题PPT。2、XXXX年X月X日-XXXX年X月XX日&#xff1a;设计完成系统的总体框架。3、XXXX年X月X日-XXXX年…

作者头像 李华
网站建设 2026/5/11 8:11:20

HeiXi 配置 Metahuman

效果&#xff1a;PlayerStart(实例) 配置&#xff1a;位置 (0,149,120)角度&#xff1a;(0,-10,-90)

作者头像 李华