所谓的潇洒

导航

UDP发送频率控制

场景一:

  用户需要在界面上看到消息的发送与回复,UDP得发送1->接收回复1->发送2->接收回复2

场景二:

  无关用户体验,但是需要大量的数据发送,可能导致接收方处理不过来而端口堵塞消息丢弃

方案:

  使用EventWaitHandle的WaitOne来暂停发送,当收到消息回复或者有消息超时时Set继续发送,代码(包含伪代码,不可编译)如下:

  ConcurrentDictionary<int, DateTime> _batchMsgID = new ConcurrentDictionary<int, DateTime>();//等待中的消息<消息唯一标识,开始等待时间>

  EventWaitHandle _waitHandler = new AutoResetEvent(false);//执行下发等待信号

  SendFunc()

  {

    for()

    {

      _batchMsgID.TryAdd();//加入等待

      UdpSendOne();//发送一条数据       

      if(_batchMsgID.Count>=10) _waitHandler.WaitOne(); //场景一,不用判断直接wait;场景二,等待数超过10条暂停发送

    }    

  }

  ReceiveFunc()

  {

    while()

    {      

      UdpReceiveOne();//收到一条数据

       if (_batchMsgID.ContainsKey(msgID))
                       {
                              DateTime time;
                               _batchMsgID.TryRemove(msgID, out time);
                               if (_batchMsgID.Count <= 10) _waitHandler.Set(); //场景一,不用判断直接set;
                       }

    }

  }

  timer_tick()

  {//消息超时处理

        try
            {
                List<int> keys = new List<int>();
                foreach(var msg in _batchMsgID)
                {
                    if ((DateTime.Now - msg.Value).TotalSeconds >= 3) keys.Add(msg.Key);
                }
                DateTime time;
                keys.ForEach(x => _batchMsgID.TryRemove(x, out time));

     if (_batchMsgID.Count <= 10) _waitHandler.Set(); //场景一,不用判断直接set;
            }
            catch (Exception ex) { LogHelper.logger.Error(ex); }

  }

posted on 2020-04-28 11:36  所谓的潇洒  阅读(1505)  评论(0编辑  收藏  举报