0004-异步编程分包

如果我们要处理客户端的每一条消息,针对与上面的粘包的问题,我们无法处理消息的,我们不知道消息到底有多长(虽说我们的案例是一样的,但实际开发中可能是不一样的),那么我们就要涉及到分包的问题了。

在上一篇的文章中我们可以看到一篇文章被接受了多次才接受完,加入客户端一下发了多篇文章?那么我们怎么才能把每一篇文章分出来呢?这就涉及到了分包。
我们在客户端每次发消息的时候我们都要将发送信息的长度加到消息的前面。由于TCP消息都是按顺序到达的,那么我们在读消息的时候,先读出消息长度(len),然后从消息长度后开始读取len个字节,那么这len个字节就是我们要的消息。
对前面的代码进行修改如下:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Net;
using System.Net.Sockets;

namespace ASYNServer
{
    class Server
    {
        private TcpListener tcpListener;
        byte[] buffer = new byte[512];
        /// <summary>
        /// 接受消息
        /// </summary>
        List<byte> receiveDataList = new List<byte>();
        public Server()
        {
            tcpListener = new TcpListener(new IPEndPoint(IPAddress.Parse("127.0.0.1"),10001));
            //开启侦听,最多只能连接20个客户端数目
            tcpListener.Start(20);
            tcpListener.BeginAcceptSocket(ClientConnect, tcpListener);
        }
        /// <summary>
        /// 客户端连接
        /// </summary>
        /// <param name="ar"></param>
        private void ClientConnect(IAsyncResult ar)
        {
            try
            {
                TcpListener listener = (TcpListener)ar.AsyncState;
                Socket client = listener.EndAcceptSocket(ar);
                Console.WriteLine(client.RemoteEndPoint.ToString() + "连接成功");
                //侦听结束后开始接收数据
                ReceiveData(client);
                //重新侦听是否有新的客户端连接
                //如果这里不写  那么只能侦听一次
                tcpListener.BeginAcceptSocket(ClientConnect, tcpListener);
            }
            catch (Exception e)
            {

                Console.WriteLine("ClientConnect"+e.Message);
            }
        }
        /// <summary>
        /// 接受数据的回调
        /// </summary>
        /// <param name="client"></param>
        private  void ReceiveData(Socket client)
        {
            SocketError socketError;
            client.BeginReceive(buffer, 0, buffer.Length, SocketFlags.None, out socketError, ReceiveCallback, client);
        }
        /// <summary>
        /// 一次接收的数据的长度
        /// </summary>
        int receLen = 0;
        private void ReceiveCallback(IAsyncResult ar)
        {
            try
            {
                SocketError socketError;
                Socket client = (Socket)ar.AsyncState;
                receLen = client.EndReceive(ar, out socketError);
                if (receLen>0)
                {
                    lock (receiveDataList)
                    {
                        //处理接收到的数据

                        OnReceiveData(client);
                    }
                    
                }else//当receLen<=0的时候表示客户端断开和服务器的连接
                {
                    Console.WriteLine(client.LocalEndPoint.ToString()+"关闭了");
                }
                //重新接受数据
                ReceiveData(client);
            }
            catch (Exception e)
            {
                Console.WriteLine("ReceiveCallback:" + e.Message + "   "+e.StackTrace);
            }
        }
        /// <summary>
        /// 接受数据
        /// </summary>
        /// <param name="socket"></param>
        private void OnReceiveData(Socket socket)
        {
            
            byte[] tempArr = new byte[receLen];
            Array.Copy(buffer, tempArr, receLen);
            receiveDataList.AddRange(tempArr);

            if (receiveDataList.Count>=4)
            {
                byte[] dataArray = receiveDataList.GetRange(0, 4).ToArray();
                int len = BitConverter.ToInt32(dataArray, 0);
                
                if (receiveDataList.Count>=len+4)
                {
                    receiveDataList.RemoveRange(0, 4);
                    dataArray = receiveDataList.GetRange(0, len).ToArray();
                    string message = Encoding.UTF8.GetString(dataArray, 0, len);
                    receiveDataList.RemoveRange(0, len);
                    Console.WriteLine("来自客户端的数据是:" + message);
                    Console.WriteLine();
                }
            }
        }
        /// <summary>
        /// 发送数据
        /// </summary>
        /// <param name="socket"></param>
        /// <param name="data"></param>
        public void SendData(Socket socket, byte[] data)
        {
            SocketError socketError;
            socket.BeginSend(data, 0, data.Length, SocketFlags.None, out socketError, SendCallBack, socket);//异步发送数据
        }
        /// <summary>
        /// 发送数据的回调
        /// </summary>
        /// <param name="ar"></param>
        private void SendCallBack(IAsyncResult ar)
        {
            SocketError socketError;
            Socket client = (Socket)ar.AsyncState;
            client.EndSend(ar, out socketError);
        }
    }
}

客户端代码:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Net;
using System.Net.Sockets;
using System.IO;

namespace ASYNClient
{
    class Client
    {
        private TcpClient tcpclient = null;
        byte[] result;
        public Client()
        {
            tcpclient = new TcpClient();
            //连接服务器
            tcpclient.BeginConnect(IPAddress.Parse("127.0.0.1"), 10001, ConnectCallback, tcpclient);
        }
        /// <summary>
        /// 连接回调
        /// </summary>
        /// <param name="ar"></param>
        private void ConnectCallback(IAsyncResult ar)
        {
            TcpClient client = (TcpClient)ar.AsyncState;
            try
            {
                if (client.Connected)
                {
                    //连接成功
                    client.EndConnect(ar);
                    //开始接受数据
                    ReceiveData(client);

                }
            }
            catch (Exception e)
            {
                Console.WriteLine("ConnectCallback:"+e.Message);
            }
        }
        /// <summary>
        /// 接收数据
        /// </summary>
        /// <param name="client"></param>
        private void ReceiveData(TcpClient client)
        {
            NetworkStream stream = client.GetStream();

            result = new byte[client.Available];
          
            try
            {
                stream.BeginRead(result, 0, result.Length, ReadCallBack, stream);
            }
            catch (Exception e)
            {
                Console.WriteLine("ReceiveData:"+e.Message );
            }
        }
        /// <summary>
        /// 接受消息回调
        /// </summary>
        /// <param name="ar"></param>
        private void ReadCallBack(IAsyncResult ar)
        {
            NetworkStream stream;
            try
            {
                stream = (NetworkStream)ar.AsyncState;
                stream.EndRead(ar);
            }
            catch (IOException e)
            {
                Console.WriteLine("远程服务器关闭");
                
            }
           
        }
        /// <summary>
        /// 发送消息
        /// </summary>
        /// <param name="data"></param>
        public void SendData(byte[] data)
        {
            if (tcpclient.Connected)
            {
                NetworkStream stream = tcpclient.GetStream();
                stream.BeginWrite(data,0,data.Length, SendCallback, stream);

            }
        }
        /// <summary>
        /// 发送消息回调
        /// </summary>
        /// <param name="ar"></param>
        private void SendCallback(IAsyncResult ar)
        {
            NetworkStream stream = (NetworkStream)ar.AsyncState;
            stream.EndWrite(ar);
        }
    }
}

客户端测试代码:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;

namespace ASYNClient
{
    class Program
    {
        static void Main(string[] args)
        {
            Client client = new Client();
            //等待客户端连接服务器后我们按下回车键模拟客户端一次性发送多条数据
            Console.ReadLine();
            for (int i = 0; i < 3; i++)
            {
                //string message = "ABCDEFGHIJKLMNOPQRSTUVWXYZ1234567890";
                string message = "燕子去了,有再来的时候;杨柳枯了,有再青的时候;桃花谢了,有再开的时 候。但是,聪明的,你告诉我,我们的日子为什么一去不复返呢?——是有人偷了他 们罢:那是谁?又藏在何处呢?是他们自己逃走了罢——如今又到了哪里呢?我不知道他们给了我多少日子,但我的手确乎(1)是渐渐空虚(2)了。在默默里算着,八千多日子已经从我手中溜去,像针尖上一滴水滴在大海里,我的日子滴在时间的流里,没有声音,也没有影子。我不禁头涔涔(3)而泪潸潸(4)了。去的尽管去了,来的尽管来着;去来的中间,又怎样地匆匆呢?早上我起来的时候,小屋里射进两三方斜斜的太阳。太阳他有脚啊,轻轻悄悄地挪移了;我也茫茫然跟着旋转。于是——洗手的时候,日子从水盆里过去;吃饭的时候,日子从饭碗里过去;默默时,便从凝然的双眼前过去。我觉察他去的匆匆了,伸出手遮挽时,他又从遮挽着的手边过去,天黑时,我躺在床上,他便伶伶俐俐(5)地从我身上跨过,从我脚边飞去了。等我睁开眼和太阳再见,这算又溜走了一日。我掩着面叹息。但是新来的日子的影儿又开始在叹息里闪过了。在逃去如飞的日子里,在千门万户的世界里的我能做些什么呢?只有徘徊(6)罢了(7),只有匆匆罢了;在八千多日的匆匆里,除徘徊外,又剩些什么呢?过去的日子如轻烟,被微风吹散了,如薄雾,被初阳蒸融了;我留着些什么痕迹呢?我何曾留着像游丝(8)样的痕迹呢?我赤裸裸来到这世界,转眼间也将赤裸裸的回去罢?但不能平的,为什么偏要白白走这一遭啊?你聪明的,告诉我,我们的日子为什么一去不复返呢?"; 
                byte[] messageData = Encoding.UTF8.GetBytes(message);
                Console.WriteLine("messageData:"+ messageData.Length);
                byte[] lenData = BitConverter.GetBytes(messageData.Length);
                byte[] data = new byte[lenData.Length+messageData.Length];
                Array.Copy(lenData, data, lenData.Length);
                Array.Copy(messageData,0,data,lenData.Length,messageData.Length);
                client.SendData(data);
            }
            Console.ReadLine();
        }
    }
}
测试结果
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,386评论 6 479
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,939评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,851评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,953评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,971评论 5 369
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,784评论 1 283
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,126评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,765评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,148评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,744评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,858评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,479评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,080评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,053评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,278评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,245评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,590评论 2 343

推荐阅读更多精彩内容

  • 计算机网络概述 网络编程的实质就是两个(或多个)设备(例如计算机)之间的数据传输。 按照计算机网络的定义,通过一定...
    蛋炒饭_By阅读 1,210评论 0 10
  • 经常想王柯上课时是什么样子的,坐的正吗?能认真听课吗?老师提问题能主动举手回答吗?好多的问题都会在脑子里出现。今天...
    王柯妈妈阅读 183评论 8 6
  • 深夜适合思考。 人们总在夜深人静的时候,辗转难眠,脑袋里充满各种天马行空的幻想。 是的,我就是睡不着。 如今烦心的...
    小陈物语阅读 193评论 0 3
  • 今天就二十岁了呀。 真是一个浪漫的开始呢。 我想,二十岁往后的日子, 应该会越来越浪漫吧。 十九岁发生了太多太多跟...
    屁桃三三阅读 437评论 3 6