.NET中Socket编程的简单示例

以下示例程序实现简单的Socket通信,可以开多个客户端。本机测试通过,未做联机测试。

Server

using System.Net;

using System.Net.Sockets;

using System.Threading;

using System.Collections;

 

namespace MySocketServer1

{

    public partial class Form1 : Form

    {

        private IPAddress serverIP = IPAddress.Parse("127.0.0.1");//以本机作测试

        private IPEndPoint serverFullAddr;//完整终端地址

        private Socket sock;

        private System.Timers.Timer myTimer;

        private ArrayList alSock;//当建立了多个连接时用于保存连接

 

        public Form1()

        {

            InitializeComponent();

        }

 

        private void btStart_Click(object sender, EventArgs e)

        {

            serverFullAddr = new IPEndPoint(serverIP, 1000);//取端口号1000

//构造Socket对象,套接字类型为“流套接字”,指定五元组中的协议元

            sock = new Socket(AddressFamily.InterNetwork, SocketType.Stream,

ProtocolType.Tcp);

//指定五元组中的本地二元,即本地主机地址和端口号

            sock.Bind(serverFullAddr);

//监听是否有连接传入,指定挂起的连接队列的最大值为20

            sock.Listen(20);

 

            alSock = new ArrayList();

 

//构造定时器,时间间隙为1秒,即每隔一秒执行一次accept()方法,以获取连接请求队列中//第一个挂起的连接请求

            myTimer =new System.Timers.Timer(1000);

            myTimer.Elapsed +=new System.Timers.ElapsedEventHandler(myTimer_Elapsed);

            myTimer.Enabled = true;

        }

 

        private void myTimer_Elapsed(object sender, System.Timers.ElapsedEventArgs e)

        {

            myTimer.Enabled = false;

//执行accept(),当挂起队列为空时将阻塞本线程,同时由于上一语句,定时器将停止,直//至有连接传入

            Socket acceptSock = sock.Accept();

//accept()产生的Socket对象存入ArrayList

            alSock.Add(acceptSock);

// 构造Threading.Timer对象,这将导致程序另启线程。线程将执行回调函数,该委托限制//函数参数须为object型。Threading.Timer构造器的第二个参数即传入回调函数的参数;第//三个参数指定调用回调函数之前的延时,取0则立即启动;最后一个参数指定调用回调函数//的时间间隔,取0则只执行一次。

            System.Threading.Timer ti = new System.Threading.Timer(new

TimerCallback(ReceiveMsg), acceptSock, 0, 0);

            myTimer.Enabled = true;

        }

 

        private void ReceiveMsg(object obj)

        {

            Socket acceptSock = (Socket)obj;

            try

            {

                while (true)

                {

                    byte[] byteArray = new byte[100];

                    acceptSock.Receive(byteArray);//接收数据

//将字节数组转成字符串

                    string strRec = System.Text.Encoding.UTF8.GetString(byteArray);

                    if (this.rtbReceive.InvokeRequired)

                    {

                        this.rtbReceive.Invoke(new EventHandler(this.ChangeRickTextBox), new

object[] { strRec, EventArgs.Empty });

                    }

                }

            }

            catch(Exception ex)

            {

                acceptSock.Close();

                MessageBox.Show("S:Receive Message Error"+ex.Message);

            }

        }

 

        private void ChangeRickTextBox(object obj,EventArgs e)

        {

            string s = System.Convert.ToString(obj);

            this.rtbReceive.AppendText(s + Environment.NewLine);

        }

 

        private void btSend_Click(object sender, EventArgs e)

        {

            Socket sc=null;

            byte[] byteSend =

System.Text.Encoding.UTF8.GetBytes(this.tbSend.Text.ToCharArray());

            try

            {

//同时存在若干个客户端连接时,在textBox1中输入要发送的是哪个连接

                int index = int.Parse(this.textBox1.Text.Trim());

                sc = (Socket)alSock[index - 1];

//发送数据

                sc.Send(byteSend);

            }

            catch(Exception ex)

            {

if(sc != null)

{

                sc.Close();

}

                MessageBox.Show("S:Send Message Error"+ex.Message);

            }

        }

 

        private void btClose_Click(object sender, EventArgs e)

        {

            try

            {

                Application.Exit();

            }

            catch (Exception ex)

            {

                MessageBox.Show("S:Close Socket Error" + ex.Message);

            }

        }

    }

}

== == == == == == == == == == == == == == == == == == == == == == == == == == == == == == == ==

Client:

using System.Net;

using System.Net.Sockets;

using System.Threading;

 

namespace MySocketClient1

{

    public partial class Form1 : Form

    {

        private IPAddress serverIP = IPAddress.Parse("127.0.0.1");

        private IPEndPoint serverFullAddr;

        private Socket sock;

 

        public Form1()

        {

            InitializeComponent();

        }

 

        private void btConnect_Click(object sender, EventArgs e)

        {

            try

            {

                serverFullAddr = new IPEndPoint(serverIP, 1000);

                sock = new Socket(AddressFamily.InterNetwork, SocketType.Stream,

ProtocolType.Tcp);

                sock.Connect(serverFullAddr);//建立与远程主机的连接

 

//启动新线程用于接收数据

                Thread t = new Thread(new ThreadStart(ReceiveMsg));

                t.Name = "Receive Message";

//一个线程或者是后台线程或者是前台线程。后台线程与前台线程类似,区别是后台线//程不会防止进程终止。一旦属于某一进程的所有前台线程都终止,公共语言运行库就//会通过对任何仍然处于活动状态的后台线程调用 Abort 来结束该进程。

                t.IsBackground = true;

                t.Start();

            }

            catch(Exception ex)

            {

                MessageBox.Show(ex.Message);

            }

        }

 

        private void ReceiveMsg()

        {

            try

            {

                while (true)

                {

                    byte[] byteRec = new byte[100];

                    this.sock.Receive(byteRec);

                    string strRec = System.Text.Encoding.UTF8.GetString(byteRec);

                    if (this.rtbReceive.InvokeRequired)

                    {

                        this.rtbReceive.Invoke(new EventHandler(ChangeRtb), new object[]

{ strRec, EventArgs.Empty });

                    }

                }

            }

            catch(Exception ex)

            {

                MessageBox.Show("Receive Message Error"+ex.Message);

            }

        }

 

        private void ChangeRtb(object obj, EventArgs e)

        {

            string s = System.Convert.ToString(obj);

            this.rtbReceive.AppendText(s + Environment.NewLine);

        }

 

        private void btSend_Click(object sender, EventArgs e)

        {

            byte[] byteSend =

System.Text.Encoding.UTF8.GetBytes(this.tbSend.Text.ToCharArray());

            try

            {

                this.sock.Send(byteSend);

            }

            catch

            {

                MessageBox.Show("Send Message Error");

            }

        }

 

        private void btClose_Click(object sender, EventArgs e)

        {

            try

            {

                this.sock.Shutdown(SocketShutdown.Receive);

                this.sock.Close();

                Application.Exit();

            }

            catch

            {

                MessageBox.Show("Exit Error");

            }

        }

    }

}

不解之处:

Client端红色标注语句:this.sock.Shutdown(SocketShutdown.Receive),如改成

this.sock.Shutdown(SocketShutdown.Both);this.sock.Shutdown(SocketShutdown.Send);

则当点击Cloce按钮时,CPU使用率疯涨到100%,而使用this.sock.Shutdown(SocketShutdown.Receive);

或不调用Shutdown()方法则没有这个问题。难道客户端不应该用Shutdown()

 

posted @ 2006-08-13 17:19 KissKnife 阅读(3415) 评论(26)  编辑 收藏 所属分类: C#WinForm相关

  回复  引用    
#1楼 2006-09-08 16:59 | sunny[匿名] [未注册用户]
请问:

这是不是window form下的socket编程 啊?

不是asp.net下的吧
  回复  引用  查看    
#2楼 [楼主]2006-09-08 22:10 | KissKnife      
是的,winform下的...

同时,这个例子是不完善的,存在一些问题,发上来原为让大家提提意见的,所以仅可用于参考...
  回复  引用    
#3楼 2006-11-24 00:08 | fjxm [未注册用户]
this.rtbReceive.InvokeRequired

rtbReceive 这个变量哪里来啊
加下我 154595956 谢谢
  回复  引用    
#4楼 2006-11-24 21:28 | kissknife[匿名] [未注册用户]
rtbReceive只是一个RichTextBox控件..
  回复  引用    
#5楼 2006-11-24 23:23 | fjxm [未注册用户]
O 谢谢
  回复  引用    
#6楼 2006-11-25 00:39 | fjxm [未注册用户]
this.rtbReceive.Invoke(new EventHandler(this.ChangeRickTextBox), new

object[] { strRec, EventArgs.Empty });用这个方法好像得到的不是我们发送或接收的消息
  回复  引用    
#7楼 2006-11-25 09:24 | kissknife[匿名] [未注册用户]
不会啊,得到的就是服务器端发给客户端的消息,在richtextbox上显示
  回复  引用    
#8楼 2006-12-15 15:01 | wzn [未注册用户]
这个东西是不是必须运行在.net的环境上的?
  回复  引用    
#9楼 2006-12-16 08:40 | kissknife[匿名] [未注册用户]
这个东西是不是必须运行在.net的环境上的?
============================
是的.
  回复  引用    
#10楼 2007-01-04 19:27 | pupy_zs [未注册用户]
lz可否留个联系方式,方便大家讨论~~
小第是新手,入门时有不会的还要向lz请教
  回复  引用  查看    
#11楼 [楼主]2007-01-04 21:30 | KissKnife      
已写在公告里..
  回复  引用    
#12楼 2007-04-20 22:19 | 沈 [未注册用户]
你好,我按照你的方法调试通过了,但是有一个问题,客户端发送成功,服务端接收不到客户端发送的消息
  回复  引用  查看    
#13楼 [楼主]2007-04-21 14:07 | KissKnife      
@沈
应该是没有问题的,你再试试
最近也没有写这方面的东西,有问题共同探讨吧。
  回复  引用    
#14楼 2007-04-22 16:24 | 沈 [未注册用户]
我加了你公告里的QQ 怎么没有反映呢?这个QQ你现在是不是不用了啊?我有问题要请教你
  回复  引用    
#15楼 2007-04-24 10:48 | 沈 [未注册用户]
问题出在这句上
this.rtbReceive.Invoke(new EventHandler(ChangeRtb), new object[]

{ strRec, EventArgs.Empty });


在这里 strRec 是有值的 但是 作为参数传递给ChangeRtb的时候,
private void ChangeRtb(object obj, EventArgs e)

{

string s = System.Convert.ToString(obj);

this.rtbReceive.AppendText(s + Environment.NewLine);

}

这个函数获取不到传递过来的值 obj为空

  回复  引用  查看    
#16楼 [楼主]2007-04-24 20:10 | KissKnife      
@沈
抱歉,前几天中了威金,搞到现在才基本恢复正常。
还没安装VS2005,不能打开这个程序,但是我印象中肯定不存在你说的这个问题,因为是在本地测试通过后才贴出来的。
另外提一下,我用的是VS2005。
  回复  引用    
#17楼 2007-05-15 12:58 | 忍 [未注册用户]
给帮我发一份源程序吗?
谢了(jy19841023@126.com)
  回复  引用  查看    
#18楼 [楼主]2007-05-23 21:51 | KissKnife      
@忍
晕,上面就是完整的程序..
  回复  引用    
#19楼 2007-06-25 16:53 | BB [未注册用户]
try

{

//同时存在若干个客户端连接时,在textBox1中输入要发送的是哪个连接

int index = int.Parse(this.textBox1.Text.Trim());

sc = (Socket)alSock[index - 1];

//发送数据

sc.Send(byteSend);

}

总是说textBox1中输入不正确,不是应该输入IP吗?为什么呢?
  回复  引用