using System; using System.Collections.Generic; using System.Configuration; using System.Linq; using System.Net; using System.Net.Sockets; using System.Text; using System.Threading; using System.Threading.Tasks; using Jinher.AMP.SNS.Chat.Client; using Jinher.AMP.SNS.Chat.Deploy.CustomDTO; using Jinher.AMP.SNS.Chat.Packet; using Jinher.AMP.SNS.Chat.Utility; namespace Jinher.AMP.SNS.Chat.SocketManager { /// <summary> /// 消息服务基类 /// </summary> public abstract class MessageCenter { /// <summary> /// 表示是否连接上 /// </summary> public bool IsConnected { get; protected set; } /// <summary> /// 表示是否进行第一次握手协议 /// </summary> public bool IsHandStake { get; protected set; } /// <summary> /// 表示是否注册APP /// </summary> public bool IsRegisterApp { get; protected set; } private string _ip = string.Empty; ; private object lockObject = new object(); List<byte[]> byteList = new List<byte[]>(); //通知 private SocketAsyncEventArgsPool pool = null; private BufferManager m_bufferManager = null; /// <summary> /// 线程休眠时间(毫秒) /// </summary> ; //连接对象 private Socket _socket = null; ; protected Action<List<MessageTemplate>> MyAction = null; protected Action<WebMessageDTO> receiveAction = null; public MessageCenter() { //初始化,获取Host和Port _ip = System.Configuration.ConfigurationManager.AppSettings["serverip"]; _port = int.Parse(System.Configuration.ConfigurationManager.AppSettings["port"]); maxNumber = "); "; threadSleepTime = int.Parse(timer); //初始化连接对象 InitSocket(); } /// <summary> /// 带委托的构造函数 /// </summary> /// <param name="action"></param> public MessageCenter(Action<List<MessageTemplate>> action) : this() { MyAction = action; } /// <summary> /// 启动消息中心 /// </summary> public void Run() { try { LogHelper.WriteLog("启动服务"); //第一次连接服务器 Connect(); //开始接受 AccpetTo(); //开始处理接受的数据 ProcessAccpet(); //保持连接 KeepConnect(); //推送消息队列 //SendAsync(); Task.Factory.StartNew(new Action(() => { client = ChatClient.Instance(ConfigurationManager.AppSettings["notificationUri"]); client.OnMsgReceiveed = null; client.OnMsgReceiveed += client_OnMsgReceiveed; })); } catch (SocketException exception) { //socket出错 LogHelper.WriteLog("socket创建连接出错,尝试重新启动", exception); Thread.Sleep(); this.Run(); } catch (Exception ex) { //未知错误 LogHelper.WriteLog("启动时发生未知错误!", ex); throw; } finally { //重置连接 //ResetConnect(); } } #region 连接服务器 /// <summary> /// 连接服务器 /// </summary> protected void Connect() { try { if (_socket == null) { InitSocket(); } LogHelper.WriteLog("开始建立连接"); _socket.Connect(IPAddress.Parse(_ip), _port); LogHelper.WriteLog("连接已经建立"); } catch (Exception ex) { LogHelper.WriteLog("出错了", ex); LogHelper.WriteLog("建立连接失败,等待重新建立连接"); Thread.Sleep(threadSleepTime); Connect(); return; } //先进行第一次握手协议 HandShakeCmdOp(); //注册app RegisterApp(); IsConnected = true; } /// <summary> /// 先进行第一次握手协议 /// </summary> private void HandShakeCmdOp() { if (!IsHandStake) { LogHelper.WriteLog("开始第一次握手"); _socket.Send(pmsMessage.HandShakePacket()); IsHandStake = true; LogHelper.WriteLog("成功握手"); } } /// <summary> /// 注册app /// </summary> private void RegisterApp() { // 一级命令:XNS_ROUTER //二级命令:REGISTER_SOCIAL_APP //并且规定AppId:99999999 LogHelper.WriteLog("开始注册APP"); _socket.Send(pmsMessage.RegisterPacket()); IsRegisterApp = true; LogHelper.WriteLog("注册成功"); } /// <summary> /// 初始化连接对象 /// </summary> private void InitSocket() { _socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp); } #endregion #region 发送消息 /// <summary> /// 发送消息(异步) /// </summary> /// <param name="buffer"></param> /// <param name="count">表示失败重新发送次数</param> /// <returns></returns> ) { //先将该消息尝试发送,如果发送失败,则连接后继续发送 //byteList.Add(buffer); //启动任务,发送消息 Task.Factory.StartNew(new Action(() => { try { if (_socket != null && IsConnected && _socket.Connected && !_isResetConnecting) { lock (lockObject) { if (_socket != null && IsConnected && _socket.Connected && !_isResetConnecting) { SendByInit(buffer); } else { ResetConnect(); SendAsync(buffer); } } } else { ResetConnect(); SendAsync(buffer); } } catch (Exception) { //重发一次 ) { //表示失败重新发送 SendAsync(buffer, count + ); } else { ResetConnect(); SendAsync(buffer); } } finally { } })); } /// <summary> /// 发送数据() /// </summary> /// <param name="buffer"></param> private void SendByInit(byte[] buffer) { _socket.BeginSend(buffer, , buffer.Length, SocketFlags.None, new AsyncCallback(SendComplated), _socket); } /// <summary> /// 发送消息回调函数 /// </summary> /// <param name="ar"></param> private void SendComplated(IAsyncResult async) { try { Socket skt = async.AsyncState as Socket; if (skt.Connected) { skt.EndSend(async); } LogHelper.WriteLog("发送成功"); } catch (SocketException ex) { //日志文件 } } #endregion #region 接受消息(方式一) /// <summary> /// 接受消息方式一 /// </summary> public void AccpetOne() { LogHelper.WriteLog("开始建立接受消息(方式一)"); pool = ); // 预先分配一个对象池 SocketAsyncEventArgs readWriteEventArg; ; i < maxNumber; i++) { m_bufferManager = * ); m_bufferManager.InitBuffer(); //初始化 SocketAsyncEventArgs readWriteEventArg = new SocketAsyncEventArgs(); readWriteEventArg.Completed += new EventHandler<SocketAsyncEventArgs>(Receive_Completed); readWriteEventArg.UserToken = new AsyncUserToken() { Manager = m_bufferManager }; // 从缓冲池分配一个字节缓冲区SocketAsyncEventArg对象 m_bufferManager.SetBuffer(readWriteEventArg); // add SocketAsyncEventArg to the pool pool.Push(readWriteEventArg); } //取出 监视 Receive(); } private void Receive() { if (IsConnected && IsRegisterApp && IsHandStake) { SocketAsyncEventArgs readEventArgs = null; readEventArgs = pool.Pop(); if (readEventArgs != null) { ((AsyncUserToken)readEventArgs.UserToken).Socket = _socket; _socket.ReceiveAsync(readEventArgs); } } } private void Receive_Completed(object sender, SocketAsyncEventArgs e) { Receive(); ReceiveAsync(e); } private void ReceiveAsync(SocketAsyncEventArgs e) { AsyncUserToken token = (AsyncUserToken)e.UserToken; && e.SocketError == SocketError.Success) { //echo the data received back to the client e.SetBuffer(, e.BytesTransferred); //string txt = Encoding.Default.GetString(e.Buffer, 0, e.BytesTransferred); //Console.WriteLine(tmp); //通知 #region 收到消息平台发送来的消息 //client.Send(); ) { byte[] temp = new byte[e.BytesTransferred]; Array.Copy(e.Buffer, , temp, , e.BytesTransferred); ProcessAccpet(temp, e.BytesTransferred); } #endregion BufferManager bufferManager = ((AsyncUserToken)(e.UserToken)).Manager; if (bufferManager == null) { bufferManager = * ); e.UserToken = new AsyncUserToken() { Manager = bufferManager }; } bufferManager.FreeBuffer(e); bufferManager.SetBuffer(e); //回收 pool.Push(e); } } #endregion #region 接受消息(方式二) List<byte> testList = new List<byte>(); private void AccpetTo() { LogHelper.WriteLog("开始建立接受消息(方式二)"); Task.Factory.StartNew(new Action(() => { try { * ]; while (true) { int r = _socket.Receive(bytes); //通知 #region 收到消息平台发送来的消息 ) { byte[] temp = new byte[r]; Array.Copy(bytes, , temp, , r); lock ("我要给集合增加数据") { testList.AddRange(temp); } // ProcessAccpet(temp, r); } #endregion Array.Clear(bytes, , r); } } catch (SocketException ex) { LogHelper.WriteLog("socket出错", ex); ResetConnect(); } catch (Exception ex) { LogHelper.WriteLog("出错", ex); } })); } #endregion #region 收到的消息进行处理 /// <summary> /// 消息处理中转 /// </summary> /// <param name="p"></param> /// <param name="r"></param> private async void ProcessAccpet(byte[] p, int r) { //if (r > 10) //{ // LogHelper.WriteLog("收到正常消息"); // var result = pmsMessage.ReversePacketMessage(p, r); // //发送给客户端 // ClientSend(result); // //发送给测试端 // if (MyAction != null) // { // MyAction(result.ToList()); // } //} //else //{ // LogHelper.WriteLog(string.Format("收到心跳包")); //} } //标示处理任务是否正在运行 private bool IsProcessAccpetRuning = false; private void ProcessAccpet() { if (IsProcessAccpetRuning) {//表示任务正在执行 return; } else { IsProcessAccpetRuning = true; } Task.Factory.StartNew(new Action(() => { //IsProcessAccpetRuning = false; //Thread.Sleep(100); //IsProcessAccpetRuning = true; while (IsProcessAccpetRuning) { ) { List<byte> tempList = new List<byte>(); lock ("我要给集合增加数据") { tempList = testList.ToList(); testList.Clear(); } //启动一个任务来发送数据 Task.Factory.StartNew(new Action(() => { var result = pmsMessage.ReversePacketMessage(tempList, tempList.Count); //发送给客户端 ClientSend(result); //发送给测试端 ) { MyAction(result.ToList()); } })); } else { Thread.Sleep(); } } LogHelper.WriteLog("新的处理请求,停止处理接受"); })); } #endregion #region 保持连接 /// <summary> /// 重置连接 /// </summary> public void ResetConnect() { if (!_isResetConnecting) { lock (lockObject) { if (!_isResetConnecting) { _isResetConnecting = true; LogHelper.WriteLog("=========重新连接=============="); lock (lockObject) { this.Dispose(); InitSocket(); } Thread.Sleep(); Run(); _isResetConnecting = false; } } } } private Timer timer = null; /// <summary> /// 保持连接 /// </summary> private void KeepConnect() { timer = new Timer(new TimerCallback((o) => { if (IsConnected) { //发送心跳包 SendAsync(pmsMessage.GetHeardbeat()); LogHelper.WriteLog("已经发送心跳包"); } }), , , , ), , , , )); } #endregion #region 通知相关 private ChatClient client = null;// ChatClient.Instance(ConfigurationManager.AppSettings["notificationUri"]); PacketConvert pmsMessage = new PacketConvert(); private bool _isResetConnecting = false; /// <summary> /// 请求发送消息 /// </summary> /// <param name="msg"></param> protected void client_OnMsgReceiveed(WebMessageDTO msg) { //请求发送消息 LogHelper.WriteLog(string.Format("开始请求发送消息:消息id=>{0} **********消息内容=>:{1}", msg.MsgId, msg.Msg != null ? msg.Msg.Message : "")); byte[] bytes = pmsMessage.PacketMessage(msg); SendAsync(bytes); //发送给测试程序 if (receiveAction != null) { receiveAction(msg); } } private void ClientSend(IList<MessageTemplate> message) { if (message != null && client != null) client.SendAsync(message); } #endregion #region 资源清理 /// <summary> /// 清理资源 /// </summary> public void Dispose() { IsRegisterApp = false; IsHandStake = false; IsConnected = false; if (_socket.Connected == true) _socket.Shutdown(SocketShutdown.Both); _socket.Dispose(); _socket = null; pool = null; m_bufferManager = null; } #endregion } }
C# socket 实现消息中心向消息平台 转发消息 (修改)的更多相关文章
-
C# socket 实现消息中心向消息平台 转发消息
公司用到,直接粘代码了 using System; using System.Collections.Generic; using System.Configuration; using System ...
-
IBM MQ消息中间件jms消息中RHF2消息头的处理
公司的技术平台在和某券商对接IBM MQ消息中间件时,发送到MQ中的消息多出了消息头信息:RHF2,造成消息的接收处理不正常.在此记录此问题的处理方式. 在IBM MQ中提供了一个参数 targetC ...
-
如何在MFC DLL中向C#类发送消息
如何在MFC DLL中向C#类发送消息 一. 引言 由于Windows Message才是Windows平台的通用数据流通格式,故在跨语言传输数据时,Message是一个不错的选择,本文档将描述如何在 ...
-
微信开发——微信公众平台实现消息接收以及消息的处理(Java版)
本文主要讲述了如何在微信公众平台实现消息接收以及消息的处理,使用java语言开发,现在把实现思路和代码整理出来分先给兄弟们,希望给他们带来帮助. 温馨提示: 这篇文章是依赖前几篇的文章的. 第一篇:微 ...
-
Team Foundation 中的错误和事件消息
Visual Studio Team System Team Foundation 中的错误和事件消息 Team Foundation 通过显示错误消息和事件消息来通知您操作成功以及操作失败.一部分错 ...
-
ROS Learning-027 (提高篇-005 A Mobile Base-03) 控制移动平台 --- Twist 消息
ROS 提高篇 之 A Mobile Base-03 - 控制移动平台 - Twist 消息 我使用的虚拟机软件:VMware Workstation 11 使用的Ubuntu系统:Ubuntu 14 ...
-
Delphi中SendMessage使用说明(所有消息说明) good
Delphi中SendMessage使用说明 SendMessage基础知识 函数功能:该函数将指定的消息发送到一个或多个窗口.此函数为指定的窗口调用窗口程序,直到窗口程序处理完消息再返回.而函数Po ...
-
ASP.NET MVC5+EF6+EasyUI 后台管理系统(73)-微信公众平台开发-消息管理
系列目录 前言 回顾上一节,我们熟悉的了解了消息的请求和响应,这一节我们来建立数据库的表,表的设计蛮复杂 你也可以按自己所分析的情形结构来建表 必须非常熟悉表的结果才能运用这张表,这表表的情形涵盖比较 ...
-
SAP CRM 显示消息/在消息中进行导航
向用户展示消息,在任何软件中都是十分重要的. 在SAP CRM WEB UI中展示消息,不是一项很难的任务,只需要创建消息并在之后调用方法来显示它 消息类和消息号: 我在SE91中创建了如下的消息类和 ...
随机推荐
-
Linux2.6内核进程调度系列--scheduler_tick()函数3.更新普通进程的时间片
RT /** * 运行到此,说明进程是普通进程.现在开始更新普通进程的时间片. */ /* 首先递减普通进程的时间片计数器.如果用完,继续执行以下操作 */ if (!--p->time_sli ...
-
挡不住的好奇心:ASP.NET 5是如何通过XRE实现跨平台的
.NET程序员也有自己的幸福,.NET的跨平台是一种幸福,.NET的开源也是一种幸福,而更幸福的是可以通过开源的.NET了解.NET是如何一步步走向跨平台的,所以幸福是一种过程. 在.NET跨平台的进 ...
-
Canvas Path 绘制柱体
public class MainActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceS ...
-
SSH框架中一些技巧、处理办法
1.使用jstree插件时,操作成功直接刷新jstree 该页面(index.jsp)本身使用iframe框架jstree在leftFrame,操作页(add_input.jsp.add_succes ...
-
android AChartEngine源代码
昨天翻自己曾经下过的apache开源project项目,看到一个AChartEnginee看了一下自带的Demo才意识到这个东西的强大.立刻想把源代码down一份,在CSDN上有人挂5分让人下载,实在 ...
-
CSS基础知识摘要
元素分类 块级元素 什么是块级元素?在html中<div>. <p>.<h1>.<form>.<table>.<ul> 和 &l ...
-
ASP.NET CSS 小结
1.ASP.NET 引用CSS 1.Site.master里面设置webopt <webopt:bundlereferencerunat="server"path=" ...
-
ELF 动态链接 so的动态符号表(.dynsym)
静态链接中有一个专门的段叫符号表 -- ".symtab"(Symbol Table), 里面保存了所有关于该目标文件的符号的定义和引用. 动态链接中同样有一个段叫 动态符号表 - ...
-
[Swift]LeetCode949. 给定数字能组成的最大时间 | Largest Time for Given Digits
Given an array of 4 digits, return the largest 24 hour time that can be made. The smallest 24 hour t ...
-
用asp.net core 把用户访问记录优化到极致
菜菜呀,前几天做的用户空间,用户反映有时候比较慢呀 CEO,CTO,CFO于一身的CXO 是吗? 菜菜 我把你拉进用户反馈群,你解决一下呀 CEO,CTO,CFO于一身的CXO (完了,以后没清净时候 ...