论坛首页 Java企业应用论坛

Swing的SwingWorker学习

浏览 12910 次
精华帖 (1) :: 良好帖 (0) :: 新手帖 (0) :: 隐藏帖 (0)
作者 正文
   发表时间:2009-11-05   最后修改:2009-11-05

    一开始使用swing的时候就了解过这个东西.在项目中应用的时候.发现edt操作不能在SwingWorker线程中执行.(这里的edt操作包括什么呢?我以前做的时候只要有new这个关键字.就执行不了..现在就可以了)

一直以为是swing的规范.有很多的限制.最后就没再看这个东西.直到有一天看到这篇文章

 http://vearn.iteye.com/blog/344591

首先赞一下.作者写的非常详细.而且提供了demo.于是我下载了demo,开始读代码.最后终于搞定了这个东东.

于是自己做了个例子.首先是课前准备

 

 
EDT是啥米呀

在Java中,键盘输入、鼠标点击或者应用程序本身产生的请求会被封装成一个事件,放进一个事件队列中,java.awt.EventQueue对象负责从这个队列中取出事件并派发它们。而EventQueue的派发有一个单独的线程管理,这个线程叫做事件派发线程(Event Dispatch Thread),也就是EDT。此外,Swing的绘制请求也是通过EDT来派发的。

 

 

 

 

 

 

 

 

 

 这里就简单提下.关于更详细的介绍请看原文

http://vearn.iteye.com/blog/344591

或者

API:http://202.201.112.11/jpk/apply/teacher/preface/53/api/javax/swing/SwingWorker.html

http://www.javaresearch.org/article/7787.htm

http://www.javaresearch.org/article/7788.htm

 

这里帖下关键代码

 

    private void button_getDataActionPerformed(java.awt.event.ActionEvent evt) {                                         
        //清除表格数据
        table_data.setModel(new javax.swing.table.DefaultTableModel(
                new Object[][]{},
                new String[]{
                    "id", "parent_id", "node_name"
                }));
        //如果要查看EDT阻塞效果
        if (checkbox_edt.isSelected()) {
            //在不采用SwingWorker的时候下面这句你会发现不起作用,因为这是一个EDT事件.
            //只有处理完了这个事件之后.别的改变组件状态的时间才会被执行.这就是EDT阻塞
            label_busy.setVisible(true);//设置载入label显示
            try {
                Thread.sleep(5000);//睡眠5秒
            } catch (InterruptedException ex) {
                ex.printStackTrace();
            }
            setData();
            label_busy.setVisible(false);//设置载入label消失
        } else {//使用SwingWroker
            table_data.setEnabled(false);//设置表格不可用
            button_getData.setEnabled(false);//设置按钮不可用
            label_busy.setVisible(true);//设置载入label显示
            new SwingWorker<DefaultTableModel, Void>() {

                @Override
                protected DefaultTableModel doInBackground() throws Exception {
                    System.out.println("开始执行后台方法...");
                    JLabel jLabel = new JLabel();//测试在非EDT线程中创建组件.我以前做的总是不可以.现在好了.
                    Thread.sleep(5000);//睡眠5秒.因为效果不太明显
                    return getData();//返回取到的数据
                }

                @Override
                protected void done() {
                    System.out.println("后台方法运行结束");
                    table_data.setEnabled(true);
                    button_getData.setEnabled(true);
                    label_busy.setVisible(false);
                    try {
                        //  获取doInBackground方法的返回值
                        table_data.setModel(get());//用get()方法取到上面取得的值
                    } catch (Exception ex) {
                        ex.printStackTrace();
                    }
                }
            }.execute();
        }
    }               

 

2009-11-5晚上.


我在回家的路上想起来上次出现edt操作不能在SwingWorker线程中执行这个问题.  是由于用了开源皮肤Substance的问题.回家一试.果真如此..

 

把运行的代码

 

    public static void main(String args[]) {
        SwingUtilities.invokeLater(new Runnable() {

            public void run() {
                TestFrame frame = new TestFrame();
                frame.setVisible(true);
            }
        });
    }
 

改为:

 

    public static void main(String args[]) {
        try {
            JFrame.setDefaultLookAndFeelDecorated(true);
            JDialog.setDefaultLookAndFeelDecorated(true);
            UIManager.setLookAndFeel(new SubstanceBusinessBlueSteelLookAndFeel());
        } catch (Exception e) {
            System.out.println("Substance Raven Graphite failed to initialize");
        }
        SwingUtilities.invokeLater(new Runnable() {
            public void run() {
                TestFrame frame = new TestFrame();
                frame.setVisible(true);
            }
        });
    }

 

添加substance.jar包..

 

抛出异常:

 

 

org.jvnet.substance.api.UiThreadingViolationException: Component creation must be done on Event Dispatch Thread
        at org.jvnet.substance.utils.SubstanceCoreUtilities.testComponentCreationThreadingViolation(SubstanceCoreUtilities.java:2368)
        at org.jvnet.substance.SubstanceLabelUI.createUI(SubstanceLabelUI.java:68)
        at sun.reflect.GeneratedMethodAccessor2.invoke(Unknown Source)
        at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
        at java.lang.reflect.Method.invoke(Method.java:597)
        at sun.reflect.misc.Trampoline.invoke(MethodUtil.java:36)
        at sun.reflect.GeneratedMethodAccessor1.invoke(Unknown Source)
        at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
        at java.lang.reflect.Method.invoke(Method.java:597)
        at sun.reflect.misc.MethodUtil.invoke(MethodUtil.java:243)
        at javax.swing.UIDefaults.getUI(UIDefaults.java:751)
        at javax.swing.UIManager.getUI(UIManager.java:1016)
        at javax.swing.JLabel.updateUI(JLabel.java:256)
        at javax.swing.JLabel.<init>(JLabel.java:145)
        at javax.swing.JLabel.<init>(JLabel.java:216)
        at test.TestFrame$2.doInBackground(TestFrame.java:186)
        at test.TestFrame$2.doInBackground(TestFrame.java:181)
        at javax.swing.SwingWorker$1.call(SwingWorker.java:278)
        at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:303)
        at java.util.concurrent.FutureTask.run(FutureTask.java:138)
        at javax.swing.SwingWorker.run(SwingWorker.java:317)
        at java.util.concurrent.ThreadPoolExecutor$Worker.runTask(ThreadPoolExecutor.java:886)
        at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:908)
        at java.lang.Thread.run(Thread.java:619)
 

 

不知道是什么原因....为什么java默认的皮肤可以..开源的就不可以呢...

 

上面的代码是一个按钮点击事件中执行的代码.

详细的例子请见附件

数据库用的是derby.

效果图:

 

 

 

 

  • 描述: 效果图
  • 大小: 7.8 KB
   发表时间:2009-11-05  
不光是swing,凡是牵扯到gui的,例如c#也存在这个问题!!
0 请登录后投票
   发表时间:2009-11-06  
kjj 写道
不光是swing,凡是牵扯到gui的,例如c#也存在这个问题!!


您说的是edt操作不能在非EDT事件中执行的问题吗?那为什么java默认的皮肤可以呢?开源的那个不可以..
0 请登录后投票
   发表时间:2009-11-06  

今天又出现问题.我数据加载和创建组件分开了

我采用swingWorker的方式加载数据.然后创建组件..但是我现在取数据的时间长.创建组件的时间更长..所以还是避免不了EDT堵塞.我该怎么办呢?因为我取到的数据是很多条.我要根据这些数据动态创建大量的组件.所以还是很费时间,创建组件的时候要给组件设置图片..还是慢

不知道有什么解决方案没

 

        new SwingWorker<List<List<Solution>>, Void>() {

            @Override
            protected List<List<Solution>> doInBackground() throws Exception {
                System.out.println("开始执行后台方法...");
                label_busy.setVisible(true);//这里设置一个带加载图片的label显示
                getChoices(setting);//获取数据
                List<List<Solution>> l = getSolutions(choices);
        //在执行完上面的加载数据后.又开始了EDT阻塞!..加载label的加载图片不动了...
                logger.info("数据获取完毕..EDT开始阻塞!");
                return l;
            }

            @Override
            protected void done() {
                System.out.println("后台方法运行结束");
                logger.info("开始卡屏!");
                button_change.setEnabled(true);
                label_busy.setVisible(false);//设置
                try {
                    creatCom(get());//这个方法的执行时间居然比获取数据的时间都长.
//这里的创建组件包括的操作有:首先得到一组选择题的主体.然后查看主题是否包含图片.有相应的属性.有就创建图片.放到一个label中.循环创建.
//创建选择题主体后.循环取他的四个或者多个选项.如果选项有图片..设置之..
                } catch (Exception ex) {

                logger.info("卡屏完毕!");
            }
        }.execute();
 
0 请登录后投票
   发表时间:2009-11-18  
这种问题的根本原因是:Java Swing不是线程安全的,原因有二个,一个是效率因素,另一个是线程安全的UI工具包通常容易创建出倾向于死锁的程序。

通常在Swing中使用线程有两个简单原则:
1、非常耗时的动作要在一个单独的线程中执行,而不要在事件分派线程(EDT)中执行。
2、在任何线程中都不要碰Swing组件,除了时间分派线程。

要在耗时的线程中操作Swing组件,需要借助EventQueue中的两个静态方法向事件分派线程发送一个事件,从而通知事件分派线程执行对于Swing组件的操作。

上述的原则,对于Java Swing多线程应用是非常非常重要的。
0 请登录后投票
   发表时间:2009-11-19  
wave-labs 写道
这种问题的根本原因是:Java Swing不是线程安全的,原因有二个,一个是效率因素,另一个是线程安全的UI工具包通常容易创建出倾向于死锁的程序。

通常在Swing中使用线程有两个简单原则:
1、非常耗时的动作要在一个单独的线程中执行,而不要在事件分派线程(EDT)中执行。
2、在任何线程中都不要碰Swing组件,除了时间分派线程。

要在耗时的线程中操作Swing组件,需要借助EventQueue中的两个静态方法向事件分派线程发送一个事件,从而通知事件分派线程执行对于Swing组件的操作。

上述的原则,对于Java Swing多线程应用是非常非常重要的。


谢谢解答.我现在的疑问是为什么java默认的皮肤可以,开源的那个不可以呢?难道java默认的皮肤绘制组件的时候不碰swing组件?开源的就碰?
0 请登录后投票
   发表时间:2009-12-04  
看下org.jvnet.substance.utils.SubstanceCoreUtilities类下的testComponentCreationThreadingViolation()和testComponentStateChangeThreadingViolation()方法就明白了,它做了一个判断,将这两个方法的内容注释掉就不会有异常信息了。
0 请登录后投票
   发表时间:2009-12-04  
HeDYn 写道
看下org.jvnet.substance.utils.SubstanceCoreUtilities类下的testComponentCreationThreadingViolation()和testComponentStateChangeThreadingViolation()方法就明白了,它做了一个判断,将这两个方法的内容注释掉就不会有异常信息了。


谢谢解答..回头看看
0 请登录后投票
论坛首页 Java企业应用版

跳转论坛:
Global site tag (gtag.js) - Google Analytics