0

所以我JProgressBar的设置不能按我想要的方式工作。因此,每当我运行程序时,它都会立即从 0 变为 100。我尝试使用 a ProgressMonitor,一个 Task,并尝试了 aSwingWorker但我尝试过的没有任何效果。

这是我的程序:

int max = 10;
for (int i = 0; i <= max; i++) {
        final int progress = (int)Math.round(
            100.0 * ((double)i / (double)max)
        );

        EventQueue.invokeLater(new Runnable() {
            @Override
            public void run() {
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException ex) {
                    Logger.getLogger(BandListGenerator.class.getName()).log(Level.SEVERE, null, ex);
                }
                jProgressBar2.setValue(progress);
            }
        });
    }

@MadProgrammer 这是我尝试制作一个摇摆工人并将每个名称写入文档并更新进度条的尝试。该程序达到 86% 左右并停止,从未创建完成的文档。该程序创建一个空白文档。这里有两个方法首先是我制作的 SwingWorker 对象:

public class GreenWorker extends SwingWorker<Object, Object> {
    @Override
    protected Object doInBackground() throws Exception {
        int max = greenList.size();
        XWPFParagraph tmpParagraph;
        XWPFRun tmpRun;
        FileInputStream file = 
                new FileInputStream(location + "GreenBandList.docx");
        gDocument = new XWPFDocument(OPCPackage.open(file));
        for (int i = 0; i < max; i++) {
            tmpParagraph = gDocument.getParagraphs().get(0);
            tmpRun = tmpParagraph.createRun();
            if (greenList.get(i).length() == 1) {
                    tmpRun.setBold(true);
                    tmpRun.setText(greenList.get(i));
                    tmpRun.setBold(false);
            } else {
                    tmpRun.setText(greenList.get(i));//Write the names to the Word Doc
            }
            int progress = Math.round(((float) i / max) * 100f);
            setProgress(progress);
        }
        return null;
  }        
}

这是启动它并具有我的属性更改事件的按钮的代码。

private void GenerateGreenList() throws IOException, InterruptedException {
    //Need to fix the bug that removes the Letter Header in Yellow Band list
    //********************************************************************\\

    //Delete the old list and make a new one
    File templateFile = new File(location + "\\backup\\GreenTemplate.docx");
    FileUtils.deleteQuietly(new File(location + "GreenBandList.docx"));
    FileUtils.copyFile(templateFile, new File(location + 
            "GreenBandList.docx"));
    //Get the New Entries
    String[] entries = jTextPane3.getText().split("\n");
    for (String s : entries) {
        if (s != null) {
            greenList.add(s);
        }
    }
    //Resort the list
    Collections.sort(greenList);
    //Write the names to the document
    GreenWorker worker = new GreenWorker();
    worker.addPropertyChangeListener(new PropertyChangeListener() {
        @Override
        public void propertyChange(PropertyChangeEvent evt) {
            if ("progress".equals(evt.getPropertyName())) {
                jProgressBar2.setValue((Integer) evt.getNewValue());
            }
        }
    });
    worker.execute();
    if (worker.isDone()) {
        try {
            gDocument.write(new FileOutputStream(new File(location + "GreenBandList.docx")));
            ////////////////////////////////////////////////////////////
        } catch (IOException ex) {
            Logger.getLogger(BandListGenerator.class.getName()).log(Level.SEVERE, null, ex);
        }
        JOptionPane.showMessageDialog(null, "Green Band List Created!");
        jProgressBar2.setValue(0);
    }
}

我使用了您其他帖子中的属性更改侦听器,但我真的不明白您写的那个是做什么的或它一般做什么?

4

2 回答 2

5

Swing 是一个单线程环境,即有一个线程负责处理系统内发生的所有事件,包括重绘事件。如果有任何东西因为任何原因阻塞了这个线程,它将阻止 Swing 处理任何新事件,包括重绘事件......

所以这一切...

EventQueue.invokeLater(new Runnable() {
    @Override
    public void run() {
        try {
            Thread.sleep(1000);
        } catch (InterruptedException ex) {
            Logger.getLogger(BandListGenerator.class.getName()).log(Level.SEVERE, null, ex);        }
        jProgressBar2.setValue(progress);
    }
});

不断暂停事件调度线程,阻止它实际进行任何更新(或至少随机间隔它们)......

您的外部循环也可能是从 EDT 的上下文中运行的,这意味着在它存在之前,事件队列中的任何内容都不会被处理。您所有的重绘请求将合并为一个单独的绘画请求,瞧,即时填充进度条......

你真的应该使用SwingWorker- 我知道你说过你尝试过一个,但你没有在这方面显示任何代码,所以很难知道为什么它不起作用,但是......

如果我们之前没有说过几次,请原谅我:P

于 2014-02-26T03:31:53.963 回答
1

你在Thread.sleep里面唤起,EvokeLater这意味着它在你的 for 循环之外的另一个线程上运行。即,您的 for 循环正在瞬间完成(嗯,无论从 1 循环到 100 需要多长时间,这几乎是瞬间完成的)。

移到Thread.sleep外面,EvokeLater它应该可以按您的意愿工作。

    int max = 10;
    for (int i = 0; i <= max; i++) {
        final int progress = (int)Math.round(
            100.0 * ((double)i / (double)max)
        );

        EventQueue.invokeLater(new Runnable() {
            @Override
            public void run() {
                jProgressBar2.setValue(progress);
            }
        });
        try {
            Thread.sleep(1000);
        } catch (InterruptedException ex) {
            Logger.getLogger(BandListGenerator.class.getName()).log(Level.SEVERE, null, ex);
        }
    }

编辑:同意@MadProgrammer。看起来这只是一个说明性问题,但您应该确保在这里尝试完成的任何事情都使用SwingWorkerfor。

于 2014-02-26T03:31:20.240 回答