1

在下面的代码中,我想在 JEditorPane 上写入数字 0 到 10000。但是,JEditorPane 不显示任何内容,除非它完全完成(一次打印所有 0 到 10000)或者当操作系统给它时间刷新和显示内容。无论哪种情况,应用程序都会在一段时间内无响应,用户认为应用程序已崩溃。无论如何我们可以强制 jEditorPane 显示更新的内容,即使它仍然很忙?我尝试了invokeAndWait,但在这里没有帮助,因为我们无法从事件调度程序线程调用invokeAndWait。

换句话说,如果我将 替换为Thread.sleep(0)Thread.sleep(50)它可以正常工作并在结果发生时显示打印结果,但是添加 50 毫秒延迟会使它变得非常慢。我想要的只是不时更新 JEditorPane,这样用户就不会认为应用程序崩溃了。我更喜欢只修改 updateMessages()。 这是我的代码:

import javax.swing.JEditorPane;
import javax.swing.JFrame;
import javax.swing.JScrollPane;
import javax.swing.SwingUtilities;
import javax.swing.text.Document;
import javax.swing.text.html.HTMLEditorKit;
import javax.swing.text.html.StyleSheet;

public class CMessageWindowStack {

    private static final String MESSAGE = "msg";
    private  JScrollPane scrollPane;
    public  JEditorPane  editorPane;
    private  HTMLEditorKit kit;
    private  String msgBuffer=new String("");
    private static CMessageWindowStack window=null;
    private static JFrame frameContainer=null;

    private CMessageWindowStack()
    {
        editorPane  = new JEditorPane ();
        editorPane.setEditable(false);
        editorPane.setContentType("text/html");
        kit = new HTMLEditorKit();
        editorPane.setEditorKit(kit);

        StyleSheet styleSheet = kit.getStyleSheet();
        styleSheet.addRule("."+MESSAGE+" {font: 10px monaco; color: black; }");

        Document doc = kit.createDefaultDocument();
        editorPane.setDocument(doc);
        scrollPane = new JScrollPane(editorPane);
    }
    public static CMessageWindowStack getInstance(){
        if (null==window)
        {window=new CMessageWindowStack();}
        return window;
    }
/**
 * The core
 * @param sMessage
 * @param sType
 */
    private void updateMessages(final String sMessage, final String sType)

    {
        SwingUtilities.invokeLater( new Runnable() {
             public void run() {

        String sMessageHTML=""; 
        String sTypeText="";
        if (!sMessage.equals("\r\n")){ 
            sTypeText = sType+": ";
        }

        sMessageHTML = sMessage.replaceAll("(\r\n|\n)", "<br/>");
        if (!sMessageHTML.equals("<br/>")) 
        {
            sMessageHTML =   "<SPAN CLASS="+sType+">"+ sTypeText+sMessageHTML + "</SPAN>";
        }

        msgBuffer=msgBuffer.concat( sMessageHTML);
        editorPane.setText(msgBuffer);
        if ((editorPane.getDocument()).getLength()>1){
            editorPane.setCaretPosition((editorPane.getDocument()).getLength()-1);
        }  
             }
          });
    }

    public void setContainerFrame(JFrame jFrame){
        frameContainer = jFrame;
        javax.swing.GroupLayout layout = new javax.swing.GroupLayout(frameContainer.getContentPane());
        frameContainer.getContentPane().setLayout(layout);
        layout.setHorizontalGroup(
                layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
                .addComponent(scrollPane)
                );
        layout.setVerticalGroup(
                layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
                .addGroup(layout.createSequentialGroup()
                        .addComponent(scrollPane, javax.swing.GroupLayout.DEFAULT_SIZE, 217, Short.MAX_VALUE))
                );
    }

    public void setVisible(boolean bVisible){
        editorPane.setVisible(bVisible);
        scrollPane.setVisible(bVisible);
    }

    public void printMsg(String sMessage){
        String sType = MESSAGE;
        updateMessages(sMessage,sType);
    }

    public void printlnMsg(String sMessage){
        sMessage=sMessage.concat("\r\n");
        printMsg(sMessage);
    }


    public static void main(String args[]){
        CMessageWindowStack m_LogMgr;
        JFrame frame = new JFrame();
        m_LogMgr=CMessageWindowStack.getInstance();
        m_LogMgr.setContainerFrame(frame);
        frame.setVisible(true);
        frame.setSize(500, 500);


        for(int i=0;i<10000;++i){
            try {
                Thread.sleep(0);//becomes unresponsive if sleep(0)
            } catch (Exception e) {
            }
            m_LogMgr.printlnMsg(i+"-----------------------");
        }

    }


}
4

3 回答 3

3

您可能需要使用 SwingWorker。首先使用您的基本 HTML 加载 JEditorPane。然后你需要publish每一行数据。发布数据后,您需要将文本插入到编辑器窗格中。

基本思想是您不能一次创建整个 HTML 字符串。

我从来没有真正理解如何使用 JEditorPane 和动态变化的 HTML。也许您可以只使用 JTextPane。它支持不同的字体和颜色,并且更容易动态更新。

于 2013-05-14T15:27:47.287 回答
0

好吧 - 作为一个黑客,你可以尝试这样的事情:

    for(int i=0;i<10000;++i){
        try {
            if( i%100 == 0 ) {
                Thread.sleep(10);//becomes unresponsive if sleep(0)
            }
        } catch (Exception e) {
        }
        m_LogMgr.printlnMsg(i+"-----------------------");
    }
于 2013-05-14T15:28:32.163 回答
0

您可以尝试使用 Thread 并为其赋予最高优先级。此外,您可以在 java.exe 中添加 -XmX2020 选项。

其他解决方案可能是如果您使用 JTextArea 而不是JEdiorPane. 似乎您对 HTML 所做的一切都是样式化字体。您可以使用java.awt.FontJTextArea 上的属性来执行此操作。

于 2013-05-14T15:31:29.530 回答