C# socket 实现消息中心向消息平台 转发消息 (修改)

时间:2021-12-26 10:05:43
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 实现消息中心向消息平台 转发消息 (修改)的更多相关文章

  1. C&num; socket 实现消息中心向消息平台 转发消息

    公司用到,直接粘代码了 using System; using System.Collections.Generic; using System.Configuration; using System ...

  2. IBM MQ消息中间件jms消息中RHF2消息头的处理

    公司的技术平台在和某券商对接IBM MQ消息中间件时,发送到MQ中的消息多出了消息头信息:RHF2,造成消息的接收处理不正常.在此记录此问题的处理方式. 在IBM MQ中提供了一个参数 targetC ...

  3. 如何在MFC DLL中向C&num;类发送消息

    如何在MFC DLL中向C#类发送消息 一. 引言 由于Windows Message才是Windows平台的通用数据流通格式,故在跨语言传输数据时,Message是一个不错的选择,本文档将描述如何在 ...

  4. 微信开发——微信公众平台实现消息接收以及消息的处理(Java版)

    本文主要讲述了如何在微信公众平台实现消息接收以及消息的处理,使用java语言开发,现在把实现思路和代码整理出来分先给兄弟们,希望给他们带来帮助. 温馨提示: 这篇文章是依赖前几篇的文章的. 第一篇:微 ...

  5. Team Foundation 中的错误和事件消息

    Visual Studio Team System Team Foundation 中的错误和事件消息 Team Foundation 通过显示错误消息和事件消息来通知您操作成功以及操作失败.一部分错 ...

  6. ROS Learning-027 (提高篇-005 A Mobile Base-03) 控制移动平台 --- Twist 消息

    ROS 提高篇 之 A Mobile Base-03 - 控制移动平台 - Twist 消息 我使用的虚拟机软件:VMware Workstation 11 使用的Ubuntu系统:Ubuntu 14 ...

  7. Delphi中SendMessage使用说明(所有消息说明) good

    Delphi中SendMessage使用说明 SendMessage基础知识 函数功能:该函数将指定的消息发送到一个或多个窗口.此函数为指定的窗口调用窗口程序,直到窗口程序处理完消息再返回.而函数Po ...

  8. ASP&period;NET MVC5&plus;EF6&plus;EasyUI 后台管理系统(73)-微信公众平台开发-消息管理

    系列目录 前言 回顾上一节,我们熟悉的了解了消息的请求和响应,这一节我们来建立数据库的表,表的设计蛮复杂 你也可以按自己所分析的情形结构来建表 必须非常熟悉表的结果才能运用这张表,这表表的情形涵盖比较 ...

  9. SAP CRM 显示消息&sol;在消息中进行导航

    向用户展示消息,在任何软件中都是十分重要的. 在SAP CRM WEB UI中展示消息,不是一项很难的任务,只需要创建消息并在之后调用方法来显示它 消息类和消息号: 我在SE91中创建了如下的消息类和 ...

随机推荐

  1. Linux2&period;6内核进程调度系列--scheduler&lowbar;tick&lpar;&rpar;函数3&period;更新普通进程的时间片

    RT /** * 运行到此,说明进程是普通进程.现在开始更新普通进程的时间片. */ /* 首先递减普通进程的时间片计数器.如果用完,继续执行以下操作 */ if (!--p->time_sli ...

  2. 挡不住的好奇心:ASP&period;NET 5是如何通过XRE实现跨平台的

    .NET程序员也有自己的幸福,.NET的跨平台是一种幸福,.NET的开源也是一种幸福,而更幸福的是可以通过开源的.NET了解.NET是如何一步步走向跨平台的,所以幸福是一种过程. 在.NET跨平台的进 ...

  3. Canvas Path 绘制柱体

    public class MainActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceS ...

  4. SSH框架中一些技巧、处理办法

    1.使用jstree插件时,操作成功直接刷新jstree 该页面(index.jsp)本身使用iframe框架jstree在leftFrame,操作页(add_input.jsp.add_succes ...

  5. android AChartEngine源代码

    昨天翻自己曾经下过的apache开源project项目,看到一个AChartEnginee看了一下自带的Demo才意识到这个东西的强大.立刻想把源代码down一份,在CSDN上有人挂5分让人下载,实在 ...

  6. CSS基础知识摘要

    元素分类 块级元素 什么是块级元素?在html中<div>. <p>.<h1>.<form>.<table>.<ul> 和 &l ...

  7. ASP&period;NET CSS 小结

    1.ASP.NET 引用CSS 1.Site.master里面设置webopt <webopt:bundlereferencerunat="server"path=&quot ...

  8. ELF 动态链接 so的动态符号表(&period;dynsym)

    静态链接中有一个专门的段叫符号表 -- ".symtab"(Symbol Table), 里面保存了所有关于该目标文件的符号的定义和引用. 动态链接中同样有一个段叫 动态符号表 - ...

  9. &lbrack;Swift&rsqb;LeetCode949&period; 给定数字能组成的最大时间 &vert; 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 ...

  10. 用asp&period;net core 把用户访问记录优化到极致

    菜菜呀,前几天做的用户空间,用户反映有时候比较慢呀 CEO,CTO,CFO于一身的CXO 是吗? 菜菜 我把你拉进用户反馈群,你解决一下呀 CEO,CTO,CFO于一身的CXO (完了,以后没清净时候 ...