2

我正在编写代码来计算 Pi 的值,有时可能需要很长时间才能计算出来。我添加了一个进度条来显示进度,但是代码完全按照我的指示执行,它在计算后打开进度条然后立即关闭它(当值达到 100 时它关闭。)我试图粘贴代码对于进度条进入循环,但很快我意识到解决方案,但创建多个进度条窗口。如果放在计算之前,进度条保持在 0(显然)我在下面显示了我的代码:

import java.awt.BorderLayout;
import java.awt.Container;
import javax.swing.BorderFactory;
import javax.swing.JFrame;
import javax.swing.JProgressBar;
import javax.swing.border.Border;
import java.util.Scanner;

public class PicCalc 
{
public static void main(String args[]) 
{
  Scanner keyboard = new Scanner(System.in);
  double i = 0;
    double stage = 1;
  int iterations = 1;
    int x = 3; //Simply a variable to calculate pi
  double y = -1.0; // Another variable for pi
    double myPi; //This is my calculation of Pi, don't eat it. 

  System.out.println("Hello! I calculate Pi and compare it to the acutal value of Pi!");
  System.out.println("How many iterations?");
  iterations = keyboard.nextInt();

  //Calculates Pi
  for (i = 1.0; i <= iterations; i++)
  {
         stage = stage + y/x;
         y = - y; //Flips Sign
         x+=2; 
  }

    myPi = 4*stage;
    System.out.println("My Pi: " + myPi);
    //STOP CALCULATING PI

    //CALCULATE PERCENT COMPLETE
  double percent = 100*(i/iterations);
  int intPercent = (int) (percent + 0.5); //Adds .5 in order to round.
  //STOP CALCULATING PERCENT COMPLETE

  //MAKES LOADING SCREEN
  JFrame f = new JFrame("Loading...");
  f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
  Container content = f.getContentPane();
  JProgressBar progressBar = new JProgressBar();
  progressBar.setValue(intPercent);
  progressBar.setStringPainted(true);
  Border border = BorderFactory.createTitledBorder("Calcualating...");
  progressBar.setBorder(border);
  content.add(progressBar, BorderLayout.NORTH);
  f.setSize(300, 100);
  f.setVisible(true);
  //END OF MAKING LOADING SCREEN

 //CLOSES LOADING SCREEN WHEN 100% IS ACHIEVED
  if (percent >= 100)
  {
     f.dispose();
  }      
  //END OF CLOSING SCREEN HERE
4

3 回答 3

5

你只有一个线程,所以 GUI 永远没有机会更新(因为它正忙于计算)。

通常的方法是将计算放在后台线程中,以便事件调度线程 (EDT) [又名 GUI 线程] 可以自由地更新 GUI。

请参阅这些以获得一些指导,谷歌会出现更多。

你如何使用事件调度线程?

Java后台线程

于 2013-10-25T00:55:54.323 回答
2

编辑:我认为,如果您从我的回答中考虑到逻辑上的变化,并按照约翰的建议将其置于后台进程中,那么您将一切就绪;)

我认为您可能希望将完成百分比的计算移到 for 循环中,以便在您进行迭代时更新进度。

//MAKES LOADING SCREEN
JFrame f = new JFrame("Loading...");
f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
Container content = f.getContentPane();
JProgressBar progressBar = new JProgressBar();
progressBar.setStringPainted(true);
Border border = BorderFactory.createTitledBorder("Calcualating...");
progressBar.setBorder(border);
content.add(progressBar, BorderLayout.NORTH);
f.setSize(300, 100);
f.setVisible(true);

double percent = 0;
int intPercent = 0;
progressBar.setValue(intPercent);
//Calculates Pi
for (i = 1.0; i <= iterations; i++)
{
    stage = stage + y/x;
    y = - y; //Flips Sign
    x+=2; 
    //CALCULATE PERCENT COMPLETE
    percent = 100*(i/iterations);
    intPercent = (int) (percent + 0.5); //Adds .5 in order to round.
    progressBar.setValue(intPercent);
    //STOP CALCULATING PERCENT COMPLETE
}

myPi = 4*stage;
System.out.println("My Pi: " + myPi);
//STOP CALCULATING PI
于 2013-10-25T00:54:24.360 回答
0

你基本上有3个问题:

  1. 百分比计算超出 for 循环
  2. 百分比计算无效
  3. 计算完成后尝试显示表格

用我的评论检查完整的工作代码:

static private double myPi; // This is my calculation of Pi, don't eat it.
public static void main(String[] args)
{
    Scanner keyboard = new Scanner(System.in);

    System.out.println("Hello! I calculate Pi and compare it to the acutal value of Pi!");
    System.out.println("How many iterations?");
    final int iterations = keyboard.nextInt(); // me!!!: i needed to set 2000000000 to see actual window on my computer :)

    // MAKES LOADING SCREEN
    // me!!!: you should create frame before calculations, not after
    final JFrame f = new JFrame("Loading...");
    f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    final Container content = f.getContentPane();
    final JProgressBar progressBar = new JProgressBar();
    progressBar.setStringPainted(true);
    Border border = BorderFactory.createTitledBorder("Calcualating...");
    progressBar.setBorder(border);
    content.add(progressBar, BorderLayout.NORTH);
    f.setSize(300, 100);
    f.setVisible(true);
    // END OF MAKING LOADING SCREEN

    // Calculates Pi
    Thread thread = new Thread() {
        public void run(){
            double stage = 1;
            int x = 3; // Simply a variable to calculate pi
            double y = -1.0; // Another variable for pi

            for (int i = 1; i <= iterations; i++)
            {
                stage = stage + y / x;
                y = -y; // Flips Sign
                x += 2;

                // CALCULATE PERCENT COMPLETE
                // me!!!: you're always getting 0 with your old method here
                double percent = 100.0 * i / iterations;
                int intPercent = (int) (percent + 0.5); // Adds .5 in order to round.
                progressBar.setValue(intPercent);
                // STOP CALCULATING PERCENT COMPLETE
            }
            myPi = 4 * stage;
        }
    };
    thread.start();
    try
    {
        thread.join();
    }
    catch (InterruptedException e)
    {
        e.printStackTrace();
    }

    System.out.println("My Pi: " + myPi);
    // STOP CALCULATING PI

    // CLOSES LOADING SCREEN WHEN 100% IS ACHIEVED
    f.dispose();
    // END OF CLOSING SCREEN HERE
}
于 2013-10-25T01:10:44.110 回答