我的编程空间,编程开发者的网络收藏夹
学习永远不晚

怎么用Java Socket+多线程实现多人聊天室功能

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

怎么用Java Socket+多线程实现多人聊天室功能

这篇文章主要讲解了“怎么用Java Socket+多线程实现多人聊天室功能”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么用Java Socket+多线程实现多人聊天室功能”吧!

思路简介

分为客户端和服务器两个类,所有的客户端将聊的内容发送给服务器,服务器接受后,将每一条内容发送给每一个客户端,客户端再显示在终端上。

客户端设计

客户端包含2个线程,1个用来接受服务器的信息,再显示,1个用来接收键盘的输入,发送给服务器。

import java.io.IOException;import java.io.InputStream;import java.io.OutputStream;import java.net.Socket;import java.nio.charset.StandardCharsets;import java.util.Scanner; public class WeChatClient {  //WeChat的客户端类    private Socket client;    private String name;    private InputStream in;    private OutputStream out;    private MassageSenter massageSenter;    private MassageGeter massageGeter;    class MassageGeter extends Thread{  //一个子线程类,用于客户端接收消息        MassageGeter() throws IOException{            in = client.getInputStream();        }        @Override        public void run() {            int len;            byte[] bytes = new byte[1024];            try {                while ((len = in.read(bytes)) != -1) { //此函数是阻塞的                    System.out.println(new String(bytes,0,len, StandardCharsets.UTF_8));                }            }catch (IOException e){                System.out.println(e.toString());            }            System.out.println("Connection interruption");        }    }    class MassageSenter extends Thread{  //一个子线程类,用于发送消息给服务器        MassageSenter() throws IOException{            out = client.getOutputStream();        }         @Override        public void run() {            Scanner scanner = new Scanner(System.in);            try {                while (scanner.hasNextLine()) { //此函数为阻塞的函数                    String massage = scanner.nextLine();                    out.write((name + " : " + massage).getBytes(StandardCharsets.UTF_8));                    if(massage.equals("//exit"))                        break;                }            }catch (IOException e){                e.printStackTrace();            }        }    }     WeChatClient(String name, String host, int port) throws IOException {//初始化,实例化发送和接收2个线程        this.name = name;        client = new Socket(host,port);        massageGeter = new MassageGeter();        massageSenter = new MassageSenter();     }     void login() throws IOException{//登录时,先发送名字给服务器,在接收到服务器的正确回应之后,启动线程        out.write(name.getBytes(StandardCharsets.UTF_8));        byte[] bytes = new byte[1024];        int len;        len = in.read(bytes);        String answer = new String(bytes,0,len, StandardCharsets.UTF_8);        if(answer.equals("logined!")) {            System.out.println("Welcome to WeChat! "+name);            massageSenter.start();            massageGeter.start();            try {                massageSenter.join();//join()的作用是等线程结束之后再继续执行主线程(main)                massageGeter.join();            }catch (InterruptedException e){                System.err.println(e.toString());            }         }else{            System.out.println("Server Wrong");        }        client.close();    }      public static void main(String[] args) throws IOException{//程序入口        String host = "127.0.0.1";        WeChatClient client = new WeChatClient("Uzi",host,7777);        client.login();    } }

服务器设计

服务器包含3个线程类,端口监听线程,客户端接收信息线程,发送信息线程。

服务器类还包含并维护着一个已经连接的用户列表,和一个待发送信息列表。

服务器有一个负责监听端口的线程,此线程在接收到客户端的连接请求后,将连接的客户端添加进用户列表;并为每一个连接的客户端实例化一个接受信息的线程类,从各个客户端接收员信息,并存入待发送信息列表。

发送信息线程查看列表是否为空,若不为空,则将里面的信息发送给用户列表的每一个用户。

import java.io.IOException;import java.io.InputStream;import java.io.OutputStream;import java.net.ServerSocket;import java.net.Socket;import java.nio.charset.StandardCharsets;import java.util.ArrayList; public class WeChatServer {    private ServerSocket server;    private ArrayList<User> users;//用户列表    private ArrayList<String> massages;//待发送消息队列    private Listener listener;    private MassageSenter massageSenter;      class User{  //用户类,包含用户的登录id和一个输出流        String name;        OutputStream out;        User(String name,OutputStream out){            this.name = name;            this.out = out;        }         @Override        public String toString() {            return name;        }    }     private static String GetMassage(InputStream in) throws IOException{//从一个输入流接收一个字符串        int len;        byte[] bytes = new byte[1024];        len = in.read(bytes);        return new String(bytes,0,len,StandardCharsets.UTF_8);    }    private void UserList(){  //列出当前在线用户,调试用        for(User user : users)            System.out.println(user);    }     class Listener extends Thread{ //监听线程类,负则监听是否有客户端连接        @Override        public void run() {            try {                while (true) {                    Socket socket = server.accept();//此函数是阻塞的                    InputStream in = socket.getInputStream();                    String name = GetMassage(in);//获取接入用户的name                    System.out.println(name +" has connected");                    massages.add(name+" has joined just now!!");//向聊天室报告用户连入的信息                    OutputStream out = socket.getOutputStream();                    out.write("logined!".getBytes(StandardCharsets.UTF_8));//发送成功建立连接的反馈                    User user = new User(name,out);                    users.add(user);//添加至在线用户列表                    MassageListener listener = new MassageListener(user,in);//创建用于接收此用户信息的线程                    listener.start();                }            }catch (IOException e){                e.printStackTrace();            }        }    }    class MassageListener extends Thread{ //接收线程类,用于从一个客户端接收信息,并加入待发送列表        private User user;        private InputStream in;        MassageListener(User user,InputStream in){            this.user = user;            this.in = in;        }         @Override        public void run() {            try {                while (true){                    String massage = GetMassage(in);                    System.out.println("GET MASSAGE  "+massage);                    if(massage.contains("//exit")){ //       "/exit" 是退出指令                        break;                    }                    massages.add(massage);                }//用户退出有两种形式,输入 “//exit” 或者直接关闭程序                in.close();                user.out.close();             }catch (IOException e){//此异常是处理客户端异常关闭,即GetMassage(in)调用会抛出异常,因为in出入流已经自动关闭                e.printStackTrace();            }finally {                System.out.println(user.name+" has exited!!");                massages.add(user.name+" has exited!!");                users.remove(user);//必须将已经断开连接的用户从用户列表中移除,否则会在发送信息时产生异常                System.out.println("Now the users has");                UserList();            }         }    }    private synchronized void SentToAll(String massage)throws IOException{//将信息发送给每一个用户,加入synchronized修饰,保证在发送时,用户列表不会被其他线程更改        if(users.isEmpty())            return;        for(User user : users){            user.out.write(massage.getBytes(StandardCharsets.UTF_8));        }    }     class MassageSenter extends Thread{//消息发送线程         @Override        public void run() {            while(true){                try{                    sleep(1);//此线程中没有阻塞的函数,加入沉睡语句防止线程过多抢占资源                }catch (InterruptedException e){                    e.printStackTrace();                }                if(!massages.isEmpty()){                    String massage = massages.get(0);                    massages.remove(0);                    try {                        SentToAll(massage);                    }catch (IOException e){                        e.printStackTrace();                    }                 }            }        }    }     WeChatServer(int port) throws IOException {  //初始化        server = new ServerSocket(port);        users = new ArrayList<>();        massages = new ArrayList<>();        listener = new Listener();        massageSenter = new MassageSenter();    }     private void start(){ //线程启动        listener.start();        massageSenter.start();    }     public static void main(String[] args) throws IOException{        WeChatServer server = new WeChatServer(7777);        server.start();    } }

总结

之所以需要多线程编程,是因为有的函数是阻塞的,例如

while ((len = in.read(bytes)) != -1) { //此函数是阻塞的    System.out.println(new String(bytes,0,len, StandardCharsets.UTF_8));}
while (scanner.hasNextLine()) { //此函数为阻塞的函数        String massage = scanner.nextLine();        out.write((name + " : " + massage).getBytes(StandardCharsets.UTF_8));        if(massage.equals("//exit"))     break;  }
Socket socket = server.accept();//此函数是阻塞的

这些阻塞的函数是需要等待其他的程序,例如scanner.hasNextLine()需要等待程序员的输入才会返回值,in.read需要等待流的另一端传输数据,使用多线程就可以在这些函数处于阻塞状态时,去运行其他的线程。

所以,多线程编程的关键便是那些阻塞的函数。

感谢各位的阅读,以上就是“怎么用Java Socket+多线程实现多人聊天室功能”的内容了,经过本文的学习后,相信大家对怎么用Java Socket+多线程实现多人聊天室功能这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是编程网,小编将为大家推送更多相关知识点的文章,欢迎关注!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

怎么用Java Socket+多线程实现多人聊天室功能

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

怎么用Java Socket+多线程实现多人聊天室功能

这篇文章主要讲解了“怎么用Java Socket+多线程实现多人聊天室功能”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么用Java Socket+多线程实现多人聊天室功能”吧!思路简介分
2023-06-20

Redis实现多人多聊天室功能

本文为大家分享了Redis支持多人多聊天室功能的设计代码,供大家参考,具体内容如下 设计原理左边的一个数据域,代表两个聊天室,聊天室id分别是827,729 在聊天室827里,有2个人,分别是jason22,jeff24他们分别已经阅读过聊
2022-06-04

Node.js+express+socket怎么实现在线实时多人聊天室

本文小编为大家详细介绍“Node.js+express+socket怎么实现在线实时多人聊天室”,内容详细,步骤清晰,细节处理妥当,希望这篇“Node.js+express+socket怎么实现在线实时多人聊天室”文章能帮助大家解决疑惑,下
2023-06-17

java课程设计做一个多人聊天室(socket+多线程)

目录课设要求相关知识点1.服务端能够看到所有在线用户2.服务端能够强制用户下线3.客户端能够看到所有在线用户4.客户端要求能够向某个用户发送消息5.运用JDBC实现持久化存储用户信息6.使用JSONObject对象封装数据7.使用Maven构建管理项目类图项目
2014-10-09

怎么在Java中使用Socket实现一个多人聊天室

本篇文章给大家分享的是有关怎么在Java中使用Socket实现一个多人聊天室,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。流程首先建立一个服务器端,构建ServerSocket
2023-06-14

怎么使用Java socket通信模拟QQ实现多人聊天室

这篇文章主要介绍“怎么使用Java socket通信模拟QQ实现多人聊天室”,在日常操作中,相信很多人在怎么使用Java socket通信模拟QQ实现多人聊天室问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”怎
2023-07-02

怎么使用Java NIO实现多人聊天室

本篇内容主要讲解“怎么使用Java NIO实现多人聊天室”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“怎么使用Java NIO实现多人聊天室”吧!NIO服务端public class NioSe
2023-06-21

java怎么实现多人聊天对话室

这篇文章给大家分享的是有关java怎么实现多人聊天对话室的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。使用ServerSocket和Socket实现服务器端和客户端的Socket通信。了解完socket通信步骤后
2023-06-20

Python+Socket实现多人聊天室,功能:好友聊天、群聊、图片、表情、文件等

一、项目简介 本项目主要基于python实现的多人聊天室,主要的功能如下: 登录注册添加好友与好友进行私聊创建群聊邀请/申请加入群聊聊天发送图片聊天发送表情聊天发送文件聊天记录保存在本地中聊天过程中发送的文件保存本地 二、环境介绍 pyth
2023-08-17

Node.js怎么实现在线实时多人聊天室

本篇内容介绍了“Node.js怎么实现在线实时多人聊天室”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!文件结构如下:前端部分:登录页面Log
2023-07-04

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录