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# 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#类发送消息

    如何在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. vivo鲁班RocketMQ平台的消息灰度方案

    一.方案背景 RocketMQ(以下简称MQ)作为消息中间件在事务管理,异步解耦,削峰填谷,数据同步等应用场景中有着广泛使用.当业务系统进行灰度发布时,Dubbo与HTTP的调用可以基于业界通用的灰度 ...

  9. ASP.NET MVC5+EF6+EasyUI 后台管理系统(73)-微信公众平台开发-消息管理

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

随机推荐

  1. 洛谷P1984 SDOI2008烧水问题

    P1984 [SDOI2008]烧水问题 186通过 438提交 题目提供者lych 标签数论(数学相关)模拟各省省选 难度普及+/提高 提交该题 讨论 题解 记录 最新讨论 求助! 也是醉了... ...

  2. eclipse中logcat偶尔不显示log的问题解决办法

    Android开发过程中  eclipse 经常会出现 logcat突然就是不现实log的情况.经常遇到,一直没有解决.后来解决了,记录一下. 默认的设置是error 改成verbos 问题解决.

  3. codevs 1049 棋盘染色

    题目描述 Description 有一个5×5的棋盘,上面有一些格子被染成了黑色,其他的格子都是白色,你的任务的对棋盘一些格子进行染色,使得所有的黑色格子能连成一块,并且你染色的格子数目要最少.读入一 ...

  4. hdu 3336【Count the string】(KMP)

    一道字符串匹配的题目,仅仅借此题练习一下KMP 因为这道题目就是要求用从头开始的n个字符串去匹配原来的字符串,很明显与KMP中求next的过程很相似,所以只要把能够从头开始匹配一定个数的字符串的个数加 ...

  5. CSS 之 Opacity多浏览器透明度兼容处理

    用来设定元素透明度的 Opacity 是CSS 3里的一个属性.当然现在还只有少部分浏览器支持. 不过各个浏览器都有自己的私有属性来支持,其中包括老版本的Mozilla和Safari: IE: fil ...

  6. 【MVC】关于Action返回结果类型的事儿(下)

    代码

  7. leetcode 38

    38. Count and Say The count-and-say sequence is the sequence of integers beginning as follows:1, 11, ...

  8. Android IOS WebRTC 音视频开发总结(四一)-- QQ和webrtc打洞能力pk

    很多人知道webrtc打洞能力很强,到底有多强但是不知道,比较好的方法就是跟QQ对比,但大多数公司很难模拟各种网络环境进行测试,比如联通,铁通,电信,移动,所以这次请小师妹在实验室下进行了一个比较全面 ...

  9. Nginx 开启 path_info功能

    server { listen ; server_name localhost; #charset koi8-r; #access_log logs/host.access.log main; roo ...

  10. [转]基于AnyCAD的准双曲面齿轮建模

    基于AnyCAD的准双曲面齿轮建模 作者:谨阳 (文章来源:http://www.opencascade.net/ask/?/article/6) 摘要:根据准双面齿轮的加工方法和传动特性,对准双面齿 ...