java聊天程序源代码

简介: 服务端: import java.io.*;import java.net.*;import java.util.

服务端:

 

import java.io.*;

import java.net.*;

import java.util.*;

 

public class ChatServer {

 boolean stat = false;

 ServerSocket ss = null;

 

 List<Client> clients = new ArrayList<Client>();//用于存客户端

 

 public static void main(String[] args) {

  new ChatServer().start();

 }

 

 public void start(){

  try {

   ss = new ServerSocket(8888);

   stat = true;

  } catch(BindException e){  //Sever端已经运行,当重复运行时抛异常

   System.out.println("端口正在使用中。。。。");

   System.out.println("请关掉相关程序并重新运行服务器!"); //还会抛别的异常,所以直接关闭窗口

   System.exit(0);

  } catch(IOException e) {

   e.printStackTrace();

  }

 

  try{

   while(stat){

    Socket s = ss.accept();

System.out.println("a client connected!" );  //测试语句写在最左边,以后没用可以删除或注掉

    Client c = new Client(s);    //每建立一个客户端,就new一个客户端对象,启动一个线程

    new Thread(c).start();

    clients.add(c);  //勿忘写,将每个客户端加入到容器里

   }

  } catch (IOException e) {

   e.printStackTrace();

  } finally {

   try {

    ss.close();

   } catch (IOException e) {

    e.printStackTrace();

   }

  }

 }

 

 class Client implements Runnable {

  private Socket s;

  private DataInputStream dis;

  private DataOutputStream dos;

  private boolean cont = false;

 

  public Client(Socket s){

   this.s = s; 

   try {

    dis = new DataInputStream(s.getInputStream());//初始化

    dos = new DataOutputStream(s.getOutputStream());

    cont = true;

   } catch (IOException e) {

    e.printStackTrace();

   }

  }

 

  public void send(String str){  //用于发送给客户端

   try {

    dos.writeUTF(str);

   } catch (IOException e) {

    clients.remove(this);  //移除那个退出的对象

    System.out.println("一个客户退出了");

    //e.printStackTrace();

   }

  }

 

  public void run() {

   try{

    while(cont){

     String str = dis.readUTF(); //阻塞式方法

System.out.println(str);

     for(int i=0; i<clients.size(); i++){

      Client c = clients.get(i);  //取客户端

      c.send(str); 

     }

     /*     另外两种方法,但不适用,它会锁定服务端

     for(Iterator<Client> it = clients.iterator(); it.hasNext();){

      Client c = it.next();

      c.send(str);

     }

    

     Iterator<Client> it = clients.iterator();

     while(it.hasNext()){

      Client c = it.next();

      c.send(str);

     }

     */

    }

   } catch (EOFException e){   //readUTF()阻塞式方法,所以关闭客户端会抛异常

    System.out.println("Client closed!");

   } catch (IOException e) {

    e.printStackTrace();

   } finally {

    try {

     if(dis != null) dis.close();

     if(dos != null) dos.close();

     if(s != null) {

      s.close();

      s = null;//更严格的方法,等于空就没人去用了,垃圾收集器就回收走

     }

    } catch (IOException e) {

     e.printStackTrace();

    }

   }

  }

 }

}

 

 

客户端:

import java.awt.*;

import java.awt.event.*;

import java.io.*;

import java.net.*;

 

public class ChatClient extends Frame {

 Socket s = null;

 DataOutputStream dos = null;

 DataInputStream dis = null;

 private boolean cont = false;

 

 TextField tfTxt = new TextField();

 TextArea taContent = new TextArea();

 

 Thread tRecv = new Thread(new RecvThread());

 

 public static void main(String[] args) {

  new ChatClient().launchFrame();

 }

 

 public void launchFrame() {

  setLocation(400, 300);

  this.setSize(300, 300);

  add(tfTxt,BorderLayout.SOUTH);

  add(taContent,BorderLayout.NORTH);

  pack(); //包在一起,去掉中间空着的

  this.addWindowListener(new WindowAdapter(){

   public void windowClosing(WindowEvent e){

    disconnect();

    System.exit(0);

   }

  });

  tfTxt.addActionListener(new TfListent());

  setVisible(true);

  connect();

  tRecv.start();   //启动线程

 }

 

 public void connect(){

  try {

   s = new Socket("127.0.0.1",8888);//注意不要定义成Socket s,这就成了局部变量而不是成员变量了

System.out.println("connected!");

   dos = new DataOutputStream(s.getOutputStream());

   dis = new DataInputStream(s.getInputStream());

   cont = true;

  } catch (UnknownHostException e) {

   e.printStackTrace();

  } catch (IOException e) {

   e.printStackTrace();

  }

 

 }

 

 public void disconnect(){

  try {

   dos.close();

   dis.close();

   s.close();

  } catch (IOException e) {

   e.printStackTrace();

  }

 

  /*//无法解决readUTF阻塞式方法

  try {

   cont = false;  //关闭线程

   tRecv.join();  //合并线程,彻底让他停止

  } catch (InterruptedException  e) {

   e.printStackTrace();

  } finally {

   try {

    dos.close(); //线程停止之后才能关流,不然抛SocketException异常

    dis.close();

    s.close();

   } catch (IOException e) {

    e.printStackTrace();

   }

  }

  */

 }

 

 private class TfListent implements ActionListener {

 

  public void actionPerformed(ActionEvent e) {

   String str = tfTxt.getText().trim();

   tfTxt.setText("");

  

   try {

    dos.writeUTF(str);

    dos.flush();

 

   } catch (IOException e1) {

    e1.printStackTrace();

   }

  

  }

 

 }

 

 private class RecvThread implements Runnable{

 

  public void run() {

   try {

    while(cont){

     String str = dis.readUTF();

     taContent.setText(taContent.getText() + str + '\n');

    }

   } catch (SocketException e){

    System.out.println("退出了,bye!");

   } catch (IOException e) {  

    e.printStackTrace();

   }

  }

 

 }

}

 

相关文章
|
12天前
|
Java 流计算
利用java8 的 CompletableFuture 优化 Flink 程序
本文探讨了Flink使用avatorscript脚本语言时遇到的性能瓶颈,并通过CompletableFuture优化代码,显著提升了Flink的QPS。文中详细介绍了avatorscript的使用方法,包括自定义函数、从Map中取值、使用Java工具类及AviatorScript函数等,帮助读者更好地理解和应用avatorscript。
利用java8 的 CompletableFuture 优化 Flink 程序
|
29天前
|
XML 存储 JSON
Java程序部署
Java程序部署
|
2天前
|
Java Maven 数据安全/隐私保护
如何实现Java打包程序的加密代码混淆,避免被反编译?
【10月更文挑战第15天】如何实现Java打包程序的加密代码混淆,避免被反编译?
12 2
|
5天前
|
安全 Java Linux
java程序设置开机自启
java程序设置开机自启
|
8天前
|
运维 Java Linux
【运维基础知识】Linux服务器下手写启停Java程序脚本start.sh stop.sh及详细说明
### 启动Java程序脚本 `start.sh` 此脚本用于启动一个Java程序,设置JVM字符集为GBK,最大堆内存为3000M,并将程序的日志输出到`output.log`文件中,同时在后台运行。 ### 停止Java程序脚本 `stop.sh` 此脚本用于停止指定名称的服务(如`QuoteServer`),通过查找并终止该服务的Java进程,输出操作结果以确认是否成功。
14 1
|
1月前
|
消息中间件 分布式计算 Java
Linux环境下 java程序提交spark任务到Yarn报错
Linux环境下 java程序提交spark任务到Yarn报错
34 5
|
1月前
|
Java 编译器 数据库连接
探索Java中的异常处理:提升程序的鲁棒性
【9月更文挑战第25天】在Java的世界里,异常是那些不请自来、令人头疼的“客人”。它们悄无声息地潜入我们的代码,一旦出现,便可能导致程序崩溃或行为异常。但是,如果能够妥善管理这些异常,我们就能将潜在的灾难转变为增强程序鲁棒性和用户体验的机会。本文将通过深入浅出的方式,带领读者理解Java异常处理的重要性,并提供实用的策略来优雅地处理这些意外情况。让我们一起学习如何在Java中捕捉、处理和预防异常,确保我们的程序即使在面对不可预见的错误时也能保持稳健运行。
|
10天前
|
Java Python
如何通过Java程序调用python脚本
如何通过Java程序调用python脚本
11 0
|
1月前
|
监控 Java 数据库
Java程序如何进行不停机更新?
Java程序如何进行不停机更新?
40 1
|
16天前
|
Java
java的程序记录时间
java的程序记录时间
21 0