Socket入门笔记 用TcpClient实现一个简易聊天室

效果

实现思路

使用TcpListener建一个服务器,接收所有客户端发送的消息,然后由服务器再发送到其他客户端

客户端使用TcpClient,发消息给服务器,接收服务器的消息,不和其他客户端直接交互

服务器端

接收客户端

开启一个线程,死循环去接收客户端.接收到之后放到一个集合里,保存起来,以便转发消息用.每个客户端都再开启一个线程,用于接收这个客户端发送的消息.

接收客户端的方法AcceptTcpClient()是阻塞方法,在程序退出释放资源时会引发异常,可以先使用Pending()方法先判断是否有挂起的链接请求,有请求的话再去接收.这样可以避免退出时引发的异常.

这里取每隔1秒接收一次.

/// <summary>
/// 接收客户端
/// </summary>
private void AcceptClient()
{
    try
    {
        while (_isAccept)
        {
            if (_listener.Pending())
            {
                TcpClient client = _listener.AcceptTcpClient();
                IPEndPoint endpoint = client.Client.RemoteEndPoint as IPEndPoint;
                _clients.Add(endpoint.ToString(), client);

                //添加到前端客户端列表
                lbx_Clients.Dispatcher.Invoke(() =>
                {
                    lbx_Clients.Items.Add(endpoint.ToString());
                });

                //接收消息线程
                Thread reciveMessageThread = new Thread(ReciveMessage);
                reciveMessageThread.Start(client);
            }
            else
            {
                Thread.Sleep(1000);
            }
        }
    }
    catch (Exception ex)
    {
        MessageBox.Show(ex.Message);
    }
}
接收消息并转发

也是死循环接收,使用Read()方法接收.如果远程主机已关闭连接,Read()将立即返回零字节.此时跳出循环,释放资源,结束此线程.

/// <summary>
/// 接收消息
/// </summary>
/// <param name="obj">TcpClient</param>
private void ReciveMessage(object obj)
{
    TcpClient client = obj as TcpClient;
    IPEndPoint endpoint = null;
    NetworkStream stream = null;

    try
    {
        endpoint = client.Client.RemoteEndPoint as IPEndPoint;
        stream = client.GetStream();

        while (true)
        {
            byte[] data = new byte[1024];
            //如果远程主机已关闭连接,Read将立即返回零字节
            int length = stream.Read(data, 0, data.Length);
            if (length > 0)
            {
                #region if
                string msg = Encoding.UTF8.GetString(data, 0, length);

                //添加到前端消息列表
                lbx_Messages.Dispatcher.Invoke(() =>
                {
                    lbx_Messages.Items.Add(string.Format("{0}:{1}", endpoint.ToString(), msg));
                });

                //发送到其他客户端
                foreach (KeyValuePair<string, TcpClient> kvp in _clients)
                {
                    if (kvp.Value != client)
                    {
                        string writeMsg = string.Format("{0}:{1}", endpoint.ToString(), msg);
                        byte[] writeData = Encoding.UTF8.GetBytes(writeMsg);
                        NetworkStream writeStream = kvp.Value.GetStream();
                        writeStream.Write(writeData, 0, writeData.Length);
                    }
                }
                #endregion
            }
            else
            {
                //客户端断开连接 跳出循环
                break;
            }
        }
    }
    catch (Exception ex)
    {
        //Read是阻塞方法 客户端退出是会引发异常 释放资源 结束此线程
    }
    finally
    {
        //从前端客户端列表移除
        lbx_Clients.Dispatcher.Invoke(() =>
        {
            lbx_Clients.Items.Remove(endpoint.ToString());
        });
        //释放资源
        stream.Dispose();
        _clients.Remove(endpoint.ToString());
        client.Dispose();
    }
}

客户端

接收消息

开启线程,死循环接收服务器发送的消息.如果Read()返回0,说明服务器已关闭.

/// <summary>
/// 接收消息
/// </summary>
private void ReciveMessage()
{
    try
    {
        NetworkStream stream = _client.GetStream();
        while (true)
        {
            byte[] data = new byte[1024];
            int length = stream.Read(data, 0, data.Length);
            if (length > 0)
            {
                string msg = Encoding.UTF8.GetString(data, 0, length);

                //添加到前端消息列表
                lbx_Messages.Dispatcher.Invoke(() =>
                {
                    lbx_Messages.Items.Add(msg);
                });
            }
            else
            {
                MessageBox.Show("服务器已关闭");
                stream.Dispose();
                break;
            }
        }
    }
    catch (Exception ex)
    {
        //Read是阻塞方法 程序退出释放资源是会引发异常 不做处理 线程结束
    }
}

源码下载:

服务器端:SocketServerDemo.zip

客户端:SocketClientDemo.zip

时间: 2024-08-06 20:08:33

Socket入门笔记 用TcpClient实现一个简易聊天室的相关文章

vue + socket.io实现一个简易聊天室

vue + vuex + elementUi + socket.io实现一个简易的在线聊天室,提高自己在对vue系列在项目中应用的深度.因为学会一个库或者框架容易,但要结合项目使用一个库或框架就不是那么容易了.功能虽然不多,但还是有收获.设计和实现思路较为拙劣,恳请各位大大指正. 可以达到的需求 能查看在线用户列表 能发送和接受消息 使用到的框架和库 socket.io做为实时通讯基础 vuex/vue:客户端Ui层使用 Element-ui:客户端Ui组件 类文件关系图 服务端: 客户端: 服

用HTML5 Web Storage作一个简易聊天室

前 言: 2个月前为了一个评论让我潜水(潜伏)博客园2年作了一次艰难的决定.注册了一个账号! 没事瞎逛博客园以及其他技术网站,发现一个不错的留言墙.就看了其中使用的技术.呀!惊呆了.居然是HTML5,完全没接触过呀! 再 言: 所以就搜了博客园相关文档,有如下信息: @#$^%&*( $&^^**(&( http://zzk.cnblogs.com/s?t=b&w=html5%20Storage #$%&^&*( %*(&()*%$ 资料太多了,你也

socket.io入门,简易聊天室

介绍 通常我们web使用的是http协议,但是 HTTP 协议有一个缺陷:通信只能由客户端发起. 所以我们需要一个可以由服务端主动发出的协议,即WebSocket. WebSocket是HTML5新增的一种通信协议,其特点是服务端可以主动向客户端推送信息,客户端也可以主动向服务端发送信息,是真正的双向平等对话,属于服务器推送技术的一种. Socket.IO 是一个基于 Node.js 的实时应用程序框架,在即时通讯.通知与消息推送,实时分析等场景中有较为广泛的应用. socket.io 包含两个

php_3_“简易聊天室 ”实现的关键技术 详解

                  PHP+MySQL实现Internet上一个简易聊天室的关键技术  系统目标: 聊天室使用数据库汇集每个人的发言,并可将数据库内的发言信息显示在页面,让每个用户都可以看到,具体功能如下: a.用户登录:用户发言时显示其登录名信息 b.用户发言:用户输入说的话 c.显示发言信息:用户浏览所有发言信息 设计思路: (1).建立聊天室数据库及相关数据表 (2).实现用户登录页面(login.php) (3).实现发言页面(speak.php) (4).实现发言显示页

Socket编程(简易聊天室客户端/服务器编写、CocoaAsyncSocket)

Socket编程(简易聊天室客户端/服务器编写.CocoaAsyncSocket) 一.Socket 1.1 Socket简介 Socket就是为网络服务提供的一种机制.网络通信其实就是Socket间的通信,通信的两端都是Socket,数据在两个Socket间通过IO传输. 在Web服务大行其道的今天,调用Web服务的代价是高昂的,尤其是仅仅是抓取少量数据的时候尤其如此.而使用Socket,可以只传送数据本身而不用进行XML封装,大大降低数据传输的开销.Socket允许使用长连接,允许应用程序运

php+websocket搭建简易聊天室实践

1.前言 公司游戏里面有个简单的聊天室,了解了之后才知道是node+websocket做的,想想php也来做个简单的聊天室.于是搜集各种资料看文档.找实例自己也写了个简单的聊天室. http连接分为短连接和长连接.短连接一般可以用ajax实现,长连接就是websocket.短连接实现起来比较简单,但是太过于消耗资源.websocket高效不过兼容存在点问题.websocket是html5的资源 如果想要详细了解websocket长连接的原理请看https://www.zhihu.com/ques

基于C/S模式的简易聊天室

一.任务简要描述 移动互联网技术的广泛应用为人们提供了非常便捷的沟通方式.QQ.微信和微博等是便携式聊天系统的典型代表,它们的功能非常强大. 本系统利用TCP/IP协议的Socket和ServerSocket类,实现基于C/S模式的简易聊天室.该聊天室包括服务端和客户端两部分,服务端是客户端发送消息的中转站:客户端之间可以直接通信,也可以与服务器通信.聊天结束后客户端断开与服务端的连接,服务器也可以停止信息中转服务. 二.系统需求分析 本系统采用C/S软件架构,服务器端负责监听客户端发来的消息,

搭建Websocket简易聊天室

本文,我们通过Egret和Node.js实现一个在线聊天室的demo.主要包括:聊天,改用户名,查看其他用户在线状态的功能.大致流程为,用户访问网页,即进入聊天状态,成为新游客,通过底部的输入框,可以输入自己想说的话,点击发布,信息呈现给所有在聊天的人的页面.用户可以实时修改自己的昵称,用户离线上线都会实时广播给其他用户. 体验链接 http://7hds.com:8888/ 下图为最终制作完成的聊天面板 WebSocket服务器可以用其他语言编写,本文采用的方法建立在Node.js上 . 在N

docker搭建swoole简易聊天室

docker搭建swoole的简易聊天室 首先pull镜像 docker pull docker.io/kong36088/nginx-php7-swoole 创建容器 docker run --name {自己创建的名字} -p 9501:9501 -p 8089:80 -d -it kong36088/nginx-php7-swoole /bin/bash 进入容器 docker exec -it {容器名字或id} /bin/bash 进入容器之后进入nginx配置文件 cd /etc/n