一步一步开发Game服务器(二)完成登陆,聊天
我知道这样的文章在博客园已经多的大家都不想看了,但是这是我的系列文章开始,请各位大神见谅了。
多线程,线程执行器,(详见),socket通信相关 (详见)
本人blog相关文章测试代码,示例,完整版svn地址。(http://code.taobao.org/svn/flynetwork_csharp/trunk/Flynetwork/BlogTest)
提供全部源码功能块。希望各位大神,提供宝贵意见。
莫倩,完成了多线程辅助类库完整功能(或许后期会有bug需要修复或者优化),socket完成了tcp和http服务监听功能,udp和websocket还在完善的状态中。
如果有通信愿意和我一起完善这个辅助类库,请联系我,开通svn授权。
所以源码免费提供使用,欢迎各位爱好者,加入到项目中,无论是个人,企业,商用,都不限制。唯一要求请保留以下字样。谢谢合作~!
/** * * @author 失足程序员 * @Blog http://www.cnblogs.com/ty408/ * @mail 492794628@qq.com * @phone 13882122019 * */
好了开始我们的话题
在服务器项目开发中,最总要的也就是登陆问题了或者说叫授权问题。
我们先创建一个console的程序
引用我的两个库 Sz.Network.SocketPool ,Sz.Network.ThreadPool 分别是socket 帮助库线程帮助库。
Sz表示失足的意思。请见谅。偶喜欢上这个代号了“失足程序员”
我们先创建一个消息处理器 MessagePool
public class MessagePool : ISocketPool
{
public void ActiveSocket(IOSession client)
{
}
public void CloseSocket(IOSession client)
{
}
public void ReadMessage(IOSession client, SocketMessage message)
{
}
public void ActiveHttp(HttpClient client, string bind, Dictionary<string, string> parms)
{
if (bind.Equals("/test/"))
{
ThreadManager.GetInstance.AddTask(ServerManager.LoginThreadID, new LoginHttpHandler(client, parms));
}
}
public void IOSessionException(IOSession client, Exception exception)
{
Logger.Error("内部错误", exception);
}
public void HttpException(HttpClient client, Exception exception)
{
Logger.Error("内部错误", exception);
}
}
然后在mian函数里面加入
Sz.Network.SocketPool.ListenersBox.GetInstance.SetParams(new MessagePool(), typeof(MarshalEndian));
Sz.Network.SocketPool.ListenersBox.GetInstance.Start("tcp:*:9527", "http://*:8001/test/");
这样我们就开启了服务器的监听,这里简单介绍一下为什么我创建了tcp和http的两个监听呢?
这是因为经验和工作关系,因为我是致力于游戏开发的,服务器端需要创建两个tcp通常是用于正常通信的,而http监听的登陆模块,或者一些非总要性数据交换以及第三方登陆授权需要开启的。同样还因为http是短连接,无需保存通信对象,减少了系统消耗,和tcp数量级消耗。
如果你不理解可以不加入http的监听的。直接看tcp的socket。
[-- ::::Info ] Start Listen Tcp Socket -> [-- ::::Info ] Start Listen Http Socket -> /test/
运行程序,输出。
开启了tcp和http的监听,http我们今天暂时忽略其作用吧。
我们开始准备登陆模块的开发,同学都知道登陆首先要面临的就是多点同时登陆问题。
如果加入 lock 来防止多点同时登陆,那么势必照成服务器卡顿。吞吐量不高等因素。那么我们考虑把这一块加入到单独的线程惊喜处理。也就是帮登陆和登出,放到同一个线程处理。不用加锁,也能做到防止多点同时登陆。
接下来我们需要从 Sz.Network.ThreadPool 库中取出一个线程
public static readonly long LoginThreadID = ThreadManager.GetInstance.GetThreadModel("登陆处理器");
用于控制登陆和登出
创建一个 CloseTcpHandler 处理链接端口的处理程序 需要继承 Sz.Network.ThreadPool 下面的 TaskBase
public class CloseTcpHandler : TaskBase
{
IOSession client;
SocketMessage message;
public CloseTcpHandler(IOSession client)
: base("Tcp登陆处理")
{
this.client = client;
}
public override void TaskRun()
{
}
}
那么我们修改一下 MessagePool 类
public void ActiveSocket(IOSession client)
{
//client.SendMsg(new SocketMessage(1, System.Text.UTF8Encoding.Default.GetBytes("Holle Server!client")));
}
public void CloseSocket(IOSession client)
{
ThreadManager.GetInstance.AddTask(ServerManager.LoginThreadID, new CloseTcpHandler(client));
}
这样断开链接处理就交到了 ServerManager.LoginThreadID 这个线程里面处理了
我们再来创建一下 LoginTcpHandler 处理登陆程序 需要继承 Sz.Network.ThreadPool 下面的 TaskBase
修改 MessagePool 类 处理登陆消息
public void ReadMessage(IOSession client, SocketMessage message)
{
switch (message.MsgID)
{
://登陆
:
ThreadManager.GetInstance.AddTask(ServerManager.LoginThreadID, new LoginTcpHandler(client, message));
break;
default:
Logger.Error("未绑定消息ID " + message.MsgID);
break;
}
}
这里是我自定义消息ID是1和2的一个是登陆一个是登出。
这样就把登陆的事件也交给了ServerManager.LoginThreadID 这个线程里面处理了
LoginTcpHandler taskrun方法
public override void TaskRun()
{
using (MemoryStream msReader = new MemoryStream(message.MsgBuffer))
{
using (System.IO.BinaryReader srReader = new BinaryReader(msReader, UTF8Encoding.Default))
{
using (MemoryStream msWriter = new MemoryStream())
{
using (System.IO.BinaryWriter srWriter = new BinaryWriter(msWriter, UTF8Encoding.Default))
{
switch (message.MsgID)
{
://登陆
string username = srReader.ReadString();
if (!LoginManager.GetInstance.LoginNames.Contains(username))
{
LoginManager.GetInstance.LoginNames.Add(username);
if (!LoginManager.GetInstance.LoginIPs.ContainsKey(client.ID))
{
LoginManager.GetInstance.LoginIPs[client.ID] = username;
LoginManager.GetInstance.Sessions.Add(client);
}
srWriter.Write(true);
srWriter.Write(username + " 登陆聊天室");
Logger.Info(client.RemoteEndPoint + " " + username + " 登陆成功");
SocketMessage sm = new SocketMessage(message.MsgID, msWriter.GetBuffer());
ServerManager.GetInstance.Tell_All(sm);
}
else
{
srWriter.Write(false);
srWriter.Write("登录名称重复,请换一个");
Logger.Info(client.RemoteEndPoint + " " + username + " 登录名称重复!");
SocketMessage sm = new SocketMessage(message.MsgID, msWriter.GetBuffer());
client.SendMsg(sm);
}
break;
:// 退出登陆
break;
default:
break;
}
}
}
}
}
}
CloseTcpHandler taskrun方法
public override void TaskRun()
{
if (LoginManager.GetInstance.LoginIPs.ContainsKey(client.ID))
{
string username = LoginManager.GetInstance.LoginIPs[client.ID];
LoginManager.GetInstance.LoginIPs.Remove(client.ID);
LoginManager.GetInstance.LoginIPs.Remove(username);
LoginManager.GetInstance.Sessions.Remove(client);
using (MemoryStream msWriter = new MemoryStream())
{
using (System.IO.BinaryWriter srWriter = new BinaryWriter(msWriter, UTF8Encoding.Default))
{
srWriter.Write(username + "退出聊天室");
SocketMessage sm = , msWriter.GetBuffer());//3表示发送消息
ServerManager.GetInstance.Tell_All(sm);
}
}
}
}
这里由于代码没有贴全,有兴趣的可以下载源码试试

启动两个客户端后,看到创建了两个链接,并且登陆到服务器。
由于聊天和登陆所以不同的两个模块,为了提高效率 我们再次创建一个聊天线程
public static readonly long ChatThreadID = ThreadManager.GetInstance.GetThreadModel("聊天处理器");
接下来我们在 MessagePool 的 ReadMessage 方法的switch里面加入
://聊天
ThreadManager.GetInstance.AddTask(ServerManager.ChatThreadID, new Chat.ChatHandler(client, message));
break;
这样我们就把所有的聊天消息转发的ServerManager.ChatThreadID这个线程处理。就是卡顿情况,也不会影响客户端聊天发送消息和新客户端请求登陆。
这里我们还可以考虑分组,聊天分组功能。比如私聊,群聊等分组线程进行执行。
创建一个 ChatHandler 需要继承 Sz.Network.ThreadPool 下面的 TaskBase
public class ChatHandler : TaskBase
{
IOSession client;
SocketMessage message;
public ChatHandler(IOSession client, SocketMessage message)
: base("聊天处理任务")
{
this.client = client;
this.message = message;
}
public override void TaskRun()
{
using (MemoryStream msWriter = new MemoryStream())
{
using (System.IO.BinaryWriter srWriter = new BinaryWriter(msWriter, UTF8Encoding.Default))
{
//构建输入buffer
//验证登陆情况
if (LoginManager.GetInstance.LoginIPs.ContainsKey(client.ID))
{
string username = LoginManager.GetInstance.LoginIPs[client.ID];
using (MemoryStream msReader = new MemoryStream(message.MsgBuffer))
{
using (System.IO.BinaryReader srReader = new BinaryReader(msReader, UTF8Encoding.Default))
{
string msg = srReader.ReadString();
msg = client.RemoteEndPoint + " " + username + " " + msg;
Logger.Info(msg);
srWriter.Write(msg);
SocketMessage sm = new SocketMessage(message.MsgID, msWriter.GetBuffer());
ServerManager.GetInstance.Tell_All(sm);
}
}
}
else
{
srWriter.Write("尚未登陆");
SocketMessage sm = new SocketMessage(message.MsgID, msWriter.GetBuffer());
client.SendMsg(sm);
}
}
}
}
}
发个消息试试
这里一个简单的聊天服务器,登陆到聊天就算完成了,客户端是wpf的程序,没有贴出源码和过程,有需要或者要研究的亲请下载svn源码,自行查看情况。
一步一步开发Game服务器(二)完成登陆,聊天的更多相关文章
- 一步一步开发Game服务器(三)加载脚本和服务器热更新(二)完整版
上一篇文章我介绍了如果动态加载dll文件来更新程序 一步一步开发Game服务器(三)加载脚本和服务器热更新 可是在使用过程中,也许有很多会发现,动态加载dll其实不方便,应为需要预先编译代码为dll文 ...
- 一步一步开发Game服务器(二)登陆2
上一篇文章,讲解了简单的登陆情况.接下来我们继续讲解登陆模块. 在正常的游戏服务器情况下.在尚未登录前可以查看服务器大区情况,登陆后也可以查看服务器大区情况,然后选择大区服务器.进行登录操作. 这样的 ...
- 一步一步开发Game服务器(四)地图线程
时隔这么久 才再一次的回归正题继续讲解游戏服务器开发. 开始讲解前有一个问题需要修正.之前讲的线程和定时器线程的时候是分开的. 但是真正地图线程与之前的线程模型是有区别的. 为什么会有区别呢?一个地图 ...
- 一步一步跟我学DeviceOne开发 - 仿微信应用(一,二,三)
这是一个系列的文档,长期目标是利用DeviceOne开发一些目前使用广泛的优质手机应用,我们会最大化的实现这些应用的每一个功能和细节,不只停留在简单的UI模仿和Demo阶段,而是一个基本可以使用的实际 ...
- 一步一步开发Game服务器(一)
什么是服务器?对于很多人来说也许只是简单成为在服务器端运行的程序的确如此,服务器通常意义就是说在服务器端运行的程序而已.那么我们怎么理解和分析游戏服务器哪? 传统意义上来说,程序运行后,正常流程, 启 ...
- JAVA+PHP+阿里云组件纯手工实现POP、SMTP、IMAP开发邮件服务器(二)
java开发邮件服务器的接收模块 用java建立socket服务端,监听端口25,实现SMTP协议.即可完成邮件服务器的接收模块. 这里要注意的是,SMTP协议其实可以分为两种.一种是你用手机.PC等 ...
- 跟我一步一步开发自己的Openfire插件
http://www.blogjava.net/hoojo/archive/2013/03/07/396146.html 跟我一步一步开发自己的Openfire插件 这篇是简单插件开发,下篇聊天记录插 ...
- xmppmini 项目详解:一步一步从原理跟我学实用 xmpp 技术开发 2.登录的实现
第二章登录的实现 金庸<倚天屠龙记> 张三丰缓缓摇头,说道:“少林派累积千年,方得达成这等绝技,决非一蹴而至,就算是绝顶聪明之人,也无法自创.”他顿了一顿,又道:“我当年在少林寺中住过,只 ...
- 一步一步实现HTTP服务器-开篇
缘起 翻开清单,一条条计划一直列在那里,一天又一天,不知道什么时候写下了它,也知不道什么时候完成它,它一直在那静静的等待着. 静下心来,反思自己,才发现自己是多么的无知,多么的没有毅力.设定了无数目标 ...
随机推荐
- Win7网上邻居提示未授予用户在此计算机上的请求登录类型解决办法
内容简介 装了Win7之后很多人遇到这样的问题,网上邻居访问Win7的电脑时出现“未授予用户在此计算机上的请求登录类型”问题.打开“控制面板”--“管理工具”--“本地安全策略”--“本地策略 ...
- 每周一书-《鸟哥的Linux私房菜基础学习篇(第四版)》台湾原版,你想要吗?
首先说明,本周活动有效时间为2016年10月19日到2016年10月31日. 目在介绍这本书之前,首先要感谢QQ号为:1084830483(路在远方),来自哈尔滨工程大学的同学赠送给玄魂工作室的 ...
- log4j2.xml实用例子
一个多月前,我写了篇关于log4j.xml配置的文章,点击此处查看:http://www.cnblogs.com/guogangj/p/3931397.html 最近,我把自己的log4j升级到2.0 ...
- [.net 面向对象程序设计进阶] (22) 团队开发利器(一)简单易用的代码管理工具VSS
[.net 面向对象程序设计进阶] (22) 团队开发利器(一)简单易用的代码管理工具VSS 本篇要点:在进阶篇快要结束的时候说说源代码管理器,我们的开发,不是一个人可以完成的事,团队协作很重要,而且 ...
- java中文乱码解决之道(二)-----字符编码详解:基础知识 + ASCII + GB**
在上篇博文(java中文乱码解决之道(一)-----认识字符集)中,LZ简单介绍了主流的字符编码,对各种编码都是点到为止,以下LZ将详细阐述字符集.字符编码等基础知识和ASCII.GB的详情. 一.基 ...
- [SDK2.2]SQL Azure (13) Azure的两种关系型数据库服务:SQL Azure与SQL Server VM的不同
<Windows Azure Platform 系列文章目录> 如果熟悉Windows Azure平台的用户不难发现,对于SQL Server数据库来说,微软提供了两种服务,分别是: -W ...
- MySQL中的全文索引
之前曾经发表了一篇关于SQL Server全文索引的文章.现在将MySQL全文索引的配置过程记录一下. Step1:创建Student表 CREATE TABLE `student` ( `id` I ...
- Understanding RabbitMQ Exchange & Queue
Exchanges are the only places where messages could be published to; while queues are the only places ...
- Module-Zero之启动模板
返回<Module Zero学习目录> 概览介绍 社交登录 基于Token的认证 单元测试 概览介绍 使用ABP和Module-Zero开始一个新的项目最简单的方式通过ABP官网的模板页面 ...
- Objective-C 观察者模式--简单介绍和使用
观察者模式(有时又被称为发布-订阅模式) 在此种模式中,一个目标物件管理所有相依于它的观察者物件,并且在它本身的状态改变时主动发出通知. 这通常透过呼叫各观察者所提供的方法来实现.此种模式通常被用来实 ...
