C#基于Socket的TCP通信实现聊天室案例
作者:qq_43552118zx 发布时间:2021-12-19 15:54:13
本文实例为大家分享了C#基于Socket的TCP通信实现聊天室的具体代码,供大家参考,具体内容如下
一、Socket(套接字)通信概念
套接字(socket)是通信的基石,用于描述IP地址和端口,是一个通信链的句柄,可以用来实现不同虚拟机或不同计算机之间的通信,是支持TCP/IP协议的网络通信的基本操作单元。它是网络通信过程中端点的抽象表示,包含进行网络通信必须的五种信息:连接使用的协议,本地主机的IP地址,本地进程的协议端口,远地主机的IP地址,远地进程的协议端口。
应用层通过传输层进行数据通信时,TCP会遇到同时为多个应用程序进程提供并发服务的问题。多个TCP连接或多个应用程序进程可能需要通过同一个 TCP协议端口传输数据。为了区别不同的应用程序进程和连接,许多计算机操作系统为应用程序与TCP/IP协议交互提供了套接字(Socket)接口。应 用层可以和传输层通过Socket接口,区分来自不同应用程序进程或网络连接的通信,实现数据传输的并发服务。
二、建立socket连接
建立Socket连接至少需要一对套接字,其中一个运行于客户端,称为ClientSocket ,另一个运行于服务器端,称为ServerSocket 。
套接字之间的连接过程分为三个步骤:服务器监听,客户端请求,连接确认。
1、服务器监听:服务器端套接字并不定位具体的客户端套接字,而是处于等待连接的状态,实时监控网络状态,等待客户端的连接请求
2、客户端请求:指客户端的套接字提出连接请求,要连接的目标是服务器端的套接字。为此,客户端的套接字必须首先描述它要连接的服务器的套接字,指出服务器端套接字的地址和端口号,然后就向服务器端套接字提出连接请求。
3、连接确认:当服务器端套接字监听到或者说接收到客户端套接字的连接请求时,就响应客户端套接字的请求,建立一个新的线程,把服务器端套接字的描述发给客户 端,一旦客户端确认了此描述,双方就正式建立连接。而服务器端套接字继续处于监听状态,继续接收其他客户端套接字的连接请求。
三、SOCKET连接与TCP连接
创建Socket连接时,可以指定使用的传输层协议,Socket可以支持不同的传输层协议(TCP或UDP),当使用TCP协议进行连接时,该Socket连接就是一个TCP连接。
socket通信:分为同步和异步通信,通信两端分别为客户端(Client)和服务器(Server),,本文简单介绍一下同步通信及案例
聊天室案例 服务端
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
using System.Net;
using System.Net.Sockets;
using System.Threading;
namespace 服务器
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
//在多线程编程中,如果需要使用大到主线程需要进行检查取消
CheckForIllegalCrossThreadCalls = false;
}
IDictionary<string, Socket> clientList = new Dictionary<string, Socket>();
private void button1_Click(object sender, EventArgs e)
{
Thread th = new Thread(StartSever);
th.IsBackground = true;
th.Start();
}
void StartSever()
{
//1.创建服务器端电话
Socket server = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.IP);
//2.创建手机卡
IPAddress ip = IPAddress.Parse(txtIP.Text);
//把ip和端口转化为IPEndPoint实例
IPEndPoint endpoint = new IPEndPoint(ip, int.Parse(txtPort.Text));
//3.将电话卡插入到电话中,绑定端口
server.Bind(endpoint);
//4.开始监听电话
server.Listen(20);
listMsg.Items.Add("服务器已成功开启");
//5.等待接电话
while (true)
{
//接收接入的一个客户端
Socket connectClient = server.Accept();
if (connectClient != null)
{
string infor = connectClient.RemoteEndPoint.ToString();
clientList.Add(infor, connectClient);
listMsg.Items.Add(infor + "加入服务器");
string msg =infor+"已成功进入聊天室";
SendMsg(msg);
Thread thread = new Thread(ReciveMsg);
thread.IsBackground = true;
thread.Start(connectClient);
}
}
}
void ReciveMsg(object o)
{
Socket client = o as Socket;
while (true)
{
try
{
byte[] arrMsg = new byte[1024 * 1024];
int length = client.Receive(arrMsg);
if (length > 0)
{
string recMsg = Encoding.UTF8.GetString(arrMsg, 0, length);
IPEndPoint endpoint = client.RemoteEndPoint as IPEndPoint;
listMsg.Items.Add(DateTime.Now + "[" + endpoint.Port.ToString() + "]" + recMsg);
SendMsg("[" + endpoint.Port.ToString() + "]" + recMsg);
}
}
catch (Exception)
{
client.Close();
clientList.Remove(client.RemoteEndPoint.ToString());
}
}
}
private void label1_Click(object sender, EventArgs e)
{
string ip = IPAddress.Any.ToString();
txtIP.Text = ip;
}
void SendMsg(string str)
{
foreach (var item in clientList)
{
byte[] arrMsg = Encoding.UTF8.GetBytes(str);
item.Value.Send(arrMsg);
}
}
private void button2_Click(object sender, EventArgs e)
{
if (txtMsg.Text!="")
{
SendMsg(txtMsg.Text);
}
}
}
}
客户端
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
using System.Net;
using System.Net.Sockets;
using System.Threading;
namespace 服务器2._12
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
CheckForIllegalCrossThreadCalls = false;
}
Socket Client;
private void button1_Click(object sender, EventArgs e)
{
//创建服务器端电话
Client = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.IP);
//创建手机卡
IPAddress ip = IPAddress.Parse(txtIP.Text);
IPEndPoint endinput = new IPEndPoint(ip, int.Parse(txtport.Text));
Client.Connect(endinput);
Thread th = new Thread(ReciveMsg);
th.IsBackground = true;
th.Start(Client);
}
void ReciveMsg(object o)
{
Socket client = o as Socket;
//5.等待接电话
while (true)
{
byte[] arrlist = new byte[1024*1024];
int length = client.Receive(arrlist);
string msg = DateTime.Now + Encoding.UTF8.GetString(arrlist,0,length);
listmsg.Items.Add(msg);
}
}
private void button2_Click(object sender, EventArgs e)
{
if (txtinput.Text!=null)
{
SendMsg(txtinput.Text);
}
}
void SendMsg(string msg)
{
byte[] arrMsg = Encoding.UTF8.GetBytes(msg);
Client.Send(arrMsg);
}
}
}
来源:https://blog.csdn.net/qq_43552118/article/details/87373301


猜你喜欢
- springMVC是spring的一个模块,专门做web的。SpringMVC请求处理过程:请求发送,根据url-pattern,转发发送给
- 一、树概念及结构1.1 树的概念树是一种非线性的数据结构,它是由n(n>=0)个有限结点组成一个具有层次关系的集合。把它叫做树是因 为
- 一、简介:介绍两种使用 BitmapTransformation 来实现 Glide 加载圆形图片和圆角图片的方法。Glide 并不能直接支
- 1、不要为抽象类提供公开的构造方法抽象类可以有构造方法,但是抽象类不能实例化。如果编程人员没有制定构造方法,编译器会自动生成一个默认的pro
- 一、Ctrl+F或者Ctrl+Shift+R 按照文本的内容查找1. 相当于eclipse的ctrl+H,Ctrl+F是在本页查找2. 相当
- 如图,左图是效果,右图是原理,右图X轴代表图像一个像素点的灰度,Y轴代表RGB三个颜色对应的伪彩色图颜色。代码如下:for (int y =
- xUtils简介xUtils 包含了很多实用的android工具。xUtils 支持大文件上传,更全面的http请求协议支持(10种谓词),
- 实例如下:package bys.utils;import java.util.Date;/** * Created by toutou o
- 缓冲区的建立目的就是增加传输性能,使我们传输数据更加快速缓冲区的内部其实很简单 就是在内部封装了一个数组 用数组来存储数据 对外提供一些方法
- vscode Java 开发环境配置博客地址VsCode官网教程系统需安装jdk1.8,配置好环境变量JAVA_HOME 打开vscode,
- 近期,Apache SkyWalking 修复了一个隐藏了近4年的Bug - TTL timer 可能失效问题,这个 bug 在 SkyWa
- 前言Spring 提供了 ApplicationContext 事件机制,可以发布和监听事件,这个特性非常有用。Spring 内置了一些事件
- 前两天,谷歌发布了Android Studio 1.0的正式版,也有更多的人开始迁移到Android Studio进行开发。然而,网上很多的
- 文章目录 简介增量构建自定义inputs和outputs运行时API隐式依赖输入校验自定义缓存方法输入归一化其他使用技巧简介在我们使用的各种
- 摘要在生产环境下,我们需要关闭swagger配置,避免暴露接口的这种危险行为。方法禁用方法1:使用注解 @Value() 推荐使用packa
- object对象中的 public boolean equals(Object obj),对于任何非空引用值 x 和 y,当且仅当 x 和
- 一、新建项目我们这次直接从IEDA创建项目,具体配置如下,还是万年的Java8。二、docker-compose 配置mongoDBdock
- Handler、Message、Loopler、MessageQueen首先看一下我们平常使用Handler的一个最常见用法。Handler
- 本文为大家分享了Java实现文件上传下载功能的具体代码,供大家参考,具体内容如下前端通过form表单的enctype属性,将数据传递方式修改
- 扩展:由于server端是存储了所有server与client的连接对象,因此我们是可以基于此demo的基础上实现聊天系统:* 每当一个与用