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)-微信公众平台开发-消息管理
系列目录 前言 回顾上一节,我们熟悉的了解了消息的请求和响应,这一节我们来建立数据库的表,表的设计蛮复杂 你也可以按自己所分析的情形结构来建表 必须非常熟悉表的结果才能运用这张表,这表表的情形涵盖比较 ...
随机推荐
- Google Protocol Buffer 协议
1. Protocol Buffers 简介 Protocol Buffers (ProtocolBuffer/ protobuf )是Google公司开发的一种数据描述语言,类似于XML能够将结构化 ...
- KMP算法模板
不懂的话推荐看这篇博客,讲的很清楚 http://blog.csdn.net/v_july_v/article/details/7041827 #include<iostream> #in ...
- noip2010提高组题解
NOIP2010提高组题解 T1:机器翻译 题目大意:顺序输入n个数,有一个队列容量为m,遇到未出现元素入队,求入队次数. AC做法:直接开1000的队列模拟过程. T2:乌龟棋 题目大意:有长度为n ...
- VS2010之MFC串口通信的编写教程
http://wenku.baidu.com/link?url=K1XPdj9Dcf2of_BsbIdbPeeZ452uJqiF-s773uQyMzV2cSaPRIq6RddQQH1zr1opqVBM ...
- web.config中sessionState节点的配置方案
web.config中sessionState节点的配置方案 web.config关于sessionState节点的配置方案,sessionState有五种模式:Custom,off,inProc,S ...
- Tomcat 7优化前及优化后的性能对比
Tomcat 7在我们日常开发.测试.生产环境都会使用到,但对于大部分开发人员来说,对其性能还是没有多大了解.本文就对它做一次性能测试,对比优化前后的性能区别. 一.运行环境 CPU: Intel(R ...
- 浅谈Bootstrap自适应功能在Web开发中的应用
随着移动端市场的强势崛起,web的开发也变得愈发复杂,对于个体开发者来说,自己开发的网站,在电脑.手机.Pad等上面都要有正常的显示以及良好的用户体验.如果每次都要自己去调整网页去匹配各个不同的客户端 ...
- .NET程序员吧需要知道的小知识——关于数据库
关于数据库 作为一个有“情怀的”(B格高一些的).NET开发工程师,需要多少知道一些这样的小故事. 哪怕仅仅当作一些扯淡的谈资. 1.文件型数据库(常见的) Access SQLite SQLSe ...
- Linux多线程编程阅读链接
1. 进程与线程的一个简单解释(阮一峰) 2. linux 多线程编程 3. Linux 的多线程编程的高效开发经验 (IBM)
- jQuery判断元素离页面顶部的高度
<script language="javascript" type="text/javascript"> jQuery(document).rea ...