基于C#的Socket开发快速入门
2009年11月21日
首先从原理上解释一下采用Socket接口的网络通讯,这里以最常用的C/S模式作为范例,首先,服务端有一个进程(或多个进程)在指定的端口等待客户来 连接,服务程序等待客户的连接信息,一旦连接上之后,就可以按设计的数据交换方法和格式进行数据传输。客户端在需要的时刻发出向服务端的连接请求。这里为 了便于理解,提到了一些调用及其大致的功能。使用socket调用后,仅产生了一个可以使用的socket描述符,这时还不能进行通信,还要使用其他的调 用,以使得socket所指的结构中使用的信息被填写完。
在使用TCP协议时,一般服务端进程先使用socket调用得到一个描述 符,然后使用bind调用将一个名字与socket描述符连接起来,对于Internet域就是将Internet地址联编到socket。之后,服务端 使用listen调用指出等待服务请求队列的长度。然后就可以使用accept调用等待客户端发起连接,一般是阻塞等待连接,一旦有客户端发出连接, accept返回客户的地址信息,并返回一个新的socket描述符,该描述符与原先的socket有相同的特性,这时服务端就可以使用这个新的 socket进行读写操作了。一般服务端可能在accept返回后创建一个新的进程进行与客户的通信,父进程则再到accept调用处等待另一个连接。客 户端进程一般先使用socket调用得到一个socket描述符,然后使用connect向指定的服务器上的指定端口发起连接,一旦连接成功返回,就说明 已经建立了与服务器的连接,这时就可以通过socket描述符进行读写操作了。
.NetFrameWork为Socket通讯提供了System.Net.Socket命名空间,在这个命名空间里面有以下几个常用的重要类分别是:
??Socket类 这个低层的类用于管理连接,WebRequest,TcpClient和UdpClient在内部使用这个类。
??NetworkStream类 这个类是从Stream派生出来的,它表示来自网络的数据流
??TcpClient类 允许创建和使用TCP连接
??TcpListener类 允许监听传入的TCP连接请求
??UdpClient类 用于UDP客户创建连接(UDP是另外一种TCP协议,但没有得到广泛的使用,主要用于本地网络)
下面我们来看一个基于Socket的双机通信代码的C#版本
首先创建Socket对象的实例,这可以通过Socket类的构造方法来实现:
public Socket(AddressFamily addressFamily,SocketType socketType,ProtocolType protocolType);
其中,addressFamily 参数指定 Socket 使用的寻址方案,socketType 参数指定 Socket 的类型,protocolType 参数指定 Socket 使用的协议。
下面的示例语句创建一个 Socket,它可用于在基于 TCP/IP 的网络(如 Internet)上通讯。
Socket temp = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
若要使用 UDP 而不是 TCP,需要更改协议类型,如下面的示例所示:
Socket temp = new Socket(AddressFamily.InterNetwork, SocketType.Dgram, ProtocolType.Udp);
一旦创建 Socket,在客户端,你将可以通过Connect方法连接到指定的服务器(你可以在Connect方法前Bind端口,就是以指定的端口 发起连接,如果不事先Bind端口号的话,系统会默认在1024到5000随机绑定一个端口号),并通过Send方法向远程服务器发送数据,而后可以通过 Receive从服务端接收数据;而在服务器端,你需要使用Bind方法绑定所指定的接口使Socket与一个本地终结点相联,并通过Listen方法侦 听该接口上的请求,当侦听到用户端的连接时,调用Accept完成连接的操作,创建新的Socket以处理传入的连接请求。使用完 Socket 后,使 用 Close 方法关闭 Socket。
可以看出,以上许多方法包含EndPoint类型的参数,在Internet中, TCP/IP 使用一个网络地址和一个服务端口号来唯一标识设备。网络地址标识网络上的特定设备;端口号标识要连接到的该设备上的特定服务。网络地址和服 务端口的组合称为终结点,在 .NET 框架中正是由 EndPoint 类表示这个终结点,它提供表示网络资源或服务的抽象,用以标志网络地址等信 息。.Net同时也为每个受支持的地址族定义了 EndPoint 的子代;对于 IP 地址族,该类为 IPEndPoint。IPEndPoint 类包含应用程序连接到主机上的服务所需的主机和端口信息,通过组合服务的主机IP地址和端口号,IPEndPoint 类形成到服务的连接点。
用到IPEndPoint类的时候就不可避免地涉及到计算机IP地址,System.Net命名空间中有两种类可以得到IP地址实例:
??IPAddress类:IPAddress 类包含计算机在 IP 网络上的地址。其Parse方法可将 IP 地址字符串转换为 IPAddress 实例。下面的语句创建一个 IPAddress 实例:
IPAddress myIP = IPAddress.Parse("192.168.0.1");
需要知道的是:Socket 类支持两种基本模式:同步和异步。其区别在于:在同步模式中,按块传输,对执行网络操作的函数(如 Send 和 Receive)的调用一直等到所有内容传送操作完成后才将控制返回给调用程序。在异步模式中,是按位传输,需要指定发送的开始和结束。同步模式是最常 用的模式,我们这里的例子也是使用同步模式。
下面看一个完整的例子,client向server发送一段测试字符串,server接收并显示出来,给予client成功响应。
//client端
using System;
using System.Text;
using System.IO;
using System.Net;
using System.Net.Sockets;
namespace socketsample
{
class Class1
{
static void Main()
{
try
{
int port = 2000;
string host = "127.0.0.1";
IPAddress ip = IPAddress.Parse(host);
IPEndPoint ipe = new IPEndPoint(ip, port);//把ip和端口转化为IPEndPoint实例
Socket c = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);//创建一个Socket
Console.WriteLine("Conneting...");
c.Connect(ipe);//连接到服务器
string sendStr = "hello!This is a socket test";
byte[] bs = Encoding.ASCII.GetBytes(sendStr);
Console.WriteLine("Send Message");
c.Send(bs, bs.Length, 0);//发送测试信息
string recvStr = "";
byte[] recvBytes = new byte[1024];
int bytes;
bytes = c.Receive(recvBytes, recvBytes.Length, 0);//从服务器端接受返回信息
recvStr += Encoding.ASCII.GetString(recvBytes, 0, bytes);
Console.WriteLine("Client Get Message:{0}", recvStr);//显示服务器返回信息
c.Close();
}
catch (ArgumentNullException e)
{
Console.WriteLine("ArgumentNullException: {0}", e);
}
catch (SocketException e)
{
Console.WriteLine("SocketException: {0}", e);
}
Console.WriteLine("Press Enter to Exit");
Console.ReadLine();
}
}
}
//server端
using System;
using System.Text;
using System.IO;
using System.Net;
using System.Net.Sockets;
namespace Project1
{
class Class2
{
static void Main()
{
try
{
int port = 2000;
string host = "127.0.0.1";
IPAddress ip = IPAddress.Parse(host);
IPEndPoint ipe = new IPEndPoint(ip, port);
Socket s = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);//创建一个Socket类
s.Bind(ipe);//绑定2000端口
s.Listen(0);//开始监听
Console.WriteLine("Wait for connect");
Socket temp = s.Accept();//为新建连接创建新的Socket。
Console.WriteLine("Get a connect");
string recvStr = "";
byte[] recvBytes = new byte[1024];
int bytes;
bytes = temp.Receive(recvBytes, recvBytes.Length, 0);//从客户端接受信息
recvStr += Encoding.ASCII.GetString(recvBytes, 0, bytes);
Console.WriteLine("Server Get Message:{0}",recvStr);//把客户端传来的信息显示出来
string sendStr = "Ok!Client Send Message Sucessful!";
byte[] bs = Encoding.ASCII.GetBytes(sendStr);
temp.Send(bs, bs.Length, 0);//返回客户端成功信息
temp.Close();
s.Close();
}
catch (ArgumentNullException e)
{
Console.WriteLine("ArgumentNullException: {0}", e);
}
catch (SocketException e)
{
Console.WriteLine("SocketException: {0}", e);
}
Console.WriteLine("Press Enter to Exit");
Console.ReadLine();
}
}
}
上面的例子是用的Socket类,System.Net.Socket命名空间还提供了两个抽象高级类TCPClient和UDPClient和用于通讯流处理的NetWorkStream,让我们看下例子
客户端
TcpClient tcpClient=new TcpCLient(主机IP,端口号);
NetworkStream ns=tcp.Client.GetStream();
服务端
TcpListener tcpListener=new TcpListener(监听端口);
tcpListener.Start();
TcpClient tcpClient=tcpListener.AcceptTcpClient();
NetworkStream ns=tcpClient.GetStream();
服务端用TcpListener监听,然后把连接的对象实例化为一个TcpClient,调用TcpClient.GetStream()方法,返回网络流实例化为一个NetworlStream流,下面就是用流的方法进行Send,Receive
如果是UdpClient的话,就直接UdpClient实例化,然后调用UdpClient的Send和Receive方法,需要注意的事, UdpClient没有返回网络流的方法,就是说没有GetStream方法,所以无法流化,而且使用Udp通信的时候,不要服务器监听。
现在我们大致了解了.Net Socket通信的流程,下面我们来作一个稍微复杂点的程序,一个广播式的C/S聊天程序。
客户端设计需要一个1个ListBox,用于显示聊天内容,一个TextBox输入你要说的话,一个Button发送留言,一个Button建立连接。
点击建立连接的Button后出来一个对话框,提示输入连接服务器的IP,端口,和你的昵称,启动一个接受线程,负责接受从服务器传来的信息并显示在ListBox上面。
服务器端2个Button,一个启动服务,一个T掉已建立连接的客户端,一个ListBox显示连接上的客户端的Ip和端口。
比较重要的地方是字符串编码的问题,需要先把需要传送的字符串按照UTF8编码,然后接受的时候再还原成为GB2312,不然中文显示会是乱码。
还有一个就是接收线程,我这里简单写成一个While(ture)循环,不断判断是否有信息流入,有就接收,并显示在ListBox上,这里有问题,在.Net2.0里面,交错线程修改窗体空间属性的时候会引发一个异常,不可以直接修改,需要定义一个委托来修改。
当客户端需要断开连接的时候,比如点击窗体右上角的XX,就需要定义一个this.FormClosing += new System.Windows.Forms.FormClosingEventHandler(this.Closing);(.Net2.0 是FormClosing系统事件),在Closing()函数里面,发送Close字符给服务端,服务器判断循环判断所有的连接上的客户端传来的信息, 如果是以Close开头,断开与其的连接。看到这里,读者就会问了,如果我在聊天窗口输入Close是不是也断开连接呢?不是的,在聊天窗口输入的信息传 给服务器的时候开头都要加上Ip信息和昵称,所以不会冲突。
本文来自CSDN博客,转载请标明出处:http://blog.csdn.net/dzfb/archive/2006/12/21/1452139.aspx
发表评论
-
linux 电源管理
2012-01-20 09:01 2180linux 电源管理 2011年06 ... -
memcached完全剖析
2012-01-20 09:01 773memcached完全剖析 2011年0 ... -
关于文件的一些总结
2012-01-20 09:01 931关于文件的一些总结 2010年08月29日 Create ... -
low level I/O和stream I/O
2012-01-20 09:01 797low level I/O和stream I/O 201 ... -
充分利用 Xerces-C++,第 1 部分
2012-01-20 09:01 924充分利用 Xerces-C++,第 1 ... -
利用VBS脚本让qq永远在线
2012-01-19 14:05 829利用VBS脚本让qq永远在线 2011年06月07日 让 ... -
vbs脚本实例
2012-01-19 14:05 889vbs脚本实例 2011年02月28日 rem 结束QQ ... -
vbs 脚本没事测试玩
2012-01-19 14:05 672vbs 脚本没事测试玩 2011年03月27日 Set ... -
VBS脚本文件大全
2012-01-19 14:05 877VBS脚本文件大全 2011年05月24日 一、自动打开 ... -
自动下载并运行的VBS脚本代码[转载]
2012-01-19 14:04 1249自动下载并运行的VBS脚本代码[转载] 2012年01月15 ... -
解决系统提示:内存不能为“read”或"written"的办法
2012-01-17 03:55 728解决系统提示:内存不能为“read”或"writte ... -
内存不能为“read”或"written"的解决
2012-01-17 03:55 625内存不能为“read”或"written" ... -
0x08e629ab 指令引用的 0x0000000c内存不能为read 怎么解决11
2012-01-17 03:54 14390x08e629ab 指令引用的 0x00 ... -
内存不能为read和无法定位程序输入点 +@于动态链接库上
2012-01-17 03:54 1462内存不能为read和无法定位程序输入点 +@于动态链接库上 ... -
操作系统为XP 控制面板中的“添加删除程序”打不开,显示“rundll32.exe 遇到问题需要关闭。
2012-01-17 03:54 2699操作系统为XP 控制面板中的“添加删除程序”打不开,显示“ru ... -
C# Socket多线程编程实例
2012-01-16 02:42 596C# Socket多线程编程实例 ... -
C++ socket编程(tcp udp)
2012-01-16 02:42 655C++ socket编程(tcp udp) 2011 ... -
最基本的Socket编程C#
2012-01-16 02:41 610最基本的Socket编程C# 201 ... -
Dev C++ 中socket编程
2012-01-16 02:41 1177Dev C++ 中socket编程 2010年06月07日 ...
相关推荐
### C# Socket编程从入门到精通 #### 一、引言 随着互联网技术的发展,网络编程成为了软件开发中不可或缺的一部分。Socket编程作为一种基础且强大的...希望本文能够帮助初学者快速入门并熟练掌握C# Socket编程技术。
C#版QQ是一个基于C#编程语言开发的即时通讯软件,它允许用户通过输入IP地址与其他用户进行聊天并实现添加好友等功能。这个项目不仅涵盖了C#的基础语法和面向对象编程思想,还涉及到网络通信、多线程处理以及UI设计等...
在本文中,我们将深入探讨如何使用C#的SuperSocket库创建自定义的内置命令行协议。SuperSocket是一个轻量级且高度可扩展的网络通信框架,适用于开发各种类型的TCP服务,如聊天应用、游戏服务器或者数据传输服务。...
《C#专业项目实例开发》电子教材的知识点包含了C#编程语言在项目开发中的应用,涉及C#语言的多个方面,包括但不限于面向对象编程、数据库交互、网络编程、图形用户界面(GUI)设计、异常处理、以及文件操作等。...
本项目是一套基于SuperSocket框架的C#开发的简单聊天程序,全面展示了如何利用C#语言和SuperSocket框架来构建一个客户端与服务器之间的实时通信平台。 SuperSocket是一个高性能、可扩展的异步Socket服务器框架,它...
7. 网络编程:介绍在C#中创建基于HTTP的网络服务、使用Socket进行底层网络通信的方法。 8. Web开发:解释如何利用***技术开发动态网站和Web应用程序,以及使用MVC(Model-View-Controller)模式进行Web开发。 9. ...
本例子读写三菱PLC数据,使用的是基于以太网的TCP/IP实现,不需要额外的组件,读取操作只要放到后台线程就不会卡死线程,本组件支持超级方便的高性能读写操作 1)支持三菱Q系统 MC协议A-1E通信 2)支持三菱Q系统 MC...
在IT行业中,网络通信是不可或缺的一部分,特别是在开发服务器端应用或客户端应用时。C#作为.NET框架的主要编程语言,提供了强大的网络通信支持,包括TcpClient、UdpClient和Socket类。这三个类分别对应TCP、UDP和低...
使用场景及目标:① 初学者可以通过本项目快速入门Socket通信,理解其基本原理;② 开发者可以在现有项目中集成该框架,提高通信模块的稳定性和效率;③ 提供的多种优化功能(如断线重连、心跳检测)有助于应对实际...
### C# Socket编程入门 #### 一、C# Socket编程概述 C# 是一种现代的、面向对象的编程语言,广泛应用于各种类型的软件开发中,包括网络应用开发。在网络编程领域,Socket 编程是非常核心的技术之一。在 .NET ...
本教程将详细讲解如何使用C#开发基于TCP协议的聊天程序,非常适合初学者入门。 TCP(Transmission Control Protocol)是一种面向连接的、可靠的传输层协议,它确保了数据在网络中的有序、无损传输。C#中的System...
Socket编程是计算机网络通信中的重要组成部分...总的来说,这个"Socket源码Demo"是学习TCP/IP通信的一个良好起点,它涵盖了基本的Socket编程技巧,可以帮助初学者快速入门,并为进一步深入学习网络编程打下坚实的基础。
标题中的“supersocket基本类库与测试”指的是SuperSocket,这是一个开源的.NET框架,用于简化TCP/IP协议的开发,使开发者能够快速构建基于Socket的应用程序。SuperSocket提供了一个强大的、可扩展的基础类库,方便...
Socket在C#中的应用主要基于System.Net命名空间,该命名空间提供了Socket类,用于实现TCP和UDP通信。TCP(传输控制协议)是一种面向连接的、可靠的协议,而UDP(用户数据报协议)则是一种无连接的、不可靠的协议。...
本书以C#作为开发语言,以.NET Framework 3.5作为开发平台,站在开发人员的视角,深入剖析了10个企业级应用程序的构建,内容涉及Windows Forms数据库应用程序、WPF华丽界面的多媒体软件、Web数据库应用程序,以及...
2. **面向对象编程**:C#是基于面向对象的,所以理解类、对象、封装、继承、多态这些概念至关重要。类是对象的蓝图,对象是类的实例。封装通过访问修饰符(public、private、protected等)实现数据保护。继承允许...
根据提供的信息,“C#入门经典 第6版”这本书主要针对初学者介绍了C#语言的基础知识与编程技巧。尽管提供的具体内容部分并未包含实际的学习材料或章节概述,但基于书名及通常此类书籍涵盖的主题,我们可以合理推测书...
1. **C#基础**:C#是微软公司推出的一种面向对象的编程语言,基于.NET框架。它的基础语法类似于Java,包括变量声明、数据类型、运算符、控制流语句(如if、for、while)、函数等。学习C#的第一步是理解这些基本元素...
在本文中,我们将深入探讨如何使用C#语言开发Windows Forms(WinForm)游戏。C#是一种面向对象的、类型安全的编程语言,由Microsoft开发,主要用于构建Windows平台的应用程序,包括游戏。WinForms是.NET Framework...