- 浏览: 43829 次
- 性别:
- 来自: 北京
文章分类
最新评论
-
jiangduxi:
您好,请教下这个开源中间件是否适用 IM。尤其是移动通讯中的弱 ...
metaq源码分析(二) -
尔今尔后:
写的很好,简明扼要的概述。谢谢
Java-String类型的参数传递问题
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返回后事件才能派发。
ex)actionPerformed();返回的时候事件派发线程才能派发线程,而在actionPerformed中使用invokeAndWait则会导致actionPerformed不能返回。所以也就无法派发invokeAndWait中的线程。
由于Swing是线程不安全的,所以,从事件派发线程之外的线程访问Swing组件是不安全的,SwingUtilities类提供这两种方法用于执行事件派发线程中的代码
在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返回后事件才能派发。
ex)actionPerformed();返回的时候事件派发线程才能派发线程,而在actionPerformed中使用invokeAndWait则会导致actionPerformed不能返回。所以也就无法派发invokeAndWait中的线程。
由于Swing是线程不安全的,所以,从事件派发线程之外的线程访问Swing组件是不安全的,SwingUtilities类提供这两种方法用于执行事件派发线程中的代码
发表评论
-
httpclient应用
2015-05-14 17:50 527httpclient应用: 准 ... -
Spring AOP+ehCache 页面、对象缓存
2014-11-10 14:11 472使用aop来实现一个Cache简单的解决方案,具体需求如下 ... -
并发程序设计详解
2014-11-05 17:57 682Java性能优化系列之三--并发程序设计详解 来源: ... -
并发程序设计-java.util.concurrent介绍
2014-11-05 17:48 701并发程序设计-java.util ... -
java 并发
2014-11-05 14:07 412文章来源:http://dapple.iteye.com/b ... -
Xms Xmx PermSize MaxPermSize 区别
2014-11-04 11:43 555来源: http://www.cnblogs.com/mi ... -
java nio 简介
2014-11-01 16:06 5411. 基本 概念 IO 是主 ... -
Java-String类型的参数传递问题
2014-10-20 16:00 715刚才看见一个兄弟在为Java的String传值/传引用问题困 ... -
encode,decode
2014-07-15 11:41 0public class Decode{ public S ... -
JMS实现简单聊天程序
2014-06-25 17:58 1101使用了JMS的开源实现activemq消息中间件。各版本都 ... -
设计模式:享元模式
2014-06-05 00:50 745十. 享元模式 整理自 《java与模式》阎宏编著 1 ... -
设计模式:代理模式
2014-06-04 17:35 478设计模式 代理模式:给某一对象提供代理对象,并由代理对象 ... -
设计模式:单例模式singleton示例分析
2014-06-03 13:16 589public class singleton { pr ... -
Swing Worker应用举例
2014-05-29 12:53 617在开发Java Swing应用程序的过程中,有两个原则是必须 ... -
浅析SwingWorker
2014-05-29 02:05 748正确理解和 ... -
SwingWorker分析
2014-05-29 00:52 647SwingWorker可以帮助我们在后台执行耗时的任务,而避 ... -
SwingUtilities中invokeLater
2014-05-29 00:31 1584在Java中Swing是线程不安全的,是单线程的设 ... -
POI与JXL 操作excel比较
2014-05-06 10:36 967POI是一个标准的Apache项目,提供了完整的接口,不过也 ... -
JXL操作EXCEL
2014-05-06 10:10 498public ImpExpRecord upLoadExc ... -
POI导出EXCEL
2014-05-06 09:49 560import java.sql.ResultSet;impo ...
相关推荐
除了`invokeLater`,Swing还提供了`invokeAndWait`方法,它会阻塞当前线程直到指定的任务完成。但是,通常推荐使用`invokeLater`,因为它非阻塞性质允许应用程序保持响应性,特别是在处理耗时操作时。 总的来说,...
为了确保Swing组件的操作是在EDT中执行,Swing提供了`SwingUtilities.invokeLater(Runnable)`和`SwingUtilities.invokeAndWait(Runnable)`两个方法来帮助开发者解决线程安全问题。 - **SwingUtilities.invokeLater...
如果需要在非EDT线程中修改Swing组件,必须使用SwingUtilities的invokeLater或invokeAndWait方法。 SwingWorker是JDK 1.6引入的一个类,用于处理耗时任务。在doInBackground方法中执行计算密集型任务,然后通过...
SwingUtilities.invokeLater和SwingUtilities.invokeAndWait等方法提供了一种安全的机制,可以在非EDT线程中安全地执行Swing组件的操作。 ### Lambda表达式与回调接口 Lambda表达式是一种允许将代码块作为方法参数...
在这个例子中,`SwingUtilities.invokeLater()`被用来在EDT中创建和显示一个`JFrame`。如果不在EDT中执行这些操作,可能会导致界面无法正确显示或者出现其他未定义的行为。 总的来说,理解和正确使用`EventQueue`...
- 使用`SwingUtilities.invokeLater`或`invokeAndWait`:这两个方法将任务添加到EDT的队列中,确保任务在线程安全的环境中执行。 - 创建独立的线程处理长时间任务,然后通过`invokeLater`或`invokeAndWait`在EDT上...
因此,开发者需要了解和使用SwingUtilities的invokeLater和invokeAndWait方法。 通过"ch9"、"ch7"、"ch6"、"ch8"这些章节,我们可以依次深入学习Swing的高级特性、事件处理机制、组件的使用以及布局管理等内容,...
- 使用`SwingUtilities.invokeLater`或`invokeAndWait`方法将任务添加到EDT队列,确保在UI线程中安全地执行代码。 当事件发生时,EDT按照以下流程处理: 1. 事件调度线程从事件队列中取出事件。 2. 调用事件源组件...
因此,你需要使用SwingUtilities的invokeLater或invokeAndWait方法来确保UI操作的正确性。 7. **国际化和本地化**:Swing支持多语言环境,开发者可以为不同的地区提供相应的资源文件,实现应用程序的国际化和本地化...
为了确保UI更新操作能够在正确的线程中执行,Swing提供了一些方法,例如`SwingUtilities.invokeLater()`和`SwingUtilities.invokeAndWait()`。 2. **正确使用方法**: - 使用`SwingUtilities.invokeLater()`来异步...
因此,你需要使用SwingUtilities的invokeLater或invokeAndWait方法来确保UI更新的正确性。 通过深入学习和实践Java Swing,你可以创建出高效、美观的桌面应用程序。本教程中的实例代码将帮助你更好地掌握Swing的...
使用SwingUtilities的invokeLater或invokeAndWait方法可以确保操作在EDT上执行。 SwingWorker: SwingWorker类是处理耗时任务的工具,它允许在后台线程执行计算,同时在主线程更新UI。这样可以避免UI冻结,提升用户...
因此,涉及UI更新的操作必须在EDT中执行,可以通过SwingUtilities的invokeLater或invokeAndWait方法实现。 10. **可扩展性**:一个完善的记事本程序可能还包含其他功能,如查找替换、撤销重做、复制粘贴等。这些...
当收到新消息时,通过Swing的异步方法(如SwingUtilities.invokeLater或invokeAndWait)在EDT中更新JTextArea的内容。 5. **错误处理**:良好的错误处理是任何程序的关键部分。需要捕获和处理可能的异常,例如网络...
因此,通常我们会使用SwingUtilities的invokeLater或invokeAndWait方法来确保所有对GUI的操作都在EDT中执行。 最后,第9章的第二部分可能涵盖了更高级的话题,比如对话框(JOptionPane)、拖放功能(Drag and Drop...
可以使用SwingUtilities的invokeLater和invokeAndWait方法来保证线程安全。 10. **国际化和本地化**:Java GUI支持多语言环境,你可以为不同的区域和语言创建资源文件,使程序能够适应不同的文化背景。 通过学习...
可以使用 SwingUtilities.invokeLater 或 invokeAndWait 方法确保代码在正确的线程中执行。 10. **国际化(Internationalization)**:Swing 支持多语言应用,允许开发者轻松地为不同地区提供本地化版本。 深入...
因此,更新界面状态时,通常会使用SwingUtilities的invokeLater或invokeAndWait方法。 总结起来,"一个简单Swing窗体"项目展示了如何使用Java Swing创建一个具有交互功能的窗口应用,包括创建JFrame、添加组件、...
- **SwingUtilities.invokeLater和SwingUtilities.invokeAndWait**:这两个方法用于确保Swing组件只能在EDT中更新。`invokeLater`会在EDT的末尾执行指定的任务,而`invokeAndWait`则会阻塞当前线程,直到指定的任务...
为了确保界面的响应性和避免竞态条件,开发者需要使用SwingUtilities的invokeLater或invokeAndWait方法。 8. **Swing的高级特性**: Swing还提供了对话框(JOptionPane)、拖放支持、国际化和本地化、以及对...