Socket的重要性

     Socket不仅在聊天应用程序中需要用到,而且对于学习Asp.net也非常有帮助。你懂得!
     实现服务端和客户端互发信息,以及服务端向多个客户端发送信息。
     服务端代码1:
        public Form1()
        {
            InitializeComponent();
            Control.CheckForIllegalCrossThreadCalls = false;
        }

        //创建监听用的socket
        private void btnStartMonitor_Click(object sender, EventArgs e)
        {
            //监听的ip地址
            IPAddress ip = IPAddress.Parse(txtIP.Text);
            //监听的ip地址和端口号
            IPEndPoint point = new IPEndPoint(ip, int.Parse(txtPortNumber.Text));
            //监听用的socket
            Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);

            //通常每个套接字地址(协议/网络地址/端口)只允许使用一次
            //绑定监听的ip地址和端口号
            socket.Bind(point);
            //监听端口   监听队列的长度:同一个时间等待的个数。
            socket.Listen(10);
            //点击之后,禁用掉按钮
            btnStartMonitor.Enabled = false;
            //提示用户程序开始运行了
            ShowMsg("开始运行了...");

            //监听端口,如果有客户端连接,创建通信用socket
            //阻塞窗体的运行
            //Socket connSocket = socket.Accept();
            //用线程来解决窗体假死的问题
            Thread th = new Thread(Listen);
            th.IsBackground = true;//后台线程
            th.Start(socket);
        }

        //监听是否有客户端连接
        void Listen(object o)
        {
            //监听用的socket
            Socket socket = o as Socket;
            //监听端口
            Socket connSocket = socket.Accept();
            //显示远程连接的IP地址和端口号
            ShowMsg(connSocket.RemoteEndPoint.ToString());
        }

        //在文本框显示信息
        void ShowMsg(string msg)
        {
            txtShowMessage.AppendText(msg + "\r\n");
        }
     以上代码只能让一个客户端连接服务端,弊端。
     服务端代码2完善——循环接收客户端连接
        //监听是否有客户端连接
       void Listen(object o)
        {
            //监听用的socket
            Socket socket = o as Socket;

            //可以不停地接受客户端的连接
            while (true)
            {
                //负责通信用的socket
                Socket connSocket = socket.Accept();
                //显示远程连接的IP地址和端口号
                ShowMsg(connSocket.RemoteEndPoint.ToString());
                //本机的IP地址和端口号
                //connSocket.LocalEndPoint

                //接收数据
                byte[] buffer = new byte[1024 * 1023];
                //接收数据,把数据发到buffer中
                //num是实际接收到的字节个数
                int num=connSocket.Receive(buffer);
                //把实际有效的字节转换成字符串
                string str = Encoding.UTF8.GetString(buffer, 0, num);
                ShowMsg(str);
            }
        }
     服务端代码3完善——循环接收消息:
        //监听是否有客户端连接
        void Listen(object o)
        {
            //监听用的socket
            Socket socket = o as Socket;

            //可以不停地接受客户端的连接
            while (true)
            {
                //负责通信用的socket
                Socket connSocket = socket.Accept();
                //显示远程连接的IP地址和端口号
                ShowMsg(connSocket.RemoteEndPoint.ToString());
                //本机的IP地址和端口号
                //connSocket.LocalEndPoint


//循环接收消息
while(true)
{
                //接收数据
                byte[] buffer = new byte[1024 * 1023];
                //接收数据,把数据发到buffer中
                //num是实际接收到的字节个数
                int num=connSocket.Receive(buffer);
                //把实际有效的字节转换成字符串
                string str = Encoding.UTF8.GetString(buffer, 0, num);
                ShowMsg(str);
}
            }
        }
     服务端代码4完善——循环多个客户端并接收信息:
        //监听是否有客户端连接
        void Listen(object o)
        {
            //监听用的socket
            Socket socket = o as Socket;

            //可以不停地接受客户端的连接
            while (true)
            {
                //负责通信用的socket
                Socket connSocket = socket.Accept();
                //显示远程连接的IP地址和端口号
                ShowMsg(connSocket.RemoteEndPoint.ToString());
                //本机的IP地址和端口号
                //connSocket.LocalEndPoint

                //线程接收信息
                Thread th = new Thread(RecMsg);
                th.IsBackground = true;
                th.Start(connSocket);            
}
        }

        //接收信息
        void RecMsg(object o)
        {
            //通信用的socket
            Socket connSocket = o as Socket;
            //可以不停地接受客户端的内容
            while (true)
            {
                //接受数据
                byte[] buffer = new byte[1024 * 1024];
                //接受数据,把数据发到buffer中
                //num就是实际接受到的字节个数
                int num = connSocket.Receive(buffer);

//当telnet关闭,会不停的发送空(0)字节

                //把实际有效的字节转化成字符串
                string str = Encoding.UTF8.GetString(buffer, 0, num);
                ShowMsg(connSocket.RemoteEndPoint.ToString() + ":" + str);
                }
            }
        }
     服务端代码5完善——限制客户端连接数:
        //限制连接客户端的个数
        int count=0;
        void Listen(object o)
        {
            //监听用的socket
            Socket socket = o as Socket;

            //可以不停地接受客户端的连接
            while (count<1000)
            {
                //负责通信用的socket
                Socket connSocket = socket.Accept();
                //显示远程连接的IP地址和端口号
                ShowMsg(connSocket.RemoteEndPoint.ToString());
                //本机的IP地址和端口号
                //connSocket.LocalEndPoint

                //线程接收信息
                Thread th = new Thread(RecMsg);
                th.IsBackground = true;
                th.Start(connSocket);
}
        }
     服务端代码6完善——解决“当tenet关闭,会不停地发送空(0)字节”:
        //接收信息
        void RecMsg(object o)
        {
            //通信用的socket
            Socket connSocket = o as Socket;
            //可以不停地接受客户端的内容
            while (true)
            {
                //接受数据
                byte[] buffer = new byte[1024 * 1024];
                //接受数据,把数据发到buffer中
                //num就是实际接受到的字节个数
                int num = connSocket.Receive(buffer);

//当telnet关闭,会不停的发送空(0)字节
                    //当telnet关闭,会不停的发送(0)个字节  客户端断开
                    if (num == 0)
                    {
                        ShowMsg(connSocket.RemoteEndPoint.ToString() + ":退出");
                        connSocket.Shutdown(SocketShutdown.Receive);
                        connSocket.Close();
                        break;
                    }
                //把实际有效的字节转化成字符串
                string str = Encoding.UTF8.GetString(buffer, 0, num);
                ShowMsg(connSocket.RemoteEndPoint.ToString() + ":" + str);
                }
            }
        }
     客户端代码1:
        public Form1()
        {
            InitializeComponent();
            Control.CheckForIllegalCrossThreadCalls = false;
        }

        //连接服务端和发送信息都要用到同一个socket,将其定义到外边
        Socket socket;
        //连接服务端
        private void btnConnect_Click(object sender, EventArgs e)
        {
            //客户端连接服务器的ip地址
            IPAddress ip = IPAddress.Parse(txtIP.Text);
            //客户端连接的服务器的ip地址和端口号
            IPEndPoint point = new IPEndPoint(ip, int.Parse(txtPortNumber.Text));
            //创建连接使用的socket
socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
            try
            {
                //连接聊天服务器
                //服务器没开报错
                //由于目标计算机积极拒绝,无法连接。
                socket.Connect(point);
                ShowMsg("连接成功");
            }
            catch (Exception ex)
            {
                ShowMsg(ex.Message);
            }
        }

        void ShowMsg(string msg)
        {
            txtShowMessage.AppendText(msg + "\r\n");
        }

        //发送消息
        private void btnSendMessage_Click(object sender, EventArgs e)
        {
            if (socket != null)
            {
                byte[] buffer = Encoding.UTF8.GetBytes(txtInputMessage.Text);
                socket.Send(buffer);
            }
            else
            {
                MessageBox.Show("请先连接");
            }
        }
     客户端代码7完善——服务端知道客户端关闭的消息:
        //接收信息
        void RecMsg(object o)
        {
            //通信用的socket
            Socket connSocket = o as Socket;
            //可以不停地接受客户端的内容
            while (true)
            {
                //接受数据
                byte[] buffer = new byte[1024 * 1024];
                //接受数据,把数据发到buffer中
                //num就是实际接受到的字节个数
                int num = -1;
                try
                {
        //远程主机强迫关闭了一个现有的连接。  当客户端关闭,服务端未关闭时,报错。
                    num = connSocket.Receive(buffer);
                    //当telnet关闭,会不停的发送(0)个字节  客户端断开
                    if (num == 0)
                    {
                        ShowMsg(connSocket.RemoteEndPoint.ToString() + ":退出");
                        connSocket.Shutdown(SocketShutdown.Receive);
                        connSocket.Close();
                        break;
                    }
                    //把实际有效的字节转化成字符串
                    string str = Encoding.UTF8.GetString(buffer, 0, num);
                    ShowMsg(connSocket.RemoteEndPoint.ToString() + ":" + str);
                }
                catch (Exception ex)
                {
                    ShowMsg(ex.Message);
                    //出现异常,跳出循环
                    break;
                }
            }
        }
     服务端代码8完善——服务端给客户端发送消息:
        //发送消息
        private void btnSendMessage_Click(object sender, EventArgs e)
        {
            if (connSocket!=null)
            {
                byte[] buffer = Encoding.UTF8.GetBytes(txtInputMessage.Text);
                try
                {
                    connSocket.Send(buffer);
                }
                catch (Exception ex)
                {
                    ShowMsg(ex.Message);
                }
            }
        }
     客户端代码2完善——客户端接收服务端发送的消息:
        //接收消息
        void RecMsg()
        {
            while (true)
            {
                byte[] buffer = new byte[1024 * 1024];
                try
                {
                    int num = socket.Receive(buffer);
                    string str = Encoding.UTF8.GetString(buffer, 1, num );
                    ShowMsg(str);
                }
                catch (Exception ex)
                {
                    ShowMsg(ex.Message);
                    break;
                }
            }
        }
//由于接收消息是一个死循环,需要用到线程
                //线程接受消息
                Thread th = new Thread(RecMsg);
                th.IsBackground = true;
                th.Start();
     服务端代码9完善——服务端给指定客户端发送消息:
        //存储通信用的socket字典
        Dictionary<string, Socket> dic = new Dictionary<string, Socket>();

        //限制连接客户端的个数
        int count = 0;
        //监听是否有客户端连接
        void Listen(object o)
        {
            //监听用的socket
            Socket socket = o as Socket;

            //可以不停地接受客户端的连接
            while (count < 100)
            {
                count++;
                //监听端口,如果有客户端连接,创建通信用socket
                Socket connSocket = socket.Accept();
                //显示远程连接的IP地址和端口号
                ShowMsg(connSocket.RemoteEndPoint.ToString());
                //本机的IP地址和端口号
                //connSocket.LocalEndPoint

                //保存当前登录的客户端的socket
                string ipport = connSocket.RemoteEndPoint.ToString();
                cboUser.Items.Add(ipport);
                dic.Add(ipport, connSocket);

                //线程接收信息
                Thread th = new Thread(RecMsg);
                th.IsBackground = true;
                th.Start(connSocket);
            }
        }

        //发送消息
        private void btnSendMessage_Click(object sender, EventArgs e)
        {
            //获取下拉框中选择的ip和端口号
            string ipport = cboUser.Text;
            if (!string.IsNullOrEmpty(ipport))
            {
                byte[] buffer = Encoding.UTF8.GetBytes(txtInputMessage.Text);
                try
                {
                    //根据key获得socket
                    Socket connSocket = dic[ipport];
                    connSocket.Send(buffer);
                }
                catch (Exception ex)
                {
                    ShowMsg(ex.Message);
                }
            }
            else
            {
                MessageBox.Show("请选择客户端");
            }
        }
     由于还有文件发送、消息震动等功能,代码太多。

     附上代码完整资源

     备注:写于2013年9月21日

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值