gpt4 book ai didi

java - Java中的服务器客户端聊天室

转载 作者:行者123 更新时间:2023-11-30 11:24:10 25 4
gpt4 key购买 nike

我正在制作一个聊天室项目,其中服务器接受许多客户端,并且客户端写入的任何内容都会到达其他客户端等等。不幸的是,服务器最多接受 2 个客户端,并且在一个客户端写入输入后它会出错。

public class Server2 {

private static ArrayList<Socket> clients;
private ServerSocket server;
DataOutputStream os;
DataInputStream in;
Socket s;

public Server2() throws IOException {
server = new ServerSocket(5000);
clients = new ArrayList<Socket>();
System.out.println("Waiting for connections...");
runOutput();
}

public void addClient() throws IOException {
s = server.accept();
clients.add(s);
System.out.println("A new Client has joined");
}

public void runOutput() {

Thread n = new Thread(new Runnable() {
public void run() {
try {
addClient();
} catch (IOException e) {
e.printStackTrace();
}
}
});
n.start();

Thread input = new Thread(new Runnable() {
public void run() {
try {
addClient();
in = new DataInputStream(s.getInputStream());
os = new DataOutputStream(s.getOutputStream());
String st = in.readLine();
System.out.println(st);
os.writeBytes(st);
for(int i = 0; i < clients.size(); i++)
{
DataOutputStream oo = new DataOutputStream(clients.get(i).getOutputStream());
oo.writeBytes(st);
}
} catch (IOException e) {
e.printStackTrace();
}
}
});
input.start();
}

public static void main(String[] args) throws IOException {
Server2 server = new Server2();
}
}

和客户端类:

public class Client2 {
final public static String host = "localhost";
final public static int port = 5000;
Socket socket;
DataInputStream in;
DataOutputStream ou;
Scanner chat;
boolean run;
String name;

public Client2(String n) throws IOException {
name = n ;
socket = new Socket(host , port);
System.out.println("Connection Successful");
run = true;
runOutput();
}

public void runOutput() {
Thread input = new Thread(new Runnable() {
public void run() {
while (run) {
try {
in = new DataInputStream(socket.getInputStream());
String s = in.readLine();
System.out.println(s);
if(chat.nextLine().compareTo("QUIT") == 0)
run = false;
} catch (IOException e) {
e.printStackTrace();
}
}
}
});
input.start();

Thread t = new Thread(new Runnable() {
public void run() {
while (run) {
try {
ou = new DataOutputStream(socket.getOutputStream());
chat = new Scanner(System.in);
ou.writeBytes(name + " says :" + chat.nextLine() + "\n");

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

t.start();
}

public static void main(String[] args) throws IOException {
Client2 client = new Client2("Ahmed");
}
}

最佳答案

您的“n”线程没有循环,这意味着它会运行一次(接受连接),然后结束。

Thread n = new Thread(new Runnable() {
public void run() {
while(true) { //control this with your own boolean, or it will run forever
try {
addClient();
}catch(IOException e) { }
}
}
});

别担心,您的线程将在“ss.accept()”处暂停,因为它会等到套接字被接受后才能继续。

Thread thread = new Thread(new Runnable() {
public void run() {
while(running) {
String input;

while((input = inputstream.readUTF()) != null) {
//handle input
}
}
}
});

抱歉,写得太快了,没有太多时间。如果它没有帮助,我会回来并将它与您的代码联系起来。

您需要某种方式从服务器的输入流中持续检索输入,然后您可以在其中处理输入。

每次您从 inputstream.readUTF() 检索内容时,该循环都会循环(当然,它不是空的)。希望这个例子对你有帮助


在实际阅读您的代码并对其进行测试后,我注意到您的结构似乎有点不对劲。首先,服务器中的 Thread n 负责接受连接 (addClient()),但您也首先在 Thread input 中调用它>?

Thread n 正在处理接受连接,因此对于循环,Thread n 没问题。Thread input 正在处理从客户端检索到的输入。这就是您感到困惑的地方。

每个客户端都应该有自己的 InputStream 和 OutputStream 服务器端。类似于您拥有套接字列表的方式(因为您将在服务器端创建多个套接字),您也需要多个流。为此,我建议创建一个 User 类,然后如果有的话,为 User 使用 ArrayList,而不是 Socket。

public class User extends Thread { //create a Thread for each connection to handle data seperately
DataInputStream in;
DataInputStream out;

Socket socket;
public User(Socket socket) { //creates a stream for the socket that the user connected with
this.socket = socket;

initStream();
}

public void initStream() { //inits streams after socket is initiated (through ss.accept())
try {
out = new DataOutputStream(socket.getOutputStream());
in = new DataInputStream(socket.getInputStream());
}catch(IOException e) { e.printStackTrace(); }
}

public void closeStream() throws IOException { //throws exception since i have a chance to catch it in my run() method
out.close(); in.close();
socket.close();
}

public void sendMessage(String string) {
try {
out.writeUTF(string);
}catch(IOException e) { e.printStackTrace(); }
}

public void run() { //when thread starts, accept/handle data
String input;

try {
while((input = in.readUTF()) != null) {
//handle input
}
closeStream();
}catch(IOException e) { e.printStackTrace(); }
}
}

in.readLine() 已贬值,这意味着它已过时且不应使用(以避免错误)。现在您已经设置了处理数据的用户类,现在您可以在每次收到连接时创建一个新用户。

public class Server {
private static boolean running = true;

public static ArrayList<User> userlist;

public static synchronized void stop() { //synchronized in-case another thread other than the main thread stops the server
if(!running) return;
running = false;
}

public static void main(String[] args) {
/* uses the JVM thread (main thread for applications), so if you dont wanna
** dedicate your entire server to accepting connections only (since Users are in new thread)
** create a new thread for the code in this void method. */

try {
ServerSocket ss = new ServerSocket(8000);

userlist = new ArrayList<User>();
User user; //holds the user so we can put it in the list, then start
while(running) {
user = new User(ss.accept());

list.add(user);
user.start();
}
} catch(IOException e) {
e.printStackTrace(); }
}
}

希望这能让您更好地了解服务器的运行方式。

至于你的客户..好吧,你应该意识到很多事情:

首先,大问题,您在每个循环中重新初始化 DataOutputStream,这将导致流服务器端关闭。必须将其从循环中取出,可能将其放在之前(或者查看我的服务器示例以获得处理流初始化的更好想法)。

其次,名称应该保存在服务器上,而不是客户端上。这样,很容易遍历名称,检查它是否被占用,等等。

第三,与网络无关,但看到你如何从不引用 Thread nThread input,除了启动它,你应该删除引用变量。 (尽管我很确定本地变量已被垃圾收集器收集,但看起来仍然更好)。示例:

public void method() {
new Thread(new Runnable() {
public void run() {
//code
}
}).start();

new Thread(new Runnable() {
//run method
}).start();
}

希望这有助于您了解多线程服务器的工作原理! :)

PS:这是一个非常基本的服务器,真的不应该用于大型应用程序。 创建新线程 (new User()) 会导致大量开销,因此最好使用 ExecutorService 或其他类型的线程池服务来处理连接。祝你好运!

关于java - Java中的服务器客户端聊天室,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20807941/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com