QQ登录

只需要一步,快速开始

APP扫码登录

只需要一步,快速开始

手机号码,快捷登录

泡泡马甲APP 更多内容请下载泡泡马甲手机客户端APP 立即下载 ×
查看: 4311|回复: 0

[JAVA/JSP] Java Socket+多线程实现多人聊天室功能

[复制链接]

等级头衔

积分成就    金币 : 2802
   泡泡 : 1516
   精华 : 6
   在线时间 : 1242 小时
   最后登录 : 2024-4-18

丰功伟绩

优秀达人突出贡献荣誉管理论坛元老

联系方式
发表于 2021-7-16 08:49:51 | 显示全部楼层 |阅读模式
一、思路简介5 Q2 u) {1 ^6 A7 c& ]
       分为客户端和服务器两个类,所有的客户端将聊的内容发送给服务器,服务器接受后,将每一条内容发送给每一个客户端,客户端再显示在终端上。+ u( J& e% N# q$ x- r, I) S) \& g
二、客户端设计5 p/ B. ?6 p4 Y8 B) \2 o& }, }
       客户端包含2个线程,1个用来接受服务器的信息,再显示,1个用来接收键盘的输入,发送给服务器。: c! W3 g/ N$ D
  1. import java.io.IOException;
  2. import java.io.InputStream;
  3. import java.io.OutputStream;
  4. import java.net.Socket;
  5. import java.nio.charset.StandardCharsets;
  6. import java.util.Scanner;
  7. public class WeChatClient {  //WeChat的客户端类
  8.     private Socket client;
  9.     private String name;
  10.     private InputStream in;
  11.     private OutputStream out;
  12.     private MassageSenter massageSenter;
  13.     private MassageGeter massageGeter;
  14.     class MassageGeter extends Thread{  //一个子线程类,用于客户端接收消息
  15.         MassageGeter() throws IOException{
  16.             in = client.getInputStream();
  17.         }
  18.         @Override
  19.         public void run() {
  20.             int len;
  21.             byte[] bytes = new byte[1024];
  22.             try {
  23.                 while ((len = in.read(bytes)) != -1) { //此函数是阻塞的
  24.                     System.out.println(new String(bytes,0,len, StandardCharsets.UTF_8));
  25.                 }
  26.             }catch (IOException e){
  27.                 System.out.println(e.toString());
  28.             }
  29.             System.out.println("Connection interruption");
  30.         }
  31.     }
  32.     class MassageSenter extends Thread{  //一个子线程类,用于发送消息给服务器
  33.         MassageSenter() throws IOException{
  34.             out = client.getOutputStream();
  35.         }
  36.         @Override
  37.         public void run() {
  38.             Scanner scanner = new Scanner(System.in);
  39.             try {
  40.                 while (scanner.hasNextLine()) { //此函数为阻塞的函数
  41.                     String massage = scanner.nextLine();
  42.                     out.write((name + " : " + massage).getBytes(StandardCharsets.UTF_8));
  43.                     if(massage.equals("//exit"))
  44.                         break;
  45.                 }
  46.             }catch (IOException e){
  47.                 e.printStackTrace();
  48.             }
  49.         }
  50.     }
  51.     WeChatClient(String name, String host, int port) throws IOException {//初始化,实例化发送和接收2个线程
  52.         this.name = name;
  53.         client = new Socket(host,port);
  54.         massageGeter = new MassageGeter();
  55.         massageSenter = new MassageSenter();
  56.     }
  57.     void login() throws IOException{//登录时,先发送名字给服务器,在接收到服务器的正确回应之后,启动线程
  58.         out.write(name.getBytes(StandardCharsets.UTF_8));
  59.         byte[] bytes = new byte[1024];
  60.         int len;
  61.         len = in.read(bytes);
  62.         String answer = new String(bytes,0,len, StandardCharsets.UTF_8);
  63.         if(answer.equals("logined!")) {
  64.             System.out.println("Welcome to WeChat! "+name);
  65.             massageSenter.start();
  66.             massageGeter.start();
  67.             try {
  68.                 massageSenter.join();//join()的作用是等线程结束之后再继续执行主线程(main)
  69.                 massageGeter.join();
  70.             }catch (InterruptedException e){
  71.                 System.err.println(e.toString());
  72.             }
  73.         }else{
  74.             System.out.println("Server Wrong");
  75.         }
  76.         client.close();
  77.     }
  78.     public static void main(String[] args) throws IOException{//程序入口
  79.         String host = "127.0.0.1";
  80.         WeChatClient client = new WeChatClient("Uzi",host,7777);
  81.         client.login();
  82.     }
  83. }
三、服务器设计
% u! ~* X5 k5 o6 p       服务器包含3个线程类,端口监听线程,客户端接收信息线程,发送信息线程。服务器类还包含并维护着一个已经连接的用户列表,和一个待发送信息列表。$ P% k4 J6 d' {
       服务器有一个负责监听端口的线程,此线程在接收到客户端的连接请求后,将连接的客户端添加进用户列表;并为每一个连接的客户端实例化一个接受信息的线程类,从各个客户端接收员信息,并存入待发送信息列表。发送信息线程查看列表是否为空,若不为空,则将里面的信息发送给用户列表的每一个用户。
# R5 p* T: t5 m! X9 z1 c2 V
  1. import java.io.IOException;
  2. import java.io.InputStream;
  3. import java.io.OutputStream;
  4. import java.net.ServerSocket;
  5. import java.net.Socket;
  6. import java.nio.charset.StandardCharsets;
  7. import java.util.ArrayList;
  8. public class WeChatServer {
  9.     private ServerSocket server;
  10.     private ArrayList<User> users;//用户列表
  11.     private ArrayList<String> massages;//待发送消息队列
  12.     private Listener listener;
  13.     private MassageSenter massageSenter;
  14.     class User{  //用户类,包含用户的登录id和一个输出流
  15.         String name;
  16.         OutputStream out;
  17.         User(String name,OutputStream out){
  18.             this.name = name;
  19.             this.out = out;
  20.         }
  21.         @Override
  22.         public String toString() {
  23.             return name;
  24.         }
  25.     }
  26.     private static String GetMassage(InputStream in) throws IOException{//从一个输入流接收一个字符串
  27.         int len;
  28.         byte[] bytes = new byte[1024];
  29.         len = in.read(bytes);
  30.         return new String(bytes,0,len,StandardCharsets.UTF_8);
  31.     }
  32.     private void UserList(){  //列出当前在线用户,调试用
  33.         for(User user : users)
  34.             System.out.println(user);
  35.     }
  36.     class Listener extends Thread{ //监听线程类,负则监听是否有客户端连接
  37.         @Override
  38.         public void run() {
  39.             try {
  40.                 while (true) {
  41.                     Socket socket = server.accept();//此函数是阻塞的
  42.                     InputStream in = socket.getInputStream();
  43.                     String name = GetMassage(in);//获取接入用户的name
  44.                     System.out.println(name +" has connected");
  45.                     massages.add(name+" has joined just now!!");//向聊天室报告用户连入的信息
  46.                     OutputStream out = socket.getOutputStream();
  47.                     out.write("logined!".getBytes(StandardCharsets.UTF_8));//发送成功建立连接的反馈
  48.                     User user = new User(name,out);
  49.                     users.add(user);//添加至在线用户列表
  50.                     MassageListener listener = new MassageListener(user,in);//创建用于接收此用户信息的线程
  51.                     listener.start();
  52.                 }
  53.             }catch (IOException e){
  54.                 e.printStackTrace();
  55.             }
  56.         }
  57.     }
  58.     class MassageListener extends Thread{ //接收线程类,用于从一个客户端接收信息,并加入待发送列表
  59.         private User user;
  60.         private InputStream in;
  61.         MassageListener(User user,InputStream in){
  62.             this.user = user;
  63.             this.in = in;
  64.         }
  65.         @Override
  66.         public void run() {
  67.             try {
  68.                 while (true){
  69.                     String massage = GetMassage(in);
  70.                     System.out.println("GET MASSAGE  "+massage);
  71.                     if(massage.contains("//exit")){ //       "/exit" 是退出指令
  72.                         break;
  73.                     }
  74.                     massages.add(massage);
  75.                 }//用户退出有两种形式,输入 “//exit” 或者直接关闭程序
  76.                 in.close();
  77.                 user.out.close();
  78.             }catch (IOException e){//此异常是处理客户端异常关闭,即GetMassage(in)调用会抛出异常,因为in出入流已经自动关闭
  79.                 e.printStackTrace();
  80.             }finally {
  81.                 System.out.println(user.name+" has exited!!");
  82.                 massages.add(user.name+" has exited!!");
  83.                 users.remove(user);//必须将已经断开连接的用户从用户列表中移除,否则会在发送信息时产生异常
  84.                 System.out.println("Now the users has");
  85.                 UserList();
  86.             }
  87.         }
  88.     }
  89.     private synchronized void SentToAll(String massage)throws IOException{//将信息发送给每一个用户,加入synchronized修饰,保证在发送时,用户列表不会被其他线程更改
  90.         if(users.isEmpty())
  91.             return;
  92.         for(User user : users){
  93.             user.out.write(massage.getBytes(StandardCharsets.UTF_8));
  94.         }
  95.     }
  96.     class MassageSenter extends Thread{//消息发送线程
  97.         @Override
  98.         public void run() {
  99.             while(true){
  100.                 try{
  101.                     sleep(1);//此线程中没有阻塞的函数,加入沉睡语句防止线程过多抢占资源
  102.                 }catch (InterruptedException e){
  103.                     e.printStackTrace();
  104.                 }
  105.                 if(!massages.isEmpty()){
  106.                     String massage = massages.get(0);
  107.                     massages.remove(0);
  108.                     try {
  109.                         SentToAll(massage);
  110.                     }catch (IOException e){
  111.                         e.printStackTrace();
  112.                     }
  113.                 }
  114.             }
  115.         }
  116.     }
  117.     WeChatServer(int port) throws IOException {  //初始化
  118.         server = new ServerSocket(port);
  119.         users = new ArrayList<>();
  120.         massages = new ArrayList<>();
  121.         listener = new Listener();
  122.         massageSenter = new MassageSenter();
  123.     }
  124.     private void start(){ //线程启动
  125.         listener.start();
  126.         massageSenter.start();
  127.     }
  128.     public static void main(String[] args) throws IOException{
  129.         WeChatServer server = new WeChatServer(7777);
  130.         server.start();
  131.     }
  132. }
四、总结
8 w# c0 J+ p# ^3 U4 S% P# L4 s       之所以需要多线程编程,是因为有的函数是阻塞的,例如:
9 U$ ]8 R8 w/ G. n( }
  1. while ((len = in.read(bytes)) != -1) { //此函数是阻塞的
  2.     System.out.println(new String(bytes,0,len, StandardCharsets.UTF_8));
  3. }
      这些阻塞的函数是需要等待其他的程序,例如scanner.hasNextLine()需要等待程序员的输入才会返回值,in.read需要等待流的另一端传输数据,使用多线程就可以在这些函数处于阻塞状态时,去运行其他的线程。所以,多线程编程的关键便是那些阻塞的函数。
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|手机版|小黑屋|paopaomj.COM ( 渝ICP备18007172号 )

GMT+8, 2024-4-28 11:41

Powered by paopaomj X3.4 © 2016-2024 sitemap

快速回复 返回顶部 返回列表