JAVA套接字之TCP实现聊天室
java实现聊天室,通过多线程实现随时加入,随时退出
1.客户端程序
客户端有两个线程
一个线程由主类SocketClient实现向服务器发送消息
一个线程由内部类readLineThread实现监听服务器发来的消息并显示
import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.io.PrintWriter;
import java.net.Socket;
public class SocketClient extends Socket{
private static final String SERVER_IP = "127.0.0.1";
private static final int SERVER_PORT = 2017;
private Socket client;
private PrintWriter out;
private BufferedReader in;
/**
* 与服务器连接,并输入发送消息
*/
public SocketClient() throws Exception{
super(SERVER_IP, SERVER_PORT);
client = this;
out = new PrintWriter(this.getOutputStream(), true);
in = new BufferedReader(new InputStreamReader(this.getInputStream()));
new readLineThread();
while(true){
in = new BufferedReader(new InputStreamReader(System.in));
String input = in.readLine();
out.println(input);
}
}
/**
* 用于监听服务器端向客户端发送消息线程类
*/
class readLineThread extends Thread{
private BufferedReader buff;
public readLineThread(){
try {
buff = new BufferedReader(new InputStreamReader(client.getInputStream()));
start();
} catch (Exception e) {
}
}
@Override
public void run() {
try {
while(true){
String result = buff.readLine();
if("byeClient".equals(result)){//客户端申请退出,服务端返回确认退出
break;
}else{//输出服务端发送消息
System.out.println(result);
}
}
in.close();
out.close();
client.close();
} catch (Exception e) {
}
}
}
public static void main(String[] args) {
try {
new SocketClient();//启动客户端
} catch (Exception e) {
}
}
}
2.服务器程序
服务器由三个类实现
主类Server监听客户端请求,并启用线程处理请求
内部类PrintOutThread监听输出消息请求,将消息发送到所有客户端
内部类ServerThread提供与每一个用户的连接
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.PrintWriter;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;
public class Server extends ServerSocket {
private static final int SERVER_PORT = 2017;
private static boolean isPrint = false;// 是否输出消息标志
private static List user_list = new ArrayList();// 登录用户集合
private static List<ServerThread> thread_list = new ArrayList<ServerThread>();// 服务器已启用线程集合
private static LinkedList message_list = new LinkedList();// 存放消息队列
/**
* 创建服务端Socket,创建向客户端发送消息线程,监听客户端请求并处理
*/
public Server() throws IOException {
super(SERVER_PORT);// 创建ServerSocket
new PrintOutThread();// 创建向客户端发送消息线程
try {
while (true) {// 监听客户端请求,启用一个线程处理
Socket socket = accept();
new ServerThread(socket);
}
} catch (Exception e) {
} finally {
close();
}
}
/**
* 监听是否有输出消息请求线程类,向客户端发送消息
*/
class PrintOutThread extends Thread {
public PrintOutThread() {
start();
}
@Override
public void run() {
while (true) {
//没有打印这句,if里面的语句不会执行,可能是多线程访问isPrint造成的
System.out.println("运行中。。。"+isPrint);
if (isPrint) {// 将缓存在队列中的消息按顺序发送到各客户端,并从队列中清除。
String message = (String) message_list.getFirst();
for (ServerThread thread : thread_list) {
thread.sendMessage(message);
}
message_list.removeFirst();
isPrint = message_list.size() > 0 ? true : false;
}
}
}
}
/**
* 服务器线程类
*/
@SuppressWarnings("unchecked")
class ServerThread extends Thread {
private Socket client;
private PrintWriter out;
private BufferedReader in;
private String name;
public ServerThread(Socket s) throws IOException {
client = s;
out = new PrintWriter(client.getOutputStream(), true);
in = new BufferedReader(new InputStreamReader(client.getInputStream()));
//in.readLine();
out.println("成功连上聊天室,请输入你的名字:");
start();
}
@Override
public void run() {
try {
int flag = 0;
String line = in.readLine();
while (true) {
// 查看在线用户列表
if ("showuser".equals(line)) {
out.println(this.listOnlineUsers());
}
if("bye".equals(line)){
out.println("bye");
break;}
// 第一次进入,保存名字
if (flag++ == 0) {
name = line;
user_list.add(name);
thread_list.add(this);
out.println(name + "你好,可以开始聊天了...");
this.pushMessage("Client<" + name + ">进入聊天室...");
} else {
this.pushMessage("Client<" + name + "> say : " + line);
}
line = in.readLine();
}
out.println("byeClient");
} catch (Exception e) {
e.printStackTrace();
} finally {// 用户退出聊天室
try {
client.close();
} catch (IOException e) {
e.printStackTrace();
}
thread_list.remove(this);
user_list.remove(name);
pushMessage("Client<" + name + ">退出了聊天室");
}
}
// 放入消息队列末尾,准备发送给客户端
private void pushMessage(String msg) {
message_list.addLast(msg);
isPrint = true;
}
// 向客户端发送一条消息
private void sendMessage(String msg) {
out.println(msg);
}
// 统计在线用户列表
private String listOnlineUsers() {
String s = "--- 在线用户列表 ---\015\012";
for (int i = 0; i < user_list.size(); i++) {
s += "[" + user_list.get(i) + "]\015\012";
}
s += "--------------------";
return s;
}
}
public static void main(String[] args) throws IOException {
new Server();// 启动服务端
}
}
这里好像出现了多线程问题
@Override
public void run() {
while (true) {
//没有打印下面这句,if里面的语句不会执行,可能是这个线程一直访问isPrint,改变它的线程不能访问到它造成的
System.out.println("运行中。。。"+isPrint);
if (isPrint) {// 将缓存在队列中的消息按顺序发送到各客户端,并从队列中清除。
String message = (String) message_list.getFirst();
for (ServerThread thread : thread_list) {
thread.sendMessage(message);
}
message_list.removeFirst();
isPrint = message_list.size() > 0 ? true : false;
}
}
}