`
Jony.Hwong
  • 浏览: 116735 次
  • 来自: ...
社区版块
存档分类
最新评论

swing事件派发线程

阅读更多
Java Swing GUI多线程之SwingUtilities.invokeLater和invokeAndWait
      在Java中Swing是线程不安全的,是单线程的设计,这样的造成结果就是:只能从事件派发线程访问将要在屏幕上绘制的Swing组件。事件派发线程是调用paint和update等回调方法的线程,它还是事件监听器接口中定义的事件处理方法,例如,ActionListener中的actionPerformed方法在事件派发线程中调用。

       Swing是事件驱动的,所以在回调函数中更新可见的GUI是很自然的事情,比如,有一个按钮被按下,项目列表需要更新时,则通常在与该按钮相关联的事件监听器的actionPerformed方法中来实现该列表的更新,从事件派发线程以外的线程中更新Swing组件是不正常的。

       有时需要从事件派发线程以外的线程中更新Swing组件,例如,在actionPerformed中有很费时的操作,需要很长时间才能返回,按钮激活后需要很长时间才能看到更新的列表,按钮会长时间保持按下的状态只到actionPerformed返回,一般说来耗时的操作不应该在事件处理方法中执行,因为事件处理返回之前,其他事件是不能触发的,界面类似于卡住的状况,所以在独立的线程上执行比较耗时的操作可能更好,这会立即更新用户界面和释放事件派发线程去派发其他的事件。

      SwingUtilities类提供了两个方法:invokeLate和invoteAndWait,它们都使事件派发线程上的可运行对象排队。当可运行对象排在事件派发队列的队首时,就调用其run方法。其效果是允许事件派发线程调用另一个线程中的任意一个代码块。

   只有从事件派发线程才能更新组件。

程序示例:更新组件的错误方法
startButton.addActionListener(new ActionListener())
{
   public void actionPerformed(ActionEvent e)
   {
    GetInfoThread t = new GetInfoThread(Test.this);
    t.start();
    startButton.setEnabled(false);
   }
}

class GetInfoThread extends Thread
{
Test applet;
public GetInfoThread(Test applet)
{
this.applet = applet;
}

public void run()
{
   while (true)
   {
    try
    {
     Thread.sleep(500);
     applet.getProgressBar().setValue(Math.random() * 100);
    }
    catch (InterruptedException e)
    {
     e.printStackTrace();
    }
   }
}
}

错误分析:在actionPerformed中,监听器把按钮的允许状态设置为false,由于是在事件派发线程上调用actionPerformed,所以setEnabled是一个有效的操作,但是在GetInfoThread中设置进度条是一个危险的做法,因为事件派发线程以外的线程更新了进度条,所以运行是不正常的。

   1、invokeLater使用
class GetInfoThread extends Thread
{
Test applet;

Runnable runx;

int value;

public GetInfoThread(final Test applet)
{
   this.applet = applet;
   runx = new Runnable()
   {
     public void run()
     {
     JProgressBar jpb = applet.getProgressBar();
     jpb.setValue(value);
     }
   }
}

   public void run()
   {
     while (true)
     {
        try
        {
         Thread.sleep(500);
         value = (int) (Math.random() * 100);
         System.out.println(value);
         SwingUtilities.invokeLater(runx);
        }
        catch (InterruptedException e)
        {
         e.printStackTrace();
        }
     }
    }
}
  
   2、invokeAndWait
   与invoikeLater一样,invokeAndWait也把可运行对象排入事件派发线程的队列中,invokeLater在把可运行的对象放入队列后就返回,而invokeAndWait一直等待知道已启动了可运行的run方法才返回。如果一个操作在另外一个操作执行之前必须从一个组件获得信息,则invokeAndWait方法是很有用的。
  
   class GetInfoThread extends Thread
   {
   Runnable getValue,setValue;
   int value,currentValue;
   public GetInfoThread(final Test applet)
   {
    getValue=new Runnable()
    {
     public void run()
     {
      JProgressBar pb=applet.getProgressBar();
      currentValue=pb.getValue();
      }
    };
   setValue=new Runnable()
   {
    public void run()
     {
      JProgressBar pb=applet.getProgressBar();
      pb.setValue(value);
     }
}
   }
  
   public void run()
   {
    while(true)
    {
    try
    {
      Thread.currentThead().sleep(500);
      value=(int)(Math.random()*100);
      try
      {
      SwingUtilities.invokeAndWait(getValue);//直到getValue可运行的run方法返回后才返回
      }
      catch(Exception ex)
      {
      
      }
       if(currentValue!=value)
       {
       SwingUtilities.invokeLater(setValue);
       }
     }
      catch(Exception ex)
      {
      }
     }
   }
      invokeLater和invoikeAndWait的一个重要区别:可以从事件派发线程中调用invokeLater,却不能从事件派发线程中调用invokeAndWait,从事件派发线程调用invokeAndWait的问题是:invokeAndWait锁定调用它的线程,直到可运行对象从事件派发线程中派发出去并且该可运行的对象的run方法激活,如果从事件派发线程调用invoikeAndWait,则会发生死锁的状况,因为invokeAndWait正在等待事件派发,但是,由于是从事件派发线程中调用invokeAndWait,所以直到invokeAndWait返回后事件才能派发。
     
      actionPerformed();返回的时候事件派发线程才能派发线程,而在actionPerformed中使用invokeAndWait则会导致actionPerformed不能返回。所以也就无法派发invokeAndWait中的线程。

      由于Swing是线程不安全的,所以,从事件派发线程之外的线程访问Swing组件是不安全的,SwingUtilities类提供这两种方法用于执行事件派发线程中的代码

总结: GUI中多线调用方法应该使用:SwingUtilities.invokeLater和invokeAndWait 而不是普通情况下那样应用.


分享到:
评论

相关推荐

    JAVA 开发 Swing与多线程

    Swing的单线程模型基于"事件派发线程"(Event Dispatch Thread, EDT),所有Swing组件的渲染和事件处理都在这个线程中进行。这是因为GUI组件通常不支持线程安全的操作,多线程环境下可能会导致组件状态的不一致,...

    Swing线程的最后讨论 -- 利用异步模型

    Swing框架遵循单线程规则,即所有UI组件在同一时间只能被一个线程访问,通常这个线程就是事件派发线程。这一规则是为了避免多线程环境下的竞态条件和同步问题,确保组件的线程安全性。然而,这也会限制我们在执行...

    Swing 线程之SwingUtilities invokeLater docx

    Swing 组件是线程不安全的,这意味着只能从事件派发线程(Event Dispatch Thread,简称 EDT)访问将要在屏幕上绘制的 Swing 组件。如果从其他线程访问 Swing 组件,将会导致程序崩溃或出现不可预测的结果。 在上面...

    桌面应用程序架构.pdf

    - **主线程**:调用`main()`函数的线程,也是事件派发线程(Event Dispatch Thread,简称EDT),负责处理Swing组件的所有事件。 - **系统线程**:与窗口系统交互,派发本地事件。 - **事件派发线程**:专门用于派发...

    Java Swing界面的文本编辑器源代码.rar

    一般来说,这个线程是事件派发线程(event-dispatching thread)。 如果需要从事件处理(event-handling)或绘制代码以外的地方访问UI,那么可以使用SwingUtilities类的invokeLater()或invokeAndWait()方法。

    Java Swing 24种皮肤合集以及使用方法

    由于GUI更新必须在事件派发线程(Event Dispatch Thread,EDT)中进行,所以设置L&F前应确保代码运行在正确的线程。可以使用`SwingUtilities.invokeLater()`来确保这一点: ```java SwingUtilities.invokeLater(...

    Java的Swing编程中使用SwingWorker线程模式及顶层容器

    这是因为大部分Swing组件的构造和操作必须在事件派发线程内进行,以保证线程安全。 2. **事件派发线程**: 事件派发线程(EDT,Event Dispatch Thread)是Swing处理用户交互的核心。所有事件处理器(如监听器的回...

    java线程电子书

    由于GUI组件的更新必须在事件派发线程(EDT)中进行,因此,任何可能阻塞EDT的耗时操作都应该在单独的线程中执行。例如,在Swing框架中,可以使用`SwingWorker`类来执行后台任务,同时保持UI的响应性。 #### 七、...

    关于Bootstrap原理1

    在Java Swing中,所有的用户界面更新必须在事件派发线程(Event Dispatch Thread, EDT)中进行,以保证界面的线程安全。`SwingUtilities` 类的 `invokeLater` 和 `invokeAndWait` 方法就是用于在这条线程上执行任务...

    GUI包括JAVA程序实现编程

    4. **线程管理**:在GUI编程中,所有的UI更新必须在事件派发线程(Event Dispatch Thread,EDT)上进行,以确保界面的响应性和一致性。因此,当需要进行耗时操作时,通常会使用SwingWorker或FutureTask等机制来避免...

    PythaGUI:并发问题

    在GUI编程中,由于UI更新通常需要在主线程(也称为事件派发线程)中进行,所以需要特别注意线程间通信,避免阻塞主线程导致界面无响应。 在分析PythaGUI项目时,我们需要查看源代码(假设压缩包中的PythaGUI-master...

    快递信息管理系统源码.zip

    - 使用Java进行前端界面设计,如JFrame或Swing组件,提供友好的用户交互界面,同时通过Java的 JDBC (Java Database Connectivity) API 连接并操作MySQL数据库。 2. **派送管理**: - 派送管理涉及快递的接收、...

Global site tag (gtag.js) - Google Analytics