一开始使用swing的时候就了解过这个东西.在项目中应用的时候.发现edt操作不能在SwingWorker线程中执行.(这里的edt操作包括什么呢?我以前做的时候只要有new这个关键字.就执行不了..现在就可以了)
一直以为是swing的规范.有很多的限制.最后就没再看这个东西.直到有一天看到这篇文章
http://vearn.iteye.com/blog/344591
首先赞一下.作者写的非常详细.而且提供了demo.于是我下载了demo,开始读代码.最后终于搞定了这个东东.
于是自己做了个例子.首先是课前准备
这里就简单提下.关于更详细的介绍请看原文
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.