.NET开源高性能Socket通信中间件Helios介绍及演示
一:Helios是什么
Helios是一套高性能的Socket通信中间件,使用C#编写。Helios的开发受到Netty的启发,使用非阻塞的事件驱动模型架构来实现高并发高吞吐量。Helios为我们大大的简化了Socket编程,它已经为我们处理好了高并发情况下的解包,粘包,buffer管理等等。
GitHub:https://github.com/helios-io/helios/
为避免误会特别提示:helios不是本人作品,小弟还在努力的路上。
二:Helios的特点
1.Powerful APIs
Takes the complexity out of socket programming with intelligent I/O, concurrency, buffer management, and pipelining APIs.
使用socket编程不再复杂。提供智能的I/O,并发,buffer管理,管道形式的API。
2.Event-Driven
Helios is Reactive - it uses a event-driven architecture to simplify development and build responsive systems that scale.
Helios是反应式的,它使用事件驱动的架构来简化开发和构建易伸缩的系统。
3.Performant
Performance is a cross-cutting concern we factor in at every level in the design of the framework in order to eliminate overhead for your apps and clients.
这个系统在开发和设计的时候都充分考虑到了性能,构建你的app和client的时候请消除这方面的顾虑。
4.Battle-Tested
Helios powers the clustering and remoting capbilities built into Akka.NET and more.
Akka.net的集群,远程功能构建在Helios之上。
三:一个基于Helios的聊天室示例
要用来演示Socket通信那么最好的示例无非就是聊天程序了。
整个解决方案包含3个项目:
1.HeliosChat.Common
这个项目里是一些公共的类型,新建完之后使用nuget添加helios的库
Message 类:所有发送的消息都是通过Message包装的,每一个消息都有一个Command跟Content来构成。
public class Message
{
public Command Command { get; set; } public string Content { get; set; }
}
Command枚举:用来描述消息的命令
public enum Command
{
Join,
Send,
}
MessageConverter静态类:这个类用来转换Message对象为Byte[],或者把Byte[]转换成Message对象。Message对象在通过Helios传输的时候需要先转成Byte[],所以我们需要自己定义包的格式。我们用Byte[]的前四位来存放Command,Content转成Byte后从第5位开始存放。
public class MessageConverter
{
public static Message ToMessage(NetworkData data)
{
try
{
var commandData = data.Buffer.Take().ToArray();
var contentData = data.Buffer.Skip().Take(data.Buffer.Length - ).ToArray(); var command = BitConverter.ToInt32(commandData,);
var content = Encoding.UTF8.GetString(contentData); return new Message()
{
Command = (Command)command,
Content = content
};
}
catch (Exception exc)
{
Console.WriteLine("Cant convert NetworkData to Message : {0}", exc.Message);
} return null; } public static byte[] ToBytes(Message message)
{
try
{
var commandBytes = BitConverter.GetBytes((int)message.Command);
var messageBytes = Encoding.UTF8.GetBytes(message.Content);
var bytes = new byte[commandBytes.Length + messageBytes.Length];
commandBytes.CopyTo(bytes, );
messageBytes.CopyTo(bytes, commandBytes.Length); return bytes;
}
catch (Exception exc)
{
Console.WriteLine("Cant convert message to bytes : {0}", exc.Message);
} return null;
} }
2.HeliosChat.Server
不用说也知道,这是聊天室的服务端,负责连接用户及转发消息。
internal class Program
{
private static readonly ConcurrentDictionary<string, IConnection> Clients =
new ConcurrentDictionary<string, IConnection>(); private static void Main(string[] args)
{
var host = IPAddress.Any;
var port = ;
Console.Title = "Server";
Console.WriteLine("Starting server on {0}:{1}", host, port); var serverFactory =
new ServerBootstrap()
.SetTransport(TransportType.Tcp)
.Build();
var server = serverFactory.NewReactor(NodeBuilder.BuildNode().Host(host).WithPort(port));
server.OnConnection += (address, connection) =>
{
Console.WriteLine("Connected: {0}", address);
connection.BeginReceive(Receive);
};
server.OnDisconnection += (reason, address) =>
Console.WriteLine("Disconnected: {0}; Reason: {1}", address.RemoteHost, reason.Type);
server.Start();
Console.WriteLine("Running, press any key to exit");
Console.ReadKey();
} /// <summary>
/// 处理接受到的消息
/// </summary>
/// <param name="data"></param>
/// <param name="channel"></param>
public static void Receive(NetworkData data, IConnection channel)
{
var message = MessageConverter.ToMessage(data);
switch (message.Command)
{
case Command.Join:
JoinGroup(message.Content, channel);
break;
case Command.Send:
Broadcast(message.Content);
break;
}
} public static void JoinGroup(string clientName, IConnection channel)
{
if (Clients.TryAdd(clientName, channel))
{
Broadcast(string.Format("{0} join group successful .", clientName));
}
else
{
var errMsg = new Message()
{
Command = Command.Send,
Content = "client name is used."
};
SendMessage(channel, errMsg);
}
} /// <summary>
/// 广播消息
/// </summary>
/// <param name="clientMessage"></param>
public static void Broadcast(string clientMessage)
{
Console.WriteLine(clientMessage);
var clientName = clientMessage.Split(':')[];
var message = new Message
{
Command = Command.Send,
Content = clientMessage
};
foreach (var client in Clients)
{
if (client.Key != clientName)
{
SendMessage(client.Value, message);
}
}
} public static void SendMessage(IConnection connection, Message message)
{
var messageBytes = MessageConverter.ToBytes(message);
connection.Send(new NetworkData { Buffer = messageBytes, Length = messageBytes.Length });
}
}
3.HeliosChat.Client
聊天服务的客户端
internal class Program
{
public static IConnection Client;
public static string ClientName; private static void Main(string[] args)
{
var host = IPAddress.Loopback;
var port = ;
var connectionFactory =
new ClientBootstrap()
.SetTransport(TransportType.Tcp).Build();
//New一个Client
Client = connectionFactory.NewConnection(Node.Empty(), NodeBuilder.BuildNode().Host(host).WithPort(port));
Client.OnConnection += (address, connection) =>
{
Console.WriteLine("Connect server successful.");
connection.BeginReceive(Received);
};
Client.OnDisconnection += (address, reason) => Console.WriteLine("Disconnected.");
Console.WriteLine("Input ClientName ");
ClientName = Console.ReadLine();
Console.Title = string.Format("Client {0}", ClientName);
//建立连接
Client.Open();
//加入聊天组
Join();
//等待输入
WaitInput();
} public static void WaitInput()
{
while (true)
{
var input = Console.ReadLine();
if (!string.IsNullOrEmpty(input))
{
var message = MakeSendMessage(input);
SendMessage(Client, message);
}
}
} /// <summary>
/// Jion chat group
/// </summary>
public static void Join()
{
var message = MakeJoinMessage();
SendMessage(Client,message);
} /// <summary>
/// 处理接受到的消息
/// </summary>
/// <param name="data"></param>
/// <param name="responseChannel"></param>
public static void Received(NetworkData data, IConnection responseChannel)
{
var message = MessageConverter.ToMessage(data);
if (message.Command == Command.Send)
{
Console.WriteLine(message.Content);
}
} /// <summary>
/// 构造聊天消息
/// </summary>
/// <param name="input"></param>
/// <returns></returns>
public static Message MakeSendMessage(string input)
{
return new Message
{
Command = Command.Send,
Content = string.Format("{0}:{1}", ClientName, input)
};
}
/// <summary>
/// 构造加入组的消息
/// </summary>
/// <returns></returns>
public static Message MakeJoinMessage()
{
var message = new Message();
message.Command = Command.Join;
message.Content = ClientName; return message;
} public static void SendMessage(IConnection connection, Message message)
{
var messageBytes = MessageConverter.ToBytes(message);
connection.Send(new NetworkData { Buffer = messageBytes, Length = messageBytes.Length });
}
}
4.运行结果
这样一个简单的聊天室程序就完成了。
四:Helios 2.0
helios 1.0的异步编程模型是基于APM的,从helios 2.0开始会改成SocketAsyncEventArgs方式来实现异步。SocketAsyncEventArgs底层封装了IOCP,IOCP是Windows server上Socket通讯性能最高的技术,使用了IOCP的helios 2.0势必具有更高的性能,所以对于helios 2.0还是非常期待的。
示例下载:http://files.cnblogs.com/files/kklldog/HeliosChat.7z
.NET开源高性能Socket通信中间件Helios介绍及演示的更多相关文章
- 高性能分布式应用开发中间件ICE介绍
作为一个技术人员,你是否在为不断增长的数据量和日益复杂的业务逻辑而头疼不已,杂乱堆砌在一起的庞大业务让系统越来越脆弱,于是你想到了网格,想到了利用分布式来重组一个健壮的系统架构. 随后,RMI,EJB ...
- AgileEAS.NET SOA 中间件平台.Net Socket通信框架-介绍
一.前言 AgileEAS.NET SOA 中间件平台是一款基于基于敏捷并行开发思想和Microsoft .Net构件(组件)开发技术而构建的一个快速开发应用平台.用于帮助中小型软件企业建立一条适合市 ...
- AgileEAS.NET SOA 中间件平台.Net Socket通信框架-简单例子-实现简单的服务端客户端消息应答
一.AgileEAS.NET SOA中间件Socket/Tcp框架介绍 在文章AgileEAS.NET SOA 中间件平台Socket/Tcp通信框架介绍一文之中我们对AgileEAS.NET SOA ...
- AgileEAS.NET SOA 中间件平台.Net Socket通信框架-完整应用例子-在线聊天室系统-下载配置
一.AgileEAS.NET SOA中间件Socket/Tcp框架介绍 在文章AgileEAS.NET SOA 中间件平台Socket/Tcp通信框架介绍一文之中我们对AgileEAS.NET SOA ...
- 介绍开源的.net通信框架NetworkComms框架 源码分析
原文网址: http://www.cnblogs.com/csdev Networkcomms 是一款C# 语言编写的TCP/UDP通信框架 作者是英国人 以前是收费的 售价249英镑 我曾经花了 ...
- 介绍开源的.net通信框架NetworkComms框架之二 传递类
原文网址: http://www.cnblogs.com/csdev Networkcomms 是一款C# 语言编写的TCP/UDP通信框架 作者是英国人 以前是收费的 目前作者已经开源 开源地 ...
- 2017年8月9日学习内容存放 #socket通信介绍
2017年8月9日学习内容存放 #socket通信介绍 ''' OSI七层 应用 表示 会话 传输 网络 ip 数据链路 mac 物理层 网线 http smtp dns ftp ssh snmp i ...
- EQueue - 一个C#写的开源分布式消息队列的总体介绍
前言 本文想介绍一下前段时间在写enode时,顺便实现的一个分布式消息队列equeue.这个消息队列的思想不是我想出来的,而是通过学习阿里的rocketmq后,自己用c#实现了一个轻量级的简单版本.一 ...
- 跨平台通信中间件thrift学习【Java版本】(转)
转自:http://neoremind.com/2012/03/%E8%B7%A8%E5%B9%B3%E5%8F%B0%E9%80%9A%E4%BF%A1%E4%B8%AD%E9%97%B4%E4%B ...
随机推荐
- LogStash-2.4.0自定义区域信息插件-ISP
由于直接复制粘贴有问题,所以给出链接: http://note.youdao.com/share/?id=6dfb1f03240e156c1db4a56c85e3b6db&type=note# ...
- XCode的个人使用经验
Xcode是强大的IDE(但个人觉得不如Visual Studio做得好),其强大功能无需本人再赘述,本文也不是一篇“快捷键列表”,因为XCode上的快捷键极其多,而且还有不少是需要同时按下四个按键的 ...
- 从天猫和支付宝身上学习opcity与rgba
不知道什么时候,一个双层透明的对话框悄然流行了起来. 我们在各大网站上都能见到类似这样的对话框: 这样的聚焦更明显,用户体验更好,也显得更加的高大上. 先说点题外话,这种布局如何用CSS+DIV去实现 ...
- .Net组件程序设计之序列化
.Net组件程序设计之序列化 自动序列化 本篇给大家讲解一下在.NET中的序列化,它的用处非常之多,特别是对于某种环境下保存某种状态是很好的方法,接下来就来看一下吧. Serializable属性 ...
- 基于GIS的旅游辐射区人口统计
在旅游规划中,考虑旅游景点周边的人口负载量是很重要的一个方面,这将直接影响资源的投入和配置,开发潜力和规模等.基于GIS可以将人口信息进行空间化的展示,还可以通过空间分析的方法计算出旅游景点辐射区的人 ...
- TODO:Golang UDP连接简单测试慎用Deadline
TODO:Golang UDP连接简单测试慎用Deadline UDP 是User Datagram Protocol的简称, 中文名是用户数据报协议,是OSI(Open System Interco ...
- Chrome开发者工具不完全指南(五、移动篇)
前面介绍了Chrome开发者工具的大部分内容工具,现在介绍最后两块功能Audits和Console面板.一.Audits Audits面板会针对目前网页提出若干条优化的建议,这些建议分为两大类,一类是 ...
- Win10系统菜单打不开问题的解决,难道是Win10的一个Bug ?
Win10左下角菜单打不开,好痛苦,点击右下角的时间也没反应,各种不爽,折磨了我好几天,重装又不忍心,实在费劲,一堆开发环境要安装,上网找了很多方法都不适用.今天偶然解决了,仔细想了下,难道是Win1 ...
- Attach Volume 操作(Part I) - 每天5分钟玩转 OpenStack(53)
上一节我们创建了 volume,本节讨论如何将 volume attach 到 Instance,今天是第一部分. Volume 的最主要用途是作为虚拟硬盘提供给 instance 使用.Volume ...
- react+redux教程(四)undo、devtools、router
上节课,我们介绍了一些es6的新语法:react+redux教程(三)reduce().filter().map().some().every()....展开属性 今天我们通过解读redux-undo ...