记一次Socket编程踩的坑

闲来无事研究了下Socket,想用它做个简单的聊天室模型,结果踩了个坑,整半天才出来,惭愧啊,先上完成的代码吧

服务端:

public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
            TextBox.CheckForIllegalCrossThreadCalls = false;
        }
        Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
        byte[] receiveData = new byte[1024];
        byte[] sendData = new byte[1024];
        List<Socket> clients = new List<Socket>();
        

        private void btnBeginListen_Click(object sender, EventArgs e)
        {
            if(socket.LocalEndPoint!=null)
            {
                return;
            }
            IPEndPoint ip = new IPEndPoint(IPAddress.Any, int.Parse(tbPort.Text));
            socket.Bind(ip);
            socket.Listen(10);
            tbReceive.Text += "Begin listening...\r\n";
            Task.Run(()=>BeginListen());
        }

     
        private void BeginListen()
        {
            while (true)
            {
                Socket client = socket.Accept();
                clients.Add(client);
                IPEndPoint clientIp = client.RemoteEndPoint as IPEndPoint;
                tbReceive.Text += clientIp.Address + ":" + clientIp.Port + " connected...\r\n";
                receiveData = Encoding.UTF8.GetBytes("welcome");
                client.Send(receiveData);
                Task.Run(() => ReceiveMsg(client));
            }
        }
        private void ReceiveMsg(Socket client)
        {
            int length = 0;
            do
            {
                try
                {
                    length = client.Receive(receiveData);
                    string stringData = Encoding.UTF8.GetString(receiveData, 0, length);
                    IPEndPoint ep = (IPEndPoint)client.RemoteEndPoint;
                    tbReceive.Text += ep.Address + ":" + ep.Port + ": " + DateTime.Now.ToString() + "\r\n";
                    tbReceive.Text += stringData + "\r\n";
                }
                catch (SocketException ex)
                {
                    tbReceive.AppendText("远程端关闭.\r\n");
                    clients.Remove(client);
                    return;
                }
                catch(Exception ex)
                {
                    tbReceive.AppendText("异常:"+ex.Message);
                }
            }
            while (length > 0);
        }
        private void btnSend_Click(object sender, EventArgs e)
        {
            Socket client = clients[0];
            sendData = Encoding.UTF8.GetBytes(tbSend.Text);
            client.Send(sendData, 0, sendData.Length,SocketFlags.None);
            tbSend.Clear();
        }
    }
View Code

客户端:

public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
            TextBox.CheckForIllegalCrossThreadCalls = false;
        }
        
        Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
        byte[] receiveData = new byte[1024];
        byte[] sendData = new byte[1024];
        private void btnConnect_Click(object sender, EventArgs e)
        {
            if(socket.LocalEndPoint!=null)
            {
                return;
            }
            IPEndPoint ip = new IPEndPoint(IPAddress.Any, int.Parse(tbLocalPort.Text));
            socket.Bind(ip);
            IPEndPoint remoteIp = new IPEndPoint(IPAddress.Parse(tbIp.Text), int.Parse(tbPort.Text));
            try
            {
                socket.Connect(remoteIp);
            }
            catch (Exception ex)
            {
                tbReceive.Text += "无法连接服务器:"+ex.Message;
            }
            Task.Run(() => ReceiveMsg());
        }
        private void ReceiveMsg()
        {
            int length = 0;
            do
            {
                try
                {
                    length = socket.Receive(receiveData);
                    string stringData = Encoding.UTF8.GetString(receiveData, 0, length);
                    IPEndPoint ep = (IPEndPoint)socket.RemoteEndPoint;
                    tbReceive.Text += ep.Address + ":" + ep.Port + ": " + DateTime.Now.ToString() + "\r\n";
                    tbReceive.Text += stringData + "\r\n";
                }
                catch (SocketException ex)
                {
                    tbReceive.AppendText("连接异常.\r\n");
                    return;
                }
                catch (Exception ex)
                {
                    tbReceive.AppendText("异常:" + ex.Message);
                }
            }
            while (length > 0);
        }

        private void btnSend_Click(object sender, EventArgs e)
        {
            sendData = Encoding.UTF8.GetBytes(tbSend.Text);
            socket.Send(sendData, 0, sendData.Length, SocketFlags.None);
            tbSend.Clear();
        }
    }
View Code

多线程写textbox的时候需要开启:TextBox.CheckForIllegalCrossThreadCalls = false;

好,下面问题来了:

请注意看这两行代码:

byte[] receiveData = new byte[1024];
byte[] sendData = new byte[1024];

开始一偷懒把这两个缓冲区用同一个数组来接收,然后坑就来了:

1.当我从服务端往客户端发了个“123”,客户端接到“123”后再往服务端发送“456”,结果服务端接收到的是“123”,再次发送“789”,服务端接收到的是“456”

2.当我从服务端往客户端发了个“123”,客户端接到“123”后再往服务端发送“4567”,结果服务端就报数组index超出范围了

想来想去都想不明白问题出在哪,百度、谷歌、StackOverFlow,都没有找到我要的答案,后来看到一篇文章问Socket的缓冲区为什么要用数组来接,突然灵光一现,也许问题就出在这个缓冲区上了,当我把数组分别定义成两个后,顺利解决问题了。

为啥怎么搜都搜不到这个问题呢,估计是没人跟我一样用同一个数组干过吧,汗。。。

具体的原理还没搞明白,等高手来回答。

转载于:https://www.cnblogs.com/uptothesky/p/5316062.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值