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

服务端:

public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
            TextBox.CheckForIllegalCrossThreadCalls = false;
        }
        Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
        ];
        ];
        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();
            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)
        {
            ;
            do
            {
                try
                {
                    length = client.Receive(receiveData);
                    , 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);
                }
            }
            );
        }
        private void btnSend_Click(object sender, EventArgs e)
        {
            Socket client = clients[];
            sendData = Encoding.UTF8.GetBytes(tbSend.Text);
            client.Send(sendData, , sendData.Length,SocketFlags.None);
            tbSend.Clear();
        }
    }

客户端:

public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
            TextBox.CheckForIllegalCrossThreadCalls = false;
        }

        Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
        ];
        ];
        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()
        {
            ;
            do
            {
                try
                {
                    length = socket.Receive(receiveData);
                    , 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);
                }
            }
            );
        }

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

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

好,下面问题来了:

请注意看这两行代码:

];
];

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

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

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

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

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

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

记一次Socket编程踩的坑的更多相关文章

  1. Python网络编程踩的坑

    错误:socket.error: [Errno 10013] 原因:端口号被占用 解决:换其他的端口号或者将其他应用的端口号关闭 错误:File "D:/pyworkspace/homewo ...

  2. 记Windows下初次使用dev C++进行socket编程过程

    记初次接触socket编程,在devC++使用Winsock进行socket编程的一个过程,通过在devC++创建2个项目分别是server.client程序项目,感受通过socket使client与 ...

  3. 【爬坑】Python 3.6 在 Socket 编程时出现类型错误 TypeError: a bytes-like object is required, not 'str'

    1. 问题描述 Python 3.6 在 Socket 编程时出现错误如下 Traceback (most recent call last): File "F:/share/IdeaPro ...

  4. 记一次:Windows的Socket编程学习和分析过程

    Socket编程依赖于:WS2_32.dll --- 服务端 --- .导入我们需要的函数 #incldue <windows.h> //#include<WinSock2.h> ...

  5. Linux Socket编程

    “一切皆Socket!” 话虽些许夸张,但是事实也是,现在的网络编程几乎都是用的socket. ——有感于实际编程和开源项目研究. 我们深谙信息交流的价值,那网络中进程之间如何通信,如我们每天打开浏览 ...

  6. Linux Socket编程(不限Linux)【转】

    转自:http://www.cnblogs.com/skynet/archive/2010/12/12/1903949.html “一切皆Socket!” 话虽些许夸张,但是事实也是,现在的网络编程几 ...

  7. C#版 Socket编程(最简单的Socket通信功能)

    示例程序是同步套接字程序,功能很简单,只是客户端发给服务器一条信息,服务器向客户端返回一条信息:这里只是一个简单的示例,是一个最基本的socket编程流程,在接下来的文章中,会依次记录套接字的同步和异 ...

  8. PHP socket编程需要了解的一些基本知识

    前面讲到了 fsockopen 的各种情况,其中涉及了很多其它知识,比如chunked分段传输,Keep-Alive,HTTP头字段等额外的知识,如果对这些知识一知半解,会影响对 PHP 的 sock ...

  9. SOCKet 编程 简介

    “一切皆Socket!” 话虽些许夸张,但是事实也是,现在的网络编程几乎都是用的socket. ——有感于实际编程和开源项目研究. 我们深谙信息交流的价值,那网络中进程之间如何通信,如我们每天打开浏览 ...

随机推荐

  1. returning into 语句

    returning into 语句用于执行完语句后返回的值,具体返回执行之前或者之后的结果,多用于在存储过程中 如下所述:delete语句的returning into语句返回的是delete之前的结 ...

  2. 10.N个整数中查找是否相加为K[深度搜索]

    /*摘自书本,这种算法很绕!*/ #include <iostream> using namespace std; ,,,}; ; bool dfs(int i,int sum) { if ...

  3. *C语言的小技巧

    计算数组长度 ,,,,}; int Length=sizeof(a)/sizeof(int); 交换a和b的值,不借用辅助变量 a=a+b; b=a-b; a=a-b; 将0-9的字符转化为整数 '; ...

  4. hdu 1205 吃糖果【鸽巢原理】

    题目 这道题不难,看别人博客的时候发现大家都说用鸽巢原理,这是个什么鬼,于是乎百度之. 1.把某种糖果看做隔板,如果某种糖果有n个,那么就有n+1块区域,至少需要n-1块其他种糖果才能使得所有隔板不挨 ...

  5. hdu 2153 仙人球的残影

    题目 这道题可以有两种写法: 第一种:找规律,如下: #include <stdio.h> int main() { int n,i,j,res; while (scanf("% ...

  6. [FMX]将 Android 程序切换到后台及从后台切换到前台实现

    有时候,我们需要将自己的Android程序切换到后台运行,在必要时,将其切换到前台运行.下面提供了一种实现方式,首先需要引用三个单元:   1 uses Androidapi.JNI.App,Andr ...

  7. Thread in depth 3:Synchronization

    Synchronization means multi threads access the same resource (data, variable ,etc) should not cause ...

  8. Leader Election 选举算法

    今天讲一讲分布式系统中必不可少的选举算法. leader 就是一堆服务器中的协调者,某一个时刻只能有一个leader且所有服务器都承认这个leader. leader election就是在一组进程中 ...

  9. 微信公众平台如何与Web App结合?

    Web App简而言之就是为移动平台而优化的网页,它可以表现得和原生应用一样,并且克服了原生应用一些固有的缺点.一般而言Web App最大的入口是浏览器,但现在微信公众平台作为新兴的平台,结合其内置浏 ...

  10. HttpClient4.X发送Get请求的url参数拼接

    HttpClient4.X发送Get请求的参数拼接 使用httpClient发送get请求时,请求参数可以以?key=val&key1=val1的拼接到url后面. 但是请求参数较多时,这种方 ...