Java客户端/服务器线程快速通信消息时出现空指针异常

标签 java multithreading sockets javafx

我正在使用 JavaFX 创建一个简单的消息客户端。有问题的类是:

MainMenuController.class:此类处理 JavaFX ui 逻辑,例如将聊天消息打印到文本区域

Server.class:控制服务器套接字和输入/输出逻辑

Client.class:控制客户端套接字和输入/输出逻辑

我能够在本地主机上完美地运行应用程序,除了当我尝试太快发送消息时我的程序抛出空指针异常。我将跳过相关代码,然后解释我尝试解决此错误的方法。

MainMenuController.java(异常跟踪到此函数)

public static void printMessage(String sender, String msg) {
    //msgHistory is a textArea
    msgHistory.appendText(sender + ": " +msg + "\n");
}

服务器.java

try {
    servSocket = new ServerSocket(serverPort);

    MainMenuController.printMessage("Server running on port: "
        + serverPort + ". . .");

    // block till we accept connection request
    clientSocket = servSocket.accept();

    out = new PrintWriter(clientSocket.getOutputStream(), true);
    in = new BufferedReader(new InputStreamReader(
        clientSocket.getInputStream()));

    // receive and set friend's alias
    friendName = in.readLine();
    aliasSet = true;

    String input;

    // while connection is present, poll server socket input for new
    // messages
    while (true) {
        Thread.sleep(100);
        input = in.readLine();

        if (input == null) {
            break;
        }

        //update status 
        MainMenuController.printMessage(friendName, input);  //Line traced to exception
    }
}

客户端.java

try {   
    socket = new Socket (serverAddr, portNum);

    MainMenuController.printMessage("Connected!");

    out = new PrintWriter(socket.getOutputStream(), true);
    in = new BufferedReader(new InputStreamReader(socket.getInputStream()));

    //send server our alias
    out.println(myName);

} catch (IOException e) {
    System.err.println("Accept failed.");
}

我认为该异常是由客户端和服务器尝试同时写入 MainMenuController 中的 textArea 引起的。我尝试在服务器打印到 textArea 之前添加 thread.sleep(x) 。我还在 MainMenuController 类中实现了公共(public)锁。在服务器和客户端中对 printMessage 的每次调用中,我都会等到锁释放后再调用 printMessage 方法(我在调用 printMessage 之前设置锁,然后在方法返回后释放它)。我把这段代码拿出来以使其更具可读性。这些解决方案都不起作用,我不知道该去哪里寻找。

堆栈跟踪:

Exception in thread "Thread-4" java.lang.NullPointerException
    at com.sun.javafx.sg.prism.NGTextHelper$TextAttributes.computeLinePadding(NGTextHelper.java:405)
    at com.sun.javafx.sg.prism.NGTextHelper$TextAttributes.access$200(NGTextHelper.java:292)
    at com.sun.javafx.sg.prism.NGTextHelper.buildTextLines(NGTextHelper.java:2357)
    at com.sun.javafx.sg.prism.NGTextHelper.validateText(NGTextHelper.java:1847)
    at com.sun.javafx.sg.prism.NGTextHelper.getCaretShape(NGTextHelper.java:1435)
    at javafx.scene.text.Text.getDecorationShapes(Text.java:1150)
    at javafx.scene.text.Text.impl_geomChanged(Text.java:757)
    at javafx.scene.text.Text$1.invalidated(Text.java:214)
    at javafx.beans.property.StringPropertyBase.markInvalid(StringPropertyBase.java:127)
    at javafx.beans.property.StringPropertyBase.set(StringPropertyBase.java:161)
    at javafx.beans.property.StringPropertyBase.set(StringPropertyBase.java:67)
    at javafx.scene.text.Text.setText(Text.java:188)
    at com.sun.javafx.scene.control.skin.TextAreaSkin$17.invalidated(TextAreaSkin.java:610)
    at com.sun.javafx.binding.ExpressionHelper$Generic.fireValueChangedEvent(ExpressionHelper.java:359)
    at com.sun.javafx.binding.ExpressionHelper.fireValueChangedEvent(ExpressionHelper.java:100)
    at javafx.scene.control.TextInputControl$TextProperty.fireValueChangedEvent(TextInputControl.java:1034)
    at javafx.scene.control.TextInputControl$TextProperty.markInvalid(TextInputControl.java:1038)
    at javafx.scene.control.TextInputControl$TextProperty.invalidate(TextInputControl.java:978)
    at javafx.scene.control.TextInputControl$TextProperty.access$200(TextInputControl.java:950)
    at javafx.scene.control.TextInputControl$1.invalidated(TextInputControl.java:119)
    at com.sun.javafx.binding.ExpressionHelper$SingleInvalidation.fireValueChangedEvent(ExpressionHelper.java:155)
    at com.sun.javafx.binding.ExpressionHelper.fireValueChangedEvent(ExpressionHelper.java:100)
    at javafx.scene.control.TextArea$TextAreaContent.insert(TextArea.java:196)
    at javafx.scene.control.TextInputControl.replaceText(TextInputControl.java:373)
    at javafx.scene.control.TextInputControl.insertText(TextInputControl.java:308)
    at javafx.scene.control.TextInputControl.appendText(TextInputControl.java:298)
    at application.MainMenuController.printMessage(MainMenuController.java:96)
    at application.Server.run(Server.java:79)
    at java.lang.Thread.run(Unknown Source)

编辑:此错误仅发生在 textArea.appendText(string) 中。 textArea.setText(string) 不会引发异常。

最佳答案

您不应该从任意线程调用 JavaFX 方法,而只能从 JavaFX 应用程序线程调用。将您的 msgHistory.appendText 包装在 Platform.runLater() 中打电话。

关于Java客户端/服务器线程快速通信消息时出现空指针异常,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19192574/

相关文章:

Linux 每个进程的资源限制——一个深奥的 Red Hat 之谜

java - 当高吞吐量(3GB/s)文件系统可用时,如何在 Java 中使用多线程读取文件

c# - 我在 2 个线程之间使用 ConcurrentQueue 可以吗?

c++ getaddrinfo 不处理大 curl

java - Solr 。如何连接两个单独的查询

java - 如何在android中创建平滑的高度变化动画

java - 如何在java中调整曝光?

java - Eclipse 中的调试器不工作

java - 尝试使用 Socket 测试本地主机端口上的连接并获取 ConnectException

c++ - 有什么方法可以比套接字更快地将数据从 C++ 传输到 Flex/Air?