一、直接上效果图

二、Socket握手

三、服务端

     Thread threadWatch = null;// 负责监听客户端的线程
Socket socketWatch = null;// 负责监听客户端的套接字
Socket clientConnection = null;// 负责和客户端通信的套接字
private void btn_Click(object sender, EventArgs e)
{
if (string.IsNullOrEmpty(ipAddress.Text.ToString()))
{
MessageBox.Show("监听ip地址不能为空!");
return;
}
if (string.IsNullOrEmpty(port.Text.ToString()))
{
MessageBox.Show("监听端口不能为空!");
return;
}
// 定义一个套接字用于监听客户端发来的消息,包含三个参数(ipv4寻址协议,流式连接,tcp协议)
socketWatch = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
// 服务端发送消息需要一个ip地址和端口号
IPAddress ip = IPAddress.Parse(ipAddress.Text.Trim());
// 把ip地址和端口号绑定在网路节点endport上
IPEndPoint endPort = new IPEndPoint(ip, int.Parse(port.Text.Trim())); // 监听绑定的网路节点
socketWatch.Bind(endPort);
// 将套接字的监听队列长度设置限制为0,0表示无限
socketWatch.Listen(0);
// 创建一个监听线程
threadWatch = new Thread(WatchConnecting);
threadWatch.IsBackground = true;
threadWatch.Start();
chatContent.AppendText("成功启动监听!ip:"+ip+",端口:"+port.Text.Trim()+"\r\n"); } /// <summary>
/// 监听客户端发来的请求
/// </summary>
private void WatchConnecting()
{
//持续不断监听客户端发来的请求
while (true)
{
clientConnection = socketWatch.Accept();
chatContent.AppendText("客户端连接成功!"+"\r\n");
// 创建一个通信线程
ParameterizedThreadStart pts = new ParameterizedThreadStart(acceptMsg);
Thread thr = new Thread(pts);
thr.IsBackground = true;
thr.Start(clientConnection);
}
} /// <summary>
/// 接收客户端发来的消息
/// </summary>
/// <param name="socket">客户端套接字对象</param>
private void acceptMsg(object socket)
{
Socket socketServer = socket as Socket;
while (true)
{
//创建一个内存缓冲区 其大小为1024*1024字节 即1M
byte[] recMsg = new byte[1024 * 1024];
//将接收到的信息存入到内存缓冲区,并返回其字节数组的长度
int length = socketServer.Receive(recMsg);
//将机器接受到的字节数组转换为人可以读懂的字符串
string msg = Encoding.UTF8.GetString(recMsg,0,length);
chatContent.AppendText("客户端("+GetCurrentTime()+"):"+msg+"\r\n");
}
}
/// <summary>
/// 发送消息到客户端
/// </summary>
/// <param name="msg"></param>
private void serverSendMsg(string msg)
{
byte[] sendMsg = Encoding.UTF8.GetBytes(msg);
clientConnection.Send(sendMsg);
chatContent.AppendText("服务端("+GetCurrentTime()+"):"+msg+"\r\n");
} /// <summary>
/// 获取当前系统时间的方法
/// </summary>
/// <returns>当前时间</returns>
private DateTime GetCurrentTime()
{
DateTime currentTime = new DateTime();
currentTime = DateTime.Now;
return currentTime;
}

四、客户端

      // 创建一个客户端套接字
Socket clientSocket = null;
// 创建一个监听服务端的线程
Thread threadServer = null;
private void btn_Click(object sender, EventArgs e)
{
clientSocket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
if (string.IsNullOrEmpty(ipAddress.Text.ToString()))
{
MessageBox.Show("监听ip地址不能为空!");
return;
}
if (string.IsNullOrEmpty(port.Text.ToString()))
{
MessageBox.Show("监听端口不能为空!");
return;
}
IPAddress ip = IPAddress.Parse(ipAddress.Text.Trim());
IPEndPoint endpoint = new IPEndPoint(ip, int.Parse(port.Text.Trim())); try
{ //这里客户端套接字连接到网络节点(服务端)用的方法是Connect 而不是Bind
clientSocket.Connect(endpoint);
}
catch
{
chatContent.AppendText("连接失败!"); } // 创建一个线程监听服务端发来的消息
threadServer = new Thread(recMsg);
threadServer.IsBackground = true;
threadServer.Start();
} /// <summary>
/// 接收服务端发来的消息
/// </summary>
private void recMsg() { while (true) //持续监听服务端发来的消息
{
//定义一个1M的内存缓冲区 用于临时性存储接收到的信息
byte[] arrRecMsg = new byte[1024 * 1024];
int length = 0;
try
{
//将客户端套接字接收到的数据存入内存缓冲区, 并获取其长度
length = clientSocket.Receive(arrRecMsg);
}
catch
{
return; } //将套接字获取到的字节数组转换为人可以看懂的字符串
string strRecMsg = Encoding.UTF8.GetString(arrRecMsg, 0, length);
//将发送的信息追加到聊天内容文本框中
chatContent.AppendText("服务端(" + GetCurrentTime() + "):" + strRecMsg + "\r\n");
}
} /// <summary>
/// 发送消息到服务端
/// </summary>
/// <param name="msg"></param>
private void clientSendMsg(string msg)
{
byte[] sendMsg = Encoding.UTF8.GetBytes(msg);
clientSocket.Send(sendMsg);
chatContent.AppendText("客户端(" + GetCurrentTime() + "):" + msg + "\r\n");
}
/// <summary>
/// 获取当前系统时间的方法
/// </summary>
/// <returns>当前时间</returns>
private DateTime GetCurrentTime()
{
DateTime currentTime = new DateTime();
currentTime = DateTime.Now;
return currentTime;
}

GitHub源码地址:https://github.com/51042309/Socket

最新文章

  1. AngularJS时间轴指令
  2. [I2C]I2C总线协议图解
  3. Core Dump [Linux]
  4. Yii 框架中安装 memcache 及配置和测试
  5. C#条件编译,发布多平台和多种选择性的项目
  6. 【Android - 进阶】之自定义视图浅析
  7. 动态Script标签 解决跨域问题
  8. Linux内核中链表实现
  9. fs模块主要功能小解
  10. P2256 一中校运会之百米跑(map映射加并查集)
  11. Python中的sys.path.append()
  12. [20190130]删除tab$记录的恢复2.txt
  13. C/C++ 内存对齐原则及作用
  14. 【adaboost】周志华
  15. C# DataTbale详细操作
  16. Linux学习笔记:vi常用命令
  17. Spring Boot(一):初步认识
  18. 使用GridFsTemplate在Mongo中存取文件
  19. 利用U盘给Intel NUC安装CentOS
  20. 第二百五十四节,Bootstrap项目实战--案例

热门文章

  1. hdu2680 Choose the best route 最短路(多源转单源)
  2. PostgreSQL的HA解决方案-2负载均衡(load balance)
  3. 最简单的一致性Hash算法实现
  4. css3 y轴翻转效果
  5. mongodb报错一例
  6. Embeding如何理解?
  7. docker批量删除容器、镜像
  8. 洛谷P1231 教辅的组成 最大流
  9. BZOJ 3774 最优选择 (最小割+二分图)
  10. luogu 4240 毒瘤之神的考验 (莫比乌斯反演)