麻豆小视频在线观看_中文黄色一级片_久久久成人精品_成片免费观看视频大全_午夜精品久久久久久久99热浪潮_成人一区二区三区四区

首頁 > 編程 > C# > 正文

如何應用C#實現UDP的分包組包

2020-01-24 03:27:29
字體:
來源:轉載
供稿:網友

場景介紹
如果需要使用UDP傳輸較大數據,例如傳輸10M的圖片,這突破了UDP的設計原則。UDP的設計是基于"datagram",也就是它假設你發送的每個數據包都能包含在單一的包內。并且設定UDP數據包的最大長度受基礎網絡協議的限制。

UDP數據包的理論最大長度限制是 65535 bytes,這包含 8 bytes 數據包頭和 65527 bytes 數據。但如果基于IPv4網絡傳輸,則還需減去 20 bytes 的IP數據包頭。
則單一的UDP數據包可傳輸的數據最大長度為:

則單一的UDP數據包可傳輸的數據最大長度為:

MaxUdpDataLength = 65535 - 8 - 20 = 65507 bytes

這就需要實現UDP包的分包傳輸和接收組包功能。

分包功能

復制代碼 代碼如下:

/// <summary>
   /// UDP數據包分割器
   /// </summary>
   public static class UdpPacketSplitter
   {
     /// <summary>
     /// 分割UDP數據包
     /// </summary>
     /// <param name="sequence">UDP數據包所持有的序號</param>
     /// <param name="datagram">被分割的UDP數據包</param>
     /// <param name="chunkLength">分割塊的長度</param>
     /// <returns>
     /// 分割后的UDP數據包列表
     /// </returns>
     public static ICollection<UdpPacket> Split(long sequence, byte[] datagram, int chunkLength)
     {
       if (datagram == null)
         throw new ArgumentNullException("datagram");

       List<UdpPacket> packets = new List<UdpPacket>();

       int chunks = datagram.Length / chunkLength;
       int remainder = datagram.Length % chunkLength;
       int total = chunks;
       if (remainder > 0) total++;

       for (int i = 1; i <= chunks; i++)
       {
         byte[] chunk = new byte[chunkLength];
         Buffer.BlockCopy(datagram, (i - 1) * chunkLength, chunk, 0, chunkLength);
         packets.Add(new UdpPacket(sequence, total, i, chunk, chunkLength));
       }
       if (remainder > 0)
       {
         int length = datagram.Length - (chunkLength * chunks);
         byte[] chunk = new byte[length];
         Buffer.BlockCopy(datagram, chunkLength * chunks, chunk, 0, length);
         packets.Add(new UdpPacket(sequence, total, total, chunk, length));
       }

       return packets;
     }
   }

發送分包
復制代碼 代碼如下:

private void WorkThread()
 {
   while (IsRunning)
   {
     waiter.WaitOne();
     waiter.Reset();

     while (queue.Count > 0)
     {
       StreamPacket packet = null;
       if (queue.TryDequeue(out packet))
       {
         RtpPacket rtpPacket = RtpPacket.FromImage(
           RtpPayloadType.JPEG,
           packet.SequenceNumber,
           (long)Epoch.GetDateTimeTotalMillisecondsByYesterday(packet.Timestamp),
           packet.Frame);

         // max UDP packet length limited to 65,535 bytes
         byte[] datagram = rtpPacket.ToArray();
         packet.Frame.Dispose();

         // split udp packet to many packets
         // to reduce the size to 65507 limit by underlying IPv4 protocol
         ICollection<UdpPacket> udpPackets
           = UdpPacketSplitter.Split(
             packet.SequenceNumber,
             datagram,
             65507 - UdpPacket.HeaderSize);
         foreach (var udpPacket in udpPackets)
         {
           byte[] udpPacketDatagram = udpPacket.ToArray();
           // async sending
           udpClient.BeginSend(
             udpPacketDatagram, udpPacketDatagram.Length,
             packet.Destination.Address,
             packet.Destination.Port,
             SendCompleted, udpClient);
         }
       }
     }
   }
 }

接收組包功能
復制代碼 代碼如下:

private void OnDatagramReceived(object sender, UdpDatagramReceivedEventArgs<byte[]> e)
     {
       try
       {
         UdpPacket udpPacket = UdpPacket.FromArray(e.Datagram);

         if (udpPacket.Total == 1)
         {
           RtpPacket packet = new RtpPacket(udpPacket.Payload, udpPacket.PayloadSize);
           Bitmap bitmap = packet.ToBitmap();
           RaiseNewFrameEvent(
             bitmap, Epoch.GetDateTimeByYesterdayTotalMilliseconds(packet.Timestamp));
         }
         else
         {
           // rearrange packets to one packet
           if (packetCache.ContainsKey(udpPacket.Sequence))
           {
             List<UdpPacket> udpPackets = null;
             if (packetCache.TryGetValue(udpPacket.Sequence, out udpPackets))
             {
               udpPackets.Add(udpPacket);

               if (udpPackets.Count == udpPacket.Total)
               {
                 packetCache.TryRemove(udpPacket.Sequence, out udpPackets);

                 udpPackets = udpPackets.OrderBy(u => u.Order).ToList();
                 int rtpPacketLength = udpPackets.Sum(u => u.PayloadSize);
                 int maxPacketLength = udpPackets.Select(u => u.PayloadSize).Max();

                 byte[] rtpPacket = new byte[rtpPacketLength];
                 foreach (var item in udpPackets)
                 {
                   Buffer.BlockCopy(
                     item.Payload, 0, rtpPacket,
                     (item.Order - 1) * maxPacketLength, item.PayloadSize);
                 }

                 RtpPacket packet = new RtpPacket(rtpPacket, rtpPacket.Length);
                 Bitmap bitmap = packet.ToBitmap();
                 RaiseNewFrameEvent(
                   bitmap,
                   Epoch.GetDateTimeByYesterdayTotalMilliseconds(packet.Timestamp));

                 packetCache.Clear();
               }
             }
           }
           else
           {
             List<UdpPacket> udpPackets = new List<UdpPacket>();
             udpPackets.Add(udpPacket);
             packetCache.AddOrUpdate(
               udpPacket.Sequence,
               udpPackets, (k, v) => { return udpPackets; });
           }
         }
       }
       catch (Exception ex)
       {
         RaiseVideoSourceExceptionEvent(ex.Message);
       }
     }

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 一级在线免费 | 午夜视频色 | 黄色羞羞视频在线观看 | 欧美成a人片在线观看久 | 欧美日本一 | 深夜毛片免费看 | www.99久| 久久久久在线观看 | 国产一区二区三区高清 | 经典三级av在线 | 中文字幕在线播放一区 | 久久久久一区二区三区四区五区 | 九九久久视频 | 久久经典国产视频 | 免费a级黄色片 | 久久精品女人天堂av | 日韩精品中文字幕在线播放 | 欧美人人干 | 黄网站免费观看视频 | 精品中文字幕久久久久四十五十骆 | 成人毛片网站 | 九九精品视频观看 | 99视频有精品 | 国语自产免费精品视频在 | 91毛片网站 | 午夜小电影 | 国产成人精品区一区二区不卡 | 日本aaaa片毛片免费观看视频 | 国产一级性生活视频 | 一级色毛片| 欧美三级美国一级 | 亚洲福利视 | av免费在线观看免费 | 日本黄色一级视频 | 亚洲成人免费电影 | 亚洲一区成人 | 韩国一级免费视频 | 欧美亚洲综合在线 | 国产亚洲精品久久久久久久久久 | 国产免费资源 | 超污视频在线看 |