Socket 通信(基础原理、实时聊天系统雏形)
什么是 Socket?
Socket 英文直译为“孔或插座”,也称为套接字。用于描述 IP 地址和端口号,是一种进程间的通信机制。你可以理解为 IP 地址确定了网内的唯一计算机,而端口号则指定了将消息发送给哪一个应用程序(大多应用程序启动时会主动绑定一个端口,如果不主动绑定,操作系统自动为其分配一个端口)。
什么是端口?
一台主机一般运行了多个软件并同时提供一些服务。每种服务都会打开一个 Socket,并绑定到一个端口号上,不同端口对应于不同的应用程序。例如 http 使用 80 端口;ftp 使用 21 端口;smtp 使用 23 端口。
Socket 的类型
- Stream:一种流式 Socket,针对于面向连接的 TCP 服务应用,安全,但效率低。(本文重点)
- Datagram:数据报式的 Socket,针对于无连接的 UDP 服务应用,不安全(丢失、顺序混乱,往往在接收端要分析完整性、重排、或要求重发),但效率高。
Socket 程序一般应用模式及运行流程
- 服务器端会启动一个 Socket,开始监听端口,监听客户端的连接信息,我们称之为 Watch Socket。
- 客户端 Socket 连接服务器端的监听 Socket,一旦成功连接,服务器端会立刻创建一个新的 Socket 负责与客户端进行通信,之后,客户端将不再与 Watch Socket 通信。
- Watch Socket 继续监听可能会来自其他客户端的连接。
上述过程就像是实现了一次三方会谈。服务器端的 Socket 至少会有 2 个。一个是 Watch Socket,每成功接收到一个客户端的连接,便在服务器端创建一个通信 Socket。客户端 Socket 指定要连接的服务器端地址和端口,创建一个 Socket 对象来初始化一个到服务器的 TCP 连接。
通信的雏形
下面就看一个最简单的 Socket 示例,实现了网络聊天通信的雏形。
服务器端:
public partial class ChatServer : Form
{
public ChatServer()
{
InitializeComponent();
ListBox.CheckForIllegalCrossThreadCalls = false;
} /// <summary>
/// 监听 Socket 运行的线程
/// </summary>
Thread threadWatch = null; /// <summary>
/// 监听 Socket
/// </summary>
Socket socketWatch = null; /// <summary>
/// 服务器端通信套接字集合
/// 必须在每次客户端连接成功之后,保存新建的通讯套接字,这样才能和后续的所有客户端通信
/// </summary>
Dictionary<string, Socket> dictCommunication = new Dictionary<string, Socket>(); /// <summary>
/// 通信线程的集合,用来接收客户端发送的信息
/// </summary>
Dictionary<string, Thread> dictThread = new Dictionary<string, Thread>(); private void btnBeginListen_Click(object sender, EventArgs e)
{
// 创建服务器端监听 Socket (IP4寻址协议,流式连接,TCP协议传输数据)
socketWatch = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp); // 监听套接字绑定指定端口
IPAddress address = IPAddress.Parse(txtIP.Text.Trim());
IPEndPoint endPoint = new IPEndPoint(address, int.Parse(txtPort.Text.Trim()));
socketWatch.Bind(endPoint); // 将监听套接字置于侦听状态,并设置连接队列的最大长度
socketWatch.Listen(20); // 启动监听线程开始监听客户端请求
threadWatch = new Thread(Watch);
threadWatch.IsBackground = true;
threadWatch.Start();
ShowMsg("服务器启动完成!");
} Socket socketCommunication = null;
private void Watch()
{
while (true)
{
// Accept() 会创建新的通信 Socket,且会阻断当前线程,因此应置于非主线程上使用
// Accept() 与线程上接受的委托类型不符,因此需另建一方法做桥接
socketCommunication = socketWatch.Accept(); // 将新建的通信套接字存入集合中,以便服务器随时可以向指定客户端发送消息
// 如不置于集合中,每次 new 出的通信线程都是一个新的套接字,那么原套接字将失去引用
dictCommunication.Add(socketCommunication.RemoteEndPoint.ToString(), socketCommunication);
lbSocketOnline.Items.Add(socketCommunication.RemoteEndPoint.ToString()); // Receive 也是一个阻塞方法,不能直接运行在 Watch 中,否则监听线程会阻塞
// 另外,将每一个通信线程存入集合,方便今后的管理(如关闭、或挂起)
Thread thread = new Thread(() =>
{
while (true)
{
byte[] bytes = new byte[1024 * 1024 * 2];
int length = socketCommunication.Receive(bytes);
string msg = Encoding.UTF8.GetString(bytes, 0, length);
ShowMsg("接收到来自" + socketCommunication.RemoteEndPoint.ToString() + "的数据:" + msg);
}
});
thread.IsBackground = true;
thread.Start();
dictThread.Add(socketCommunication.RemoteEndPoint.ToString(), thread); ShowMsg("客户端连接成功!通信地址为:" + socketCommunication.RemoteEndPoint.ToString());
}
} delegate void ShowMsgCallback(string msg);
private void ShowMsg(string msg)
{
if (this.InvokeRequired) // 也可以启动时修改控件的 CheckForIllegalCrossThreadCalls 属性
{
this.Invoke(new ShowMsgCallback(ShowMsg), new object[] { msg });
}
else
{
this.txtMsg.AppendText(msg + "\r\n");
}
} private void btnSendMsg_Click(object sender, EventArgs e)
{
if (lbSocketOnline.Text.Length == 0)
MessageBox.Show("至少选择一个客户端才能发送消息!");
else
{
// Send() 只接受字节数组
string msg = txtSendMsg.Text.Trim();
dictCommunication[lbSocketOnline.Text].Send(Encoding.UTF8.GetBytes(msg));
ShowMsg("发送数据:" + msg);
}
} private void btnSendToAll_Click(object sender, EventArgs e)
{
string msg = txtSendMsg.Text.Trim();
foreach (var socket in dictCommunication.Values)
{
socket.Send(Encoding.UTF8.GetBytes(msg));
}
ShowMsg("群发数据:" + msg);
}
}
客户端:
public partial class ChatClient : Form
{
public ChatClient()
{
InitializeComponent();
} /// <summary>
/// 此线程用来接收服务器发送的数据
/// </summary>
Thread threadRecive = null; Socket socketClient = null; private void btnConnect_Click(object sender, EventArgs e)
{
// 客户端创建通讯套接字并连接服务器、开始接收服务器传来的数据
socketClient = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
socketClient.Connect(IPAddress.Parse(txtIP.Text.Trim()), int.Parse(txtPort.Text.Trim()));
ShowMsg(string.Format("连接服务器({0}:{1})成功!", txtIP.Text.Trim(), txtPort.Text.Trim())); threadRecive = new Thread(new ThreadStart(() =>
{
while (true)
{
// Receive 方法从套接字中接收数据,并存入接收缓冲区
byte[] bytes = new byte[1024 * 1024 * 2];
int length = socketClient.Receive(bytes);
string msg = Encoding.UTF8.GetString(bytes, 0, length);
ShowMsg("接收到数据:" + msg);
}
}));
threadRecive.IsBackground = true;
threadRecive.Start();
} delegate void ShowMsgCallback(string msg);
private void ShowMsg(string msg)
{
if (this.InvokeRequired) // 也可以启动时修改控件的 CheckForIllegalCrossThreadCalls 属性
{
this.Invoke(new ShowMsgCallback(ShowMsg), new object[] { msg });
}
else
{
this.txtMsg.AppendText(msg + "\r\n");
}
} private void btnSend_Click(object sender, EventArgs e)
{
string msg = txtSendMsg.Text.Trim();
socketClient.Send(Encoding.UTF8.GetBytes(msg));
ShowMsg("发送数据:" + msg);
}
}
现在所有客户都能和服务器进行通信,服务器也能和所有客户进行通信。那么,客户端之间互相通信呢?
显然,在客户端界面也应创建在线列表,每次有人登录后,服务器端除了刷新自身在线列表外,还需将新客户端的套接字信息发送给其他在线客户端,以便它们更新自己的在线列表。
客户端发送消息给服务器,服务器转发此消息给另一个客户端。当然,这个消息需要进行一些处理,至少要包含目标套接字和发送内容。
更为完善的是,服务器必须定时按制定的规则检测列表中套接字通信的有效性,通过发送响应信号,并接收客户端应答信号以确认客户端的连接性是真实的(否则,需剔除无效客户端)。
客户端上传文件
客户端:
private void btnChooseFile_Click(object sender, EventArgs e)
{
OpenFileDialog ofd = new OpenFileDialog();
if (ofd.ShowDialog() == DialogResult.OK)
{
txtFilePath.Text = ofd.FileName;
}
} private void btnSendFile_Click(object sender, EventArgs e)
{
using (FileStream fs = new FileStream(txtFilePath.Text, FileMode.Open))
{
byte[] bytes = new byte[1024 * 1024 * 2]; // 假设第一个字节为标志位:0 表示传送文件 // 方式一:整体向后偏移 1 个字节;但这样有潜在缺点,
// 有时在通信时会非常准确的按照约定的字节长度来传递,
// 那么这种偏移方案显然是不可靠的
// bytes[0] = 0;
// int length = fs.Read(bytes, 1, bytes.Length); // 方式二:创建多出 1 个字节的数组发送
int length = fs.Read(bytes, 0, bytes.Length);
byte[] newBytes = new byte[length + 1];
newBytes[0] = 0;
// BlockCopy() 会比你自己写for循环赋值更为简单合适
Buffer.BlockCopy(bytes, 0, newBytes, 1, length);
socketClient.Send(newBytes);
}
}
服务器端(Receive 方法中修改成这样):
Thread thread = new Thread(() =>
{
while (true)
{
byte[] bytes = new byte[1024 * 1024 * 2];
int length = socketCommunication.Receive(bytes); if (bytes[0] == 0) // File
{
SaveFileDialog sfd = new SaveFileDialog();
if (sfd.ShowDialog() == DialogResult.OK)
{
using (FileStream fs = new FileStream(sfd.FileName, FileMode.Create))
{
fs.Write(bytes, 1, length - 1);
fs.Flush();
ShowMsg("文件保存成功,路径为:" + sfd.FileName);
}
}
}
else // Msg
{
string msg = Encoding.UTF8.GetString(bytes, 0, length);
ShowMsg("接收到来自" + socketCommunication.RemoteEndPoint.ToString() + "的数据:" + msg);
}
}
});
异常捕捉
Socket 通信属于网络通信程序,会有许多的意外,必须进行异常处理以便程序不会被轻易的击垮。不管是客户端还是服务器端,只要和网络交互的环节(Connect、Accept、Send、Receive 等)都要做异常处理。
本例中对服务器端 Receive 方法环节做了一些异常处理,并移除了相应的资源,例如下面:
try
{
length = socketCommunication.Receive(bytes);
}
catch (SocketException ex)
{
ShowMsg("出现异常:" + ex.Message);
string key = socketCommunication.RemoteEndPoint.ToString();
lbSocketOnline.Items.Remove(key);
dictCommunication.Remove(key);
dictThread.Remove(key);
break;
}
系统界面截图
9
Socket 通信(基础原理、实时聊天系统雏形)的更多相关文章
- php简单实现socket通信
socket通信的原理在这里就不说了,它的用途还是比较广泛的,我们可以使用socket来做一个API接口出来,也可以使用socket来实现两个程序之间的通信,我们来研究一下在php里面如何实现sock ...
- php socket通信的简单实现
socket通信的原理在这里就不说了,它的用途还是比较广泛的,我们可以使用socket来做一个API接口出来,也可以使用socket来实现两个程序之间的通信,我们来研究一下在php里面如何实现sock ...
- Java基础知识强化之网络编程笔记02:Socket通信原理图解
1. Socket (1)Socket套接字 网络上具有唯一标识的IP地址和端口号组合在一起才能构成唯一能识别的标识符套接字 (2)Socket原理机制: • 通信两端都有Socket. • 网 ...
- PHP的socket通信原理及实现
对TCP/IP.UDP.Socket编程这些词你不会很陌生吧?随着网络技术的发展,这些词充斥着我们的耳朵.那么我想问: 1. 什么是TCP/IP.UDP?2. Sock ...
- 【Socket通信】关于Socket通信原理解析及python实现
Socket(套接字)通信{网络通信其实就是Socket间的通信},首先了解下概念:[来源于百度百科] "两个程序通过一个双向的通信连接实现数据的交换,这个连接的一端称为一个socket.& ...
- Socket 通信原理(Android客户端和服务器以TCP&&UDP方式互通)
转载地址:http://blog.csdn.net/mad1989/article/details/9147661 ZERO.前言 有关通信原理内容是在网上或百科整理得到,代码部分为本人所写,如果不当 ...
- Python Socket通信原理
[Python之旅]第五篇(一):Python Socket通信原理 python Socket 通信理论 socket例子 摘要: 只要和网络服务涉及的,就离不开Socket以及Socket编 ...
- 使用Socket通信实现Silverlight客户端实时数据的获取(模拟GPS数据,地图实时位置)
原文:使用Socket通信实现Silverlight客户端实时数据的获取(模拟GPS数据,地图实时位置) 在上一篇中说到了Silverlight下的Socket通信,在最后的时候说到本篇将会结合地图. ...
- Android中Socket通信之TCP与UDP传输原理
一.Socket通信简介 Android与服务器的通信方式主要有两种,一是Http通信,一是Socket通信.两者的最大差异在于,http连接使用的是"请求-响应方式",即在请求时 ...
随机推荐
- Longest Substring Without Repeating Characters(Difficulty: Medium)
题目: Given a string, find the length of the longest substring without repeating characters. Examples: ...
- 浅谈大数据神器Spark中的RDD
1.究竟什么是RDD呢? 有人可能会回答是:Resilient Distributed Dataset.没错,的确是如此.但是我们问这个实际上是想知道RDD到底是个什么东西?以及它到底能干嘛?好的,有 ...
- 学习NSRulerView
NSRulerMarker 有个imageOrigin是描述Marker中image的位置的: 在horizontal的ruler中,imageOrigin的x坐标是跟marker的location进 ...
- jQuery动画高级用法——详解animation中的.queue()函数
http://www.cnblogs.com/zhwl/p/4328279.html $('#object').hide('slow').queue(function(next){ $(thi ...
- windows Path变量优先级
系统>用户 且第一次配置无需重启即可使用 如遇到升级版本,需要重新配置Path,则需要重启方可生效~~
- Servlet向客户端发送中文数据的编码情况
(更多内容请关注本人微信订阅号:it_pupil) 本文讲述服务端servlet向客户端浏览器发送中文数据的编码情况,需要抓住下面几点: 输出流发送数据,必须是以字节形式传输的.也就是说,如果你在服务 ...
- Android DownloadProvider学习
DownloadProvider 简介 DownloadProvider 是Android提供的DownloadManager的增强版,亮点是支持断点下载,提供了“开始下载”,“暂停下载”,“重新下载 ...
- 。net新人报道
入行一年多 关注博客园也有半年的时间了 今天开始写第一篇东西 以后有什么笔记也会慢慢写上来的
- 【海洋女神原创】How to: Installshield做安装包时如何添加文件
我一直以为这不是一个问题,可是没想到在几个群内,对于如何向安装包添加文件不解的大有人在,今日稍暇,整理成篇,以供参考 首先我想再大声地说一遍:不要再跟我说英文看不懂了!!!!你做了程序员这一行,就得逼 ...
- intent打开第三方应用
有时候我们会有在自己的应用中进入另一个第三方应用的需求,首先要知道第三方应用的包名和主activity,很简单遍历一下所有的应用就能拿到了. private void go2App(String pa ...