【问题标题】:In Java,When I use multithreading in TextArea, have a problem [closed]在Java中,当我在TextArea中使用多线程时,出现问题[关闭]
【发布时间】:2020-05-23 17:40:30
【问题描述】:

我在“for”代码块中使用 TextArea

for (String s:List){
                executor.execute(new Runnable() {
                    @Override
                    public void run() {
                        try {
                             textarea.appendText(s);
                        } catch (IOException ignore) {}
                    }
                });
            }

TextArea 不是线程安全的,我读了这个页面page所以我试过了

public void changeArea(TextArea area, String text){
        synchronized(area) {
            area.appendText(text);
        }
    }

我修改了我的“for”块

changeArea(textarea, "sth");

当我运行代码时,java继续给我一个错误

Exception in thread "JavaFX Application Thread" java.lang.NullPointerException
    at com.sun.javafx.text.PrismTextLayout.addTextRun(PrismTextLayout.java:755)
    at com.sun.javafx.text.GlyphLayout.addTextRun(GlyphLayout.java:140)
    at com.sun.javafx.text.GlyphLayout.breakRuns(GlyphLayout.java:210)
    at com.sun.javafx.text.PrismTextLayout.buildRuns(PrismTextLayout.java:770)
    at com.sun.javafx.text.PrismTextLayout.layout(PrismTextLayout.java:1021)
    at com.sun.javafx.text.PrismTextLayout.ensureLayout(PrismTextLayout.java:223)
    at com.sun.javafx.text.PrismTextLayout.getBounds(PrismTextLayout.java:246)
    at javafx.scene.text.Text.getLogicalBounds(Text.java:358)
    at javafx.scene.text.Text.impl_computeGeomBounds(Text.java:1168)
    at javafx.scene.Node.updateGeomBounds(Node.java:3577)
    at javafx.scene.Node.getGeomBounds(Node.java:3530)
    at javafx.scene.Node.getLocalBounds(Node.java:3478)
    at javafx.scene.Node$MiscProperties$2.computeBounds(Node.java:6472)
    at javafx.scene.Node$LazyBoundsProperty.get(Node.java:9306)
    at javafx.scene.Node$LazyBoundsProperty.get(Node.java:9276)
    at javafx.scene.Node.getBoundsInLocal(Node.java:3156)
    at com.sun.javafx.scene.control.skin.TextAreaSkin$ContentView.layoutChildren(TextAreaSkin.java:207)
    at javafx.scene.Parent.layout(Parent.java:1087)
    at javafx.scene.Parent.layout(Parent.java:1093)
    at javafx.scene.Parent.layout(Parent.java:1093)
    at javafx.scene.Parent.layout(Parent.java:1093)
    at javafx.scene.Parent.layout(Parent.java:1093)
    at javafx.scene.Parent.layout(Parent.java:1093)
    at javafx.scene.Scene.doLayoutPass(Scene.java:552)
    at javafx.scene.Scene$ScenePulseListener.pulse(Scene.java:2397)
    at com.sun.javafx.tk.Toolkit.lambda$runPulse$2(Toolkit.java:398)
    at java.security.AccessController.doPrivileged(Native Method)
    at com.sun.javafx.tk.Toolkit.runPulse(Toolkit.java:397)
    at com.sun.javafx.tk.Toolkit.firePulse(Toolkit.java:424)
    at com.sun.javafx.tk.quantum.QuantumToolkit.pulse(QuantumToolkit.java:561)
    at com.sun.javafx.tk.quantum.QuantumToolkit.pulse(QuantumToolkit.java:541)
    at com.sun.javafx.tk.quantum.QuantumToolkit.pulseFromQueue(QuantumToolkit.java:534)
    at com.sun.javafx.tk.quantum.QuantumToolkit.lambda$runToolkit$11(QuantumToolkit.java:340)
    at com.sun.glass.ui.InvokeLaterDispatcher$Future.run(InvokeLaterDispatcher.java:95)
    at com.sun.glass.ui.win.WinApplication._runLoop(Native Method)
    at com.sun.glass.ui.win.WinApplication.lambda$null$4(WinApplication.java:186)
    at java.lang.Thread.run(Thread.java:748)

我尝试了“锁定”并在 changearea 方法中添加“同步” 当我删除 textarea.appendText(s) 并更换 system.out.println(s) ,没关系。我认为这是一个多线程安全问题

当我改变我的代码时

public synchronized void changeArea(TextArea area, String text){
        area.appendText(text);
    }

java指出

Exception in thread "pool-2-thread-1" Exception in thread "pool-2-thread-3" java.lang.IndexOutOfBoundsException: Index: 2, Size: 2
    at java.util.ArrayList.rangeCheck(ArrayList.java:657)
    at java.util.ArrayList.get(ArrayList.java:433)
    at javafx.scene.control.TextArea$TextAreaContent.get(TextArea.java:119)
    at javafx.scene.control.TextArea$TextAreaContent.get(TextArea.java:300)
    at javafx.scene.control.TextArea$TextAreaContent.get(TextArea.java:77)
    at javafx.scene.control.TextInputControl$TextProperty.get(TextInputControl.java:1316)
    at javafx.scene.control.TextInputControl$TextProperty.get(TextInputControl.java:1300)
    at javafx.beans.binding.StringExpression.getValueSafe(StringExpression.java:62)
    at com.sun.javafx.scene.control.skin.TextAreaSkin.lambda$new$16(TextAreaSkin.java:586)
    at com.sun.javafx.binding.ExpressionHelper$Generic.fireValueChangedEvent(ExpressionHelper.java:349)
    at com.sun.javafx.binding.ExpressionHelper.fireValueChangedEvent(ExpressionHelper.java:81)
    at javafx.scene.control.TextInputControl$TextProperty.fireValueChangedEvent(TextInputControl.java:1389)
    at javafx.scene.control.TextInputControl$TextProperty.markInvalid(TextInputControl.java:1393)
    at javafx.scene.control.TextInputControl$TextProperty.controlContentHasChanged(TextInputControl.java:1332)
    at javafx.scene.control.TextInputControl$TextProperty.access$1600(TextInputControl.java:1300)
    at javafx.scene.control.TextInputControl.lambda$new$0(TextInputControl.java:139)
    at com.sun.javafx.binding.ExpressionHelper$SingleInvalidation.fireValueChangedEvent(ExpressionHelper.java:137)
    at com.sun.javafx.binding.ExpressionHelper.fireValueChangedEvent(ExpressionHelper.java:81)
    at javafx.scene.control.TextArea$TextAreaContent.insert(TextArea.java:207)
    at javafx.scene.control.TextInputControl.replaceText(TextInputControl.java:1204)
    at javafx.scene.control.TextInputControl.updateContent(TextInputControl.java:556)
    at javafx.scene.control.TextInputControl.replaceText(TextInputControl.java:548)
    at javafx.scene.control.TextInputControl.insertText(TextInputControl.java:473)
    at javafx.scene.control.TextInputControl.appendText(TextInputControl.java:463)
    at sample.AreaUtils.changeArea(AreaUtils.java:11)
    at sample.Scan.hikvisionScan(Scan.java:16)
    at sample.CommonUtils$1.run(CommonUtils.java:35)
    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)
    at java.lang.Thread.run(Thread.java:748)
java.lang.IndexOutOfBoundsException: Index: 2, Size: 2
    at java.util.ArrayList.rangeCheck(ArrayList.java:657)
    at java.util.ArrayList.get(ArrayList.java:433)
    at javafx.scene.control.TextArea$TextAreaContent.get(TextArea.java:119)
    at javafx.scene.control.TextArea$TextAreaContent.get(TextArea.java:300)
    at javafx.scene.control.TextArea$TextAreaContent.get(TextArea.java:77)
    at javafx.scene.control.TextInputControl$TextProperty.get(TextInputControl.java:1316)
    at javafx.scene.control.TextInputControl$TextProperty.get(TextInputControl.java:1300)
    at javafx.beans.binding.StringExpression.getValueSafe(StringExpression.java:62)
    at com.sun.javafx.scene.control.skin.TextAreaSkin.lambda$new$16(TextAreaSkin.java:586)
    at com.sun.javafx.binding.ExpressionHelper$Generic.fireValueChangedEvent(ExpressionHelper.java:349)
    at com.sun.javafx.binding.ExpressionHelper.fireValueChangedEvent(ExpressionHelper.java:81)
    at javafx.scene.control.TextInputControl$TextProperty.fireValueChangedEvent(TextInputControl.java:1389)
    at javafx.scene.control.TextInputControl$TextProperty.markInvalid(TextInputControl.java:1393)
    at javafx.scene.control.TextInputControl$TextProperty.controlContentHasChanged(TextInputControl.java:1332)
    at javafx.scene.control.TextInputControl$TextProperty.access$1600(TextInputControl.java:1300)
    at javafx.scene.control.TextInputControl.lambda$new$0(TextInputControl.java:139)
    at com.sun.javafx.binding.ExpressionHelper$SingleInvalidation.fireValueChangedEvent(ExpressionHelper.java:137)
    at com.sun.javafx.binding.ExpressionHelper.fireValueChangedEvent(ExpressionHelper.java:81)
    at javafx.scene.control.TextArea$TextAreaContent.insert(TextArea.java:207)
    at javafx.scene.control.TextInputControl.replaceText(TextInputControl.java:1204)
    at javafx.scene.control.TextInputControl.updateContent(TextInputControl.java:556)
    at javafx.scene.control.TextInputControl.replaceText(TextInputControl.java:548)
    at javafx.scene.control.TextInputControl.insertText(TextInputControl.java:473)
    at javafx.scene.control.TextInputControl.appendText(TextInputControl.java:463)
    at sample.AreaUtils.changeArea(AreaUtils.java:11)
    at sample.Scan.hikvisionScan(Scan.java:16)
    at sample.CommonUtils$1.run(CommonUtils.java:35)
    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1149)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:624)
    at java.lang.Thread.run(Thread.java:748)

【问题讨论】:

  • 当我删除textarea.appendText(s)并替换system.out.println(s)时,没关系。我认为这是一个多线程安全问题
  • 您不得从 fx 应用程序线程更改节点(也不能更改任何属性)。该规则也不例外。
  • 将 JavaFx 应用程序视为在单线程上运行的应用程序。正如 kleopatra 所写,对 gui 的所有更改都必须在该线程上完成。要获得“我应该做什么”的答案,请发布 minimal reproducible example 或尝试查看一些示例,例如 this one。
  • 将附加代码放在 FX 线程上—Platform.runLater(() -> area.appendText(...))

标签: java multithreading javafx


【解决方案1】:

在 JavaFX 平台线程上运行您的操作,您应该不会再看到此问题。

public void changeArea(TextArea area, String text) {
    Platform.runLater(()-> area.appendText(text));
}

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2022-11-21
    • 2021-06-29
    • 1970-01-01
    • 2016-03-05
    • 1970-01-01
    • 2020-03-15
    • 1970-01-01
    • 2015-08-01
    相关资源
    最近更新 更多