热门标签 | HotTags
当前位置:  开发笔记 > 运维 > 正文

javasocket实现聊天室java实现多人聊天功能

这篇文章主要为大家详细介绍了javasocket实现聊天室,java实现多人聊天功能,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

用java socket做一个聊天室,实现多人聊天的功能。看了极客学院的视频后跟着敲的。(1DAY)

服务端:

1. 先写服务端的类MyServerSocket,里面放一个监听线程,一启动就好
2. 实现服务端监听类ServerListener.java,用accept来监听,一旦有客户端连上,生成新的socket,就新建个线程实例ChatSocket。启动线程后就把线程交给ChatManager管理
3. 在ChatSocket中实现从客户端读取内容,把读取到的内容发给集合内所有的客户端
4. ChatManager里面用vector来管理socket线程实例ChatSocket,并实现发送信息给其他的客户端

客户端:

1. 新建一个继承JFrame的MainWindow.java类,主要实现聊天窗口的UI,以及事件响应。
2. 新建StartClient.java类,把MainWindow中生成MainWindow主方法部分代码拷贝过来,这样就能在主程序中把窗体执行出来了。
3. 新建ChatManager(需要单例化的类)管理socket,实现聊天的输入输出功能。最后记得在1中新建窗口后,传一份frame的引用到ChatManager中,才能实现ChatManager对界面的显示。

工程结构如图

以下为代码

服务端:

1. 先写服务端的类MyServerSocket,里面放一个监听线程,一启动就好

package com.starnet.testserversocket.main;


public class MyServerSocket {
  public static void main(String[] args) {
    new ServerListener().start();
  }
}

2.实现服务端监听类ServerListener.java,用accept来监听,一旦有客户端连上,生成新的socket,就新建个线程实例ChatSocket。启动线程后就把线程交给ChatManager管理

package com.starnet.testserversocket.main;

import java.io.IOException;
import java.net.ServerSocket;
import java.net.Socket;

import javax.swing.JOptionPane;

public class ServerListener extends Thread {
  public void run() {
    try {
      ServerSocket serverSocket = new ServerSocket(23456);
      while (true) {
        // block
        Socket socket = serverSocket.accept();
        // 建立链接 
        JOptionPane.showMessageDialog(null, "有客户端连到本机23456端口");
        //将socket传递给新的线程
        ChatSocket cs= new ChatSocket(socket);
        cs.start();
        ChatManager.getChatManager().add(cs);
      }
    } catch (IOException e) {
      e.printStackTrace();
    }
  }
}

3.在ChatSocket中实现从客户端读取内容,把读取到的内容发给集合内所有的客户端

package com.starnet.testserversocket.main;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.UnsupportedEncodingException;
import java.net.Socket;
/*每一个连上的客户端,服务器都有一个线程为之服务*/
public class ChatSocket extends Thread {
  Socket socket;

  public ChatSocket(Socket s) {
    this.socket = s;
  }

  //发送数据
  public void out(String out) {
    try {
      socket.getOutputStream().write(out.getBytes("UTF-8"));
    } catch (UnsupportedEncodingException e) {
      // TODO Auto-generated catch block
      e.printStackTrace();
    } catch (IOException e) {
      // TODO Auto-generated catch block
      e.printStackTrace();
    }
  }

  //服务器会不断地从客户端读取内容,把读取到的内容发给集合内所有的客户端。
  public void run() {
    try {
      //接收数据
      BufferedReader br = new BufferedReader(
          new InputStreamReader(
              socket.getInputStream(), "UTF-8"));
      String line=null;
      //发送读到的内容
      while ((line = br.readLine())!=null) {
        System.out.println(line);
        ChatManager.getChatManager().publish(this, line);
      }
      br.close();
    } catch (IOException e) {
      // TODO Auto-generated catch block
      e.printStackTrace();
    }
  }
}

4.ChatManager里面用vector来管理socket线程实例ChatSocket,并实现发送信息给其他的客户端

package com.starnet.testserversocket.main;

import java.util.Vector;
//一个聊天服务器只能有一个manager,要单例化处理
public class ChatManager {
  private ChatManager(){}
  private static final ChatManager CM=new ChatManager();
  public static ChatManager getChatManager(){
    return CM;
  }

  Vector vector = new Vector();
  /*增加ChatSocket 实例到vector中*/
  public void add(ChatSocket cs){
    vector.add(cs);
  }

  /*发布消息给其他客户端
   *ChatSocket cs: 调用publish的线程
   *msg:要发送的信息 */
  public void publish(ChatSocket cs, String msg){
    for (int i = 0; i 

客户端:

1. 新建一个继承JFrame的MainWindow.java类,主要实现聊天窗口的UI,以及事件响应。

package com.starnet.javaclient.view;

import java.awt.BorderLayout;
import java.awt.EventQueue;

import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.border.EmptyBorder;
import javax.swing.JTextArea;
import javax.swing.GroupLayout;
import javax.swing.GroupLayout.Alignment;
import javax.swing.JTextField;
import javax.swing.JButton;
import javax.swing.LayoutStyle.ComponentPlacement;

import com.starnet.javaclient.main.ChatManager;
import com.sun.xml.internal.ws.client.SenderException;

import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;

public class MainWindow extends JFrame {

  /**
   * 
   */
  private static final long serialVersiOnUID= 1L;
  private JPanel contentPane;
  private JTextArea txt;
  private JTextField txtip;
  private JTextField txtSend;

  /**
   * Create the frame.
   */
  public MainWindow() {
    setAlwaysOnTop(true);
    setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    setBounds(100, 100, 450, 300);
    cOntentPane= new JPanel();
    contentPane.setBorder(new EmptyBorder(5, 5, 5, 5));
    setContentPane(contentPane);

    txt = new JTextArea();
    txt.setText("ready...");

    txtip = new JTextField();
    txtip.setText("127.0.0.1");
    txtip.setColumns(10);

    JButton btnCOnnect= new JButton("connect");
    btnConnect.addMouseListener(new MouseAdapter() {
      @Override
      public void mouseClicked(MouseEvent e) {
        ChatManager.getChatManager().connect(txtip.getText());
      }
    });

    txtSend = new JTextField();
    txtSend.setText("hello");
    txtSend.setColumns(10);

    JButton btnSend = new JButton("send");
    btnSend.addMouseListener(new MouseAdapter() {
      @Override
      public void mouseClicked(MouseEvent e) {
        ChatManager.getChatManager().send(txtSend.getText());
        appendText("我说: "+txtSend.getText());
        txtSend.setText("");
      }
    });
    GroupLayout gl_cOntentPane= new GroupLayout(contentPane);
    gl_contentPane.setHorizontalGroup(
      gl_contentPane.createParallelGroup(Alignment.LEADING)
        .addGroup(Alignment.TRAILING, gl_contentPane.createSequentialGroup()
          .addGroup(gl_contentPane.createParallelGroup(Alignment.TRAILING)
            .addGroup(gl_contentPane.createSequentialGroup()
              .addComponent(txtSend, GroupLayout.DEFAULT_SIZE, 325, Short.MAX_VALUE)
              .addPreferredGap(ComponentPlacement.RELATED)
              .addComponent(btnSend, GroupLayout.PREFERRED_SIZE, 109, GroupLayout.PREFERRED_SIZE))
            .addGroup(Alignment.LEADING, gl_contentPane.createSequentialGroup()
              .addComponent(txtip, GroupLayout.PREFERRED_SIZE, 294, GroupLayout.PREFERRED_SIZE)
              .addPreferredGap(ComponentPlacement.RELATED)
              .addComponent(btnConnect, GroupLayout.DEFAULT_SIZE, 140, Short.MAX_VALUE))
            .addComponent(txt, GroupLayout.DEFAULT_SIZE, 434, Short.MAX_VALUE))
          .addContainerGap())
    );
    gl_contentPane.setVerticalGroup(
      gl_contentPane.createParallelGroup(Alignment.LEADING)
        .addGroup(gl_contentPane.createSequentialGroup()
          .addGroup(gl_contentPane.createParallelGroup(Alignment.BASELINE)
            .addComponent(txtip, GroupLayout.PREFERRED_SIZE, GroupLayout.DEFAULT_SIZE, GroupLayout.PREFERRED_SIZE)
            .addComponent(btnConnect))
          .addPreferredGap(ComponentPlacement.RELATED)
          .addComponent(txt, GroupLayout.DEFAULT_SIZE, 198, Short.MAX_VALUE)
          .addPreferredGap(ComponentPlacement.RELATED)
          .addGroup(gl_contentPane.createParallelGroup(Alignment.TRAILING)
            .addComponent(btnSend)
            .addComponent(txtSend, GroupLayout.PREFERRED_SIZE, GroupLayout.DEFAULT_SIZE, GroupLayout.PREFERRED_SIZE)))
    );
    contentPane.setLayout(gl_contentPane);
  }
  /*客户端发送的内容添加到中间的txt控件中*/
  public void appendText(String in) {
    txt.append("\n" + in);
  }
}

2.新建StartClient.java类,把MainWindow中生成MainWindow主方法部分代码拷贝过来,这样就能在主程序中把窗体执行出来了。

package com.starnet.javaclient.main;

import java.awt.EventQueue;

import com.starnet.javaclient.view.MainWindow;

public class StartClient {
  public static void main(String[] args) {
    /*先新建一个jframe,然后把自动生成的代码贴过来*/
    EventQueue.invokeLater(new Runnable() {
      public void run() {
        try {
          MainWindow frame = new MainWindow();
          frame.setVisible(true);
          //创建完这个frame以后,传一份window的引用到ChatManager去
          ChatManager.getChatManager().setWindow(frame);
        } catch (Exception e) {
          e.printStackTrace();
        }
      }
    });
  }
}

3.新建ChatManager(需要单例化的类)管理socket,实现聊天的输入输出功能。最后记得在1中新建窗口后,传一份frame的引用到ChatManager中,才能实现ChatManager对界面的显示。

package com.starnet.javaclient.main;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;
import java.io.PrintWriter;
import java.net.Socket;
import java.net.UnknownHostException;

import com.starnet.javaclient.view.MainWindow;

public class ChatManager {
  private ChatManager(){}
  private static final ChatManager instance=new ChatManager();
  public static ChatManager getChatManager(){
    return instance;
  }
  MainWindow window;//为了能在界面上显示服务器发来的信息,就需要传一个MainWindow的引用进来
  Socket socket;
  private String IP;
  BufferedReader bReader;
  PrintWriter pWriter;
  public void setWindow(MainWindow window) {
    this.window = window;
    window.appendText("文本框已经和chatManager绑定了");
  }
  public void connect(String ip) {
    this.IP = ip;
    new Thread(){

      @Override
      public void run() {
        //实现网络方法
        try {
          socket = new Socket(IP, 23456);
          //输出流
          pWriter = new PrintWriter(
              new OutputStreamWriter(
                  socket.getOutputStream()));
          //输入流
          bReader = new BufferedReader(
              new InputStreamReader(
                  socket.getInputStream()));

          String line = null;
          //如果读取数据为空
          while ((line = bReader.readLine())!=null) {
            window.appendText("收到: "+line);
          }
          //读完数据之后要关闭
          pWriter.close();
          bReader.close();
          pWriter = null;
          bReader = null;

        } catch (UnknownHostException e) {
          e.printStackTrace();
        } catch (IOException e) {
          e.printStackTrace();
        }
      }
    }.start();
  }

  public void send(String sendMsg){
    if (pWriter!=null) {
      pWriter.write(sendMsg+"\n");
      pWriter.flush();
    } else {
      window.appendText("当前链接已经中断...");
    }
  }
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • 在Java分层设计模式中,典型的三层架构(3-tier application)将业务应用细分为表现层(UI)、业务逻辑层(BLL)和数据访问层(DAL)。这种分层结构不仅有助于提高代码的可维护性和可扩展性,还能有效分离关注点,使各层职责更加明确。通过合理的设计和实现,三层架构能够显著提升系统的整体性能和稳定性。 ... [详细]
  • 在探讨Hibernate框架的高级特性时,缓存机制和懒加载策略是提升数据操作效率的关键要素。缓存策略能够显著减少数据库访问次数,从而提高应用性能,特别是在处理频繁访问的数据时。Hibernate提供了多层次的缓存支持,包括一级缓存和二级缓存,以满足不同场景下的需求。懒加载策略则通过按需加载关联对象,进一步优化了资源利用和响应时间。本文将深入分析这些机制的实现原理及其最佳实践。 ... [详细]
  • 本文深入探讨了Ajax的工作机制及其在现代Web开发中的应用。Ajax作为一种异步通信技术,改变了传统的客户端与服务器直接交互的模式。通过引入Ajax,客户端与服务器之间的通信变得更加高效和灵活。文章详细分析了Ajax的核心原理,包括XMLHttpRequest对象的使用、数据传输格式(如JSON和XML)以及事件处理机制。此外,还介绍了Ajax在提升用户体验、实现动态页面更新等方面的具体应用,并讨论了其在当前Web开发中的重要性和未来发展趋势。 ... [详细]
  • REST与RPC:选择哪种API架构风格?
    在探讨REST与RPC这两种API架构风格的选择时,本文首先介绍了RPC(远程过程调用)的概念。RPC允许客户端通过网络调用远程服务器上的函数或方法,从而实现分布式系统的功能调用。相比之下,REST(Representational State Transfer)则基于资源的交互模型,通过HTTP协议进行数据传输和操作。本文将详细分析两种架构风格的特点、适用场景及其优缺点,帮助开发者根据具体需求做出合适的选择。 ... [详细]
  • SSAS入门指南:基础知识与核心概念解析
    ### SSAS入门指南:基础知识与核心概念解析Analysis Services 是一种专为决策支持和商业智能(BI)解决方案设计的数据引擎。该引擎能够为报告和客户端应用提供高效的分析数据,并支持在多维数据模型中构建高性能的分析应用。通过其强大的数据处理能力和灵活的数据建模功能,Analysis Services 成为了现代 BI 系统的重要组成部分。 ... [详细]
  • Spring框架的核心组件与架构解析 ... [详细]
  • 本文详细解析了微信服务端示例类的功能与应用。其中,`ClientResponseHandler` 类主要用于处理微信支付所需的响应数据,而 `TenpayHttpClient` 则是对 HTTP 请求(包括 GET 和 POST 方法)进行了封装,以便在内部调用时更加便捷和高效。这些工具类在实际开发中起到了关键作用,开发者无需深入了解其底层实现细节,即可轻松集成微信支付功能。 ... [详细]
  • 本文详细介绍了如何安全地手动卸载Exchange Server 2003,以确保系统的稳定性和数据的完整性。根据微软官方支持文档(https://support.microsoft.com/kb833396/zh-cn),在进行卸载操作前,需要特别注意备份重要数据,并遵循一系列严格的步骤,以避免对现有网络环境造成不利影响。此外,文章还提供了详细的故障排除指南,帮助管理员在遇到问题时能够迅速解决,确保整个卸载过程顺利进行。 ... [详细]
  • 在CICS应用环境中,众多客户端通过网络与CICS服务器进行连接。系统管理员可以通过CICS系统交易CEMT查询当前连接的客户端信息。然而,在非客户端模式下,识别用户连接并解决信息获取错误的问题变得更为复杂。本文将探讨如何在CICS服务器端准确识别非客户端模式的用户连接,并提供有效的解决方案,以确保系统的稳定性和数据的准确性。此外,还将介绍一些常用的诊断工具和技术,帮助管理员快速定位和解决问题。 ... [详细]
  • 本文探讨了如何有效地构建和优化微信公众平台账号,涵盖了用户信息管理、内容创作与发布、互动策略及数据分析等方面。通过合理设置用户信息字段,如用户名、昵称、密码、真实姓名和性别等,确保账号的安全性和用户体验。同时,文章还介绍了如何利用微信公众平台的各项功能,提升用户参与度和品牌影响力。 ... [详细]
  • 本文探讨了资源访问的学习路径与方法,旨在帮助学习者更高效地获取和利用各类资源。通过分析不同资源的特点和应用场景,提出了多种实用的学习策略和技术手段,为学习者提供了系统的指导和建议。 ... [详细]
  • JavaScript XML操作实用工具类:XmlUtilsJS技巧与应用 ... [详细]
  • Java解析YAML文件并转换为JSON格式(支持JSON与XML的结构化查询)
    本文探讨了如何利用Java解析YAML文件并将其转换为JSON格式,同时支持JSON和XML的结构化查询。YAML、JSON和XML这三种数据格式通过其名称作为文件扩展名,便于区分和使用。文章详细介绍了这些格式的层次结构和数据表示方法,并重点讨论了在数据传输过程中,XML的特性和优势。此外,还提供了具体的代码示例和实现步骤,帮助开发者高效地进行数据格式转换和查询操作。 ... [详细]
  • 在前文探讨了Spring如何为特定的bean选择合适的通知器后,本文将进一步深入分析Spring AOP框架中代理对象的生成机制。具体而言,我们将详细解析如何通过代理技术将通知器(Advisor)中包含的通知(Advice)应用到目标bean上,以实现切面编程的核心功能。 ... [详细]
  • 深入解析:React与Webpack配置进阶指南(第二部分)
    在本篇进阶指南的第二部分中,我们将继续探讨 React 与 Webpack 的高级配置技巧。通过实际案例,我们将展示如何使用 React 和 Webpack 构建一个简单的 Todo 应用程序,具体包括 `TodoApp.js` 文件中的代码实现,如导入 React 和自定义组件 `TodoList`。此外,我们还将深入讲解 Webpack 配置文件的优化方法,以提升开发效率和应用性能。 ... [详细]
author-avatar
彭伟波2013
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有