C# socket 实现消息中心向消息平台 转发消息
公司用到,直接粘代码了
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.Service; using Jinher.AMP.SNS.Chat.Utility; namespace Jinher.AMP.SNS.Chat.SocketManager { /// <summary> /// 消息服务基类 /// </summary> public abstract class MessageCenter { /// <summary> /// 表示是否连接上 /// </summary> public static bool IsConnected { get; protected set; } /// <summary> /// 表示是否进行第一次握手协议 /// </summary> public static bool IsHandStake { get; protected set; } /// <summary> /// 表示是否注册APP /// </summary> public static bool IsRegisterApp { get; protected set; } private static string _ip = string.Empty; ; private static object lockObject = new object(); static List<byte[]> byteList = new List<byte[]>(); //通知 private static SocketAsyncEventArgsPool pool = null; private static BufferManager m_bufferManager = null; /// <summary> /// 线程休眠时间(毫秒) /// </summary> ; //连接对象 private static Socket _socket = null; static MessageCenter() { //初始化,获取Host和Port _ip = System.Configuration.ConfigurationManager.AppSettings["serverip"]; _port = int.Parse(System.Configuration.ConfigurationManager.AppSettings["port"]); "; threadSleepTime = int.Parse(timer); //初始化连接对象 InitSocket(); } /// <summary> /// 启动消息中心 /// </summary> public static void Run() { try { LogHelper.WriteLog("启动服务"); //第一次连接服务器 Connect(); //保持连接 KeepConnect(); //推送消息队列 //SendAsync(); client.OnMsgReceiveed += client_OnMsgReceiveed; } catch (SocketException exception) { //socket出错 LogHelper.WriteLog("出错", exception); } catch (Exception ex) { //未知错误 } finally { //重置连接 //ResetConnect(); } } #region 连接服务器 /// <summary> /// 连接服务器 /// </summary> protected static async 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 static void HandShakeCmdOp() { if (!IsHandStake) { LogHelper.WriteLog("开始第一次握手"); ); SendByInit(pmsMessage.HandShakePacket(handBytes)); IsHandStake = true; LogHelper.WriteLog("成功握手"); } } /// <summary> /// 注册app /// </summary> private static void RegisterApp() { // 一级命令:XNS_ROUTER //二级命令:REGISTER_SOCIAL_APP //并且规定AppId:99999999 LogHelper.WriteLog("开始注册APP"); string rapp = "<xns=XNS_ROUTER><cmd=REGISTER_SOCIAL_APP><appid=99999999>\0"; //rapp += "<clienttype=3><companyid=0><userid=0>\0"; byte[] bytes = Encoding.Default.GetBytes(rapp); _socket.Send(pmsMessage.RegisterPacket(bytes)); AccpetOne(); IsRegisterApp = true; LogHelper.WriteLog("注册成功"); } /// <summary> /// 初始化连接对象 /// </summary> private static 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) { lock (lockObject) { if (_socket != null && IsConnected && _socket.Connected) { SendByInit(buffer); } } } else { byteList.Add(buffer); } } catch (Exception) { //重发一次 ) { //表示失败重新发送 SendAsync(buffer, count + ); } } finally { } })); } /// <summary> /// 发送数据() /// </summary> /// <param name="buffer"></param> private static void SendByInit(byte[] buffer) { _socket.BeginSend(buffer, , buffer.Length, SocketFlags.None, new AsyncCallback(SendComplated), _socket); } /// <summary> /// 发送消息回调函数 /// </summary> /// <param name="ar"></param> private static 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 static void AccpetOne() { LogHelper.WriteLog("开始建立接受消息(方式一)"); pool = ); m_bufferManager = * * , * ); m_bufferManager.InitBuffer(); // 预先分配一个对象池 SocketAsyncEventArgs readWriteEventArg; ; i < ; i++) { //初始化 SocketAsyncEventArgs readWriteEventArg = new SocketAsyncEventArgs(); readWriteEventArg.Completed += new EventHandler<SocketAsyncEventArgs>(Receive_Completed); readWriteEventArg.UserToken = new AsyncUserToken(); // 从缓冲池分配一个字节缓冲区SocketAsyncEventArg对象 m_bufferManager.SetBuffer(readWriteEventArg); // add SocketAsyncEventArg to the pool pool.Push(readWriteEventArg); } //取出 监视 Receive(); } private static void Receive() { SocketAsyncEventArgs readEventArgs = null; readEventArgs = pool.Pop(); if (readEventArgs != null) { ((AsyncUserToken)readEventArgs.UserToken).Socket = _socket; _socket.ReceiveAsync(readEventArgs); } } private static void Receive_Completed(object sender, SocketAsyncEventArgs e) { ReceiveAsync(e); } private static 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); , e.BytesTransferred); //Console.WriteLine(tmp); //通知 #region 收到消息平台发送来的消息 //client.Send(); ) { ProcessAccpet(e.Buffer, e.BytesTransferred); } #endregion //回收 pool.Push(e); m_bufferManager.FreeBuffer(e); m_bufferManager.SetBuffer(e); Receive(); } } #endregion #region 接受消息(方式二) private static void AccpetTo() { LogHelper.WriteLog("开始建立接受消息(方式二)"); ; i < ; i++) { Task.Factory.StartNew(new Action(() => { try { * ]; while (true) { int r = _socket.Receive(bytes); //通知 #region 收到消息平台发送来的消息 //client.Send(); ) { //ClientSend((byte[])bytes.Clone(), r); ProcessAccpet((byte[])bytes.Clone(), r); } #endregion //string txt = Encoding.UTF8.GetString(bytes, 0, r); //Console.WriteLine(txt); bytes = * ]; } } 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 static void ProcessAccpet(byte[] p, int r) { //加入日志 //还没有进行握手,发送来的是握手包 if (IsHandStake) { //返回去握手命令 } ) { LogHelper.WriteLog("收到正常消息"); var result = pmsMessage.ReversePacketMessage(p, r); } else { LogHelper.WriteLog(string.Format("收到心跳包")); } } #endregion #region 保持连接 /// <summary> /// 重置连接 /// </summary> private static void ResetConnect() { LogHelper.WriteLog("=========重新连接=============="); lock (lockObject) { IsConnected = false; _socket = null; InitSocket(); } Connect(); } private static Timer timer = null; /// <summary> /// 保持连接 /// </summary> private static void KeepConnect() { timer = new Timer(new TimerCallback((o) => { //发送心跳包 SendAsync(pmsMessage.GetHeardbeat()); LogHelper.WriteLog("已经发送心跳包"); }), , , , ), , , , )); } #endregion #region 通知相关 private static ChatClient client = ChatClient.Instance(ConfigurationManager.AppSettings["notificationUri"]); static PublishWebMessage pmsMessage = new PublishWebMessage(); /// <summary> /// 请求发送消息 /// </summary> /// <param name="msg"></param> protected static void client_OnMsgReceiveed(WebMessageDTO msg) { byte[] bytes = pmsMessage.PacketMessage(msg); SendAsync(bytes); } private static void ClientSend(byte[] buffer, int length) { MessageBase message = pmsMessage.ReversePacketMessage(buffer, length); } #endregion } }
C# socket 实现消息中心向消息平台 转发消息的更多相关文章
- C# socket 实现消息中心向消息平台 转发消息 (修改)
using System; using System.Collections.Generic; using System.Configuration; using System.Linq; using ...
- 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 ...
- vivo鲁班RocketMQ平台的消息灰度方案
一.方案背景 RocketMQ(以下简称MQ)作为消息中间件在事务管理,异步解耦,削峰填谷,数据同步等应用场景中有着广泛使用.当业务系统进行灰度发布时,Dubbo与HTTP的调用可以基于业界通用的灰度 ...
- ASP.NET MVC5+EF6+EasyUI 后台管理系统(73)-微信公众平台开发-消息管理
系列目录 前言 回顾上一节,我们熟悉的了解了消息的请求和响应,这一节我们来建立数据库的表,表的设计蛮复杂 你也可以按自己所分析的情形结构来建表 必须非常熟悉表的结果才能运用这张表,这表表的情形涵盖比较 ...
随机推荐
- [设计模式 3] 用设计模式的眼光看MVC框架
导读:之前一直在区分MVC和设计模式的区别,但是,既然有些人认为MVC是一种设计模式,那么它们之间肯定是有共通之处的.所以,本篇博客,就用设计模式的眼光来看MVC框架.仅是本人对于MVC的粗鄙看法,还 ...
- 【MySQL】frm文件解析
官网说明:http://dev.mysql.com/doc/internals/en/frm-file-format.html frm是MySQL表结构定义文件,通常frm文件是不会损坏的,但是如果出 ...
- 常见S1信令交互流程
0. S1 Setup
- C puzzles详解【16-20题】
第十六题 The following is a small C program split across files. What do you expect the output to be, whe ...
- dedecms 调用channel子栏目的id问题
dedecms 说明文档:http://www.dedecms.com/archives/templethelp/help/taghelp.htm {dede:channel type='son' t ...
- Dockpanel的控件加载问题
1. 正确加载模式:panel.ControlContainer.Controls.Add(control); 如果用panel.Controls.Add(control);则可能出现模块发生位移问题 ...
- c# 加密/解密 哈希
DES一共就有4个参数参与运作:明文.密文.密钥.向量.其中这4者的关系可以理解为: 密文=明文+密钥+向量: 明文=密文-密钥-向量: 为什么要向量这个参数呢?因为如果有一篇文章,有几个词重复,那么 ...
- WPF—TreeView无限极绑定集合形成树结构
1.如图所示:绑定树效果图 2.前台Xaml代码: <Window x:Class="WpfTest.MainWindow" xmlns="http://schem ...
- hive到hbase的使用
一.简单介绍 hive的元数据保存在metastore里面,真实的数据一般位于hdfs中,可以通过hql来对数据进行分析.hbase中的数据也是存放在hdfs上的,可不可以使用hive来分析hbase ...
- jquery之val()和attr("value")
1.attr("value")=原来的默认值 ,而val()=用户改变的值.