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.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 ...
- vivo鲁班RocketMQ平台的消息灰度方案
一.方案背景 RocketMQ(以下简称MQ)作为消息中间件在事务管理,异步解耦,削峰填谷,数据同步等应用场景中有着广泛使用.当业务系统进行灰度发布时,Dubbo与HTTP的调用可以基于业界通用的灰度 ...
- ASP.NET MVC5+EF6+EasyUI 后台管理系统(73)-微信公众平台开发-消息管理
系列目录 前言 回顾上一节,我们熟悉的了解了消息的请求和响应,这一节我们来建立数据库的表,表的设计蛮复杂 你也可以按自己所分析的情形结构来建表 必须非常熟悉表的结果才能运用这张表,这表表的情形涵盖比较 ...
随机推荐
- 洛谷P1984 SDOI2008烧水问题
P1984 [SDOI2008]烧水问题 186通过 438提交 题目提供者lych 标签数论(数学相关)模拟各省省选 难度普及+/提高 提交该题 讨论 题解 记录 最新讨论 求助! 也是醉了... ...
- eclipse中logcat偶尔不显示log的问题解决办法
Android开发过程中 eclipse 经常会出现 logcat突然就是不现实log的情况.经常遇到,一直没有解决.后来解决了,记录一下. 默认的设置是error 改成verbos 问题解决.
- codevs 1049 棋盘染色
题目描述 Description 有一个5×5的棋盘,上面有一些格子被染成了黑色,其他的格子都是白色,你的任务的对棋盘一些格子进行染色,使得所有的黑色格子能连成一块,并且你染色的格子数目要最少.读入一 ...
- hdu 3336【Count the string】(KMP)
一道字符串匹配的题目,仅仅借此题练习一下KMP 因为这道题目就是要求用从头开始的n个字符串去匹配原来的字符串,很明显与KMP中求next的过程很相似,所以只要把能够从头开始匹配一定个数的字符串的个数加 ...
- CSS 之 Opacity多浏览器透明度兼容处理
用来设定元素透明度的 Opacity 是CSS 3里的一个属性.当然现在还只有少部分浏览器支持. 不过各个浏览器都有自己的私有属性来支持,其中包括老版本的Mozilla和Safari: IE: fil ...
- 【MVC】关于Action返回结果类型的事儿(下)
代码
- leetcode 38
38. Count and Say The count-and-say sequence is the sequence of integers beginning as follows:1, 11, ...
- Android IOS WebRTC 音视频开发总结(四一)-- QQ和webrtc打洞能力pk
很多人知道webrtc打洞能力很强,到底有多强但是不知道,比较好的方法就是跟QQ对比,但大多数公司很难模拟各种网络环境进行测试,比如联通,铁通,电信,移动,所以这次请小师妹在实验室下进行了一个比较全面 ...
- Nginx 开启 path_info功能
server { listen ; server_name localhost; #charset koi8-r; #access_log logs/host.access.log main; roo ...
- [转]基于AnyCAD的准双曲面齿轮建模
基于AnyCAD的准双曲面齿轮建模 作者:谨阳 (文章来源:http://www.opencascade.net/ask/?/article/6) 摘要:根据准双面齿轮的加工方法和传动特性,对准双面齿 ...