« 1 23» Pages: ( 1/3 total )
本页主题: eclipse中的线程 打印 | 加为IE收藏 | 收藏主题 | 上一主题 | 下一主题

讲不出再见

状态: 离线
级别: CEC贵宾
精华: 1
发帖: 88
威望: 22 点
财富: 242 元
在线时间:18(小时)
注册时间:2005-11-29
最后登录:2006-10-23

eclipse中的线程

寒假我就上网很少了,真的非常不好意思.eclipse中的线程是开发eclipse插件中一个比较重要的基础,而很多书上都没有说,我最后抽时间写了这篇入门文章,希望对大家有帮助.

我先大致讲一讲GUI程序中的线程.
    虽然各个操作系统之间的线程机制是不一样的,但是大致是相同的.当用户使用GUI程序时,如果点鼠标或按下键盘上的键等时,操作系统会产生对应的GUI事件,它来决定哪个窗口或程序来接受每一个事件并且放到程序的事件队列中.
    任何GUI程序的底层结构就是一个事件循环.程序首先初始化事件循环,并开始循环,这个循环会从事件队列依次接收GUI事件并一一做出相应的反应.程序应该对事件做出快速的反应使程序一直对用户有响应,举个例子,用户点了一下程序里的一个按钮结果程序就没反应了,那么这个程序应该算是一个失败的程序吧.
    如果某个UI事件引发了某个需要长时间的事务,那么应该把它放到一个另外的单独的线程中,这样程序的那个事件循环就能够马上回来响应用户的下一个操作.线程是非常复杂的一个主题,如果处理的不好很容易造成死锁等很糟糕的情况.

    还好,eclipse为我们开发插件提供了一个方便的UI线程包,大大的简化了很多底层复杂的东西.先看看几个简单的概念.
1.SWT UI线程
SWT用的是操作系统直接支持的线程模式,程序会在主程序里运行一个时间循环并依次在这个线程里响应事件.看下面这段代码,UI线程就是创建Display的那个线程.
public static void main (String [] args) {
    Display display = new Display ();
    Shell shell = new Shell (display);
    shell.open ();
    // 开始事件循环
    // 关掉窗口后
    while (!shell.isDisposed ()) {
      if (!display.readAndDispatch ())
        display.sleep ();
    }
    display.dispose ();
  }
简单的小程序里,一个UI线程就能够满足需要了.
但如果是长时间的操作,你就最好不要用UI线程来做这些事,可以交给Job去做.它其实就是另外启动的线程,也就是等会我要说的非UI线程.

2.Job
Job类由org.eclipse.core.runtime插件提供.它能够让客户程序员轻松的在另外的线程中执行代码.
看一个小例子
Job job = new Job("My First Job") {
  protected IStatus run(IProgressMonitor monitor) {
      System.out.println("Hello World (from a background job)");
      return Status.OK_STATUS;
    }
  };
job.setPriority(Job.SHORT);
job.schedule(); // start as soon as possible

Job的默认优先级是Job.Long,这里例子中的优先级要比它高.
只要调用Job#schedule(),它就会尽快在另外的线程中运行run()中的代码.
再看一个小例子:
final Job job = new Job("Long Running Job") {
    protected IStatus run(IProgressMonitor monitor) {
      try {
        while(hasMoreWorkToDo()) {
          // do some work
          // ...
        if (monitor.isCanceled()) return Status.CANCEL_STATUS;
        }
        return Status.OK_STATUS;
      } finally {
        schedule(60000); // start again in an hour
      }
    }
  };
job.addJobChangeListener(new JobChangeAdapter() {
    public void done(IJobChangeEvent event) {
    if (event.getResult().isOK())
      postMessage("Job completed successfully");
      else
        postError("Job did not complete successfully");
    }
  });
job.setSystem(true);
  job.schedule(); // start as soon as possible

monitor 是一个进度显示条,它会在运行job时自动显示,如果任务成功运行完成,返回Status.OK_STATUS,如果中途被用户在进度显示条那里中断,就返回Status.CANCEL_STATUS.上面schedule(60000);它是让job每过1小时就自动运行,Job又一个非常强大的功能.
然后后面是可以给job添加监听器.
job.setSystem(true);这一句是把这个job设置为系统级别的.如果调用setUser(true),那么就被定义为用户级别的,用户级别和默认级别的job
    在运行时会以UI形式反映出来,如果是用户job,那么会弹出一个进度显示窗口,能让用户选择在后台里运行.
下图是一个job自动运行时的效果:

再介绍job常常用到的一个方法Job#join().
系统调用到某个job,调用它的run()方法:
再看下面这个例子:

  class TrivialJob extends Job {
    public TrivialJob() {
      super("Trivial Job");
    }
    public IStatus run(IProgressMonitor monitor) {
      System.out.println("This is a job");
      return Status.OK_STATUS;
    }
  }

  job的创建和计划如下所示:

  TrivialJob job = new TrivialJob();
  System.out.println("About to schedule a job");
  job.schedule();
  System.out.println("Finished scheduling a job");

  他们的执行是和时间没关系的,输出可能如下:
  About to schedule a job
  This is a job
  Finished scheduling a job

也可能是:

  About to schedule a job
  Finished scheduling a job
  This is a job
 

  如果希望某个job运行完成后在继续时,可以使用join()方法.
  join()会一直阻塞到该job运行完.

例子:
  TrivialJob job = new TrivialJob();
  System.out.println("About to schedule a job");
  job.schedule();
  job.join();
  if (job.getResult().isOk())
    System.out.println("Job completed with success");
  else
    System.out.println("Job did not complete successfully");

上面的代码执行后,输出应该就是这样:

  About to schedule a job
  This is a job
  Job completed with success

Job的功能是很强大的,还有很多功能我以后会介绍,也可以查阅官方帮助文档.这里先把几个常用的问题解决掉.
参见:
http://help.eclipse.org/help30/index.jsp?topic=/org.eclipse.platform.doc.isv/guide/runtime_jobs.htm

3.如果在Job中加上改变UI的代码就会失败.
原因如下:
如果是在非UI线程中调用UI,SWT就会抛出一个SWTException.
要在一个非UI线程改变UI的话有几种技术:

第一种,用:
Display#syncExec(Runnable)或
Diaplay#asyncExec(Runnable)

第二种:
已经开发了另外一种Job,就是UIJob,可以直接在它里面运行改变UI的代码,其实它就是在SWT的asyncExec()方法里运行的.所有继承UIJob的类应

该覆写runInUIThread方法而不是run方法.

3.关于进度显示
在Jface中:
org.eclipse.jface.operations包定义了一些接口用来在进度条下运行长时间的任务.
可以参见:
http://help.eclipse.org/help30/index.jsp?topic=/org.eclipse.platform.doc.isv/guide/jface_operations.htm

在eclipse插件和RCP开发中:
用户级别的job是互操作性最强的,它不仅能够让用户用Cancel键取消job,而且可以在Detail中展示具体情况,但是注意:
Detail只会在下面两种方法中出现:
IProgressService#busyCursorWhile或
IProgressService#runInUI
1)IProgressService#busyCursorWhile的用法例子:
注意这里的run()中做些和UI无关的事.
IProgressService progressService = PlatformUI.getWorkbench().getProgressService();
  progressService.busyCursorWhile(new IRunnableWithProgress(){
    public void run(IProgressMonitor monitor) {
      //do non-UI work
    }
  });
效果:

2)IProgressService#runInUI的用法例子:
注意这里的run()中可以做些和UI有关的事.
progressService.runInUI(
    PlatformUI.getWorkbench().getProgressService(),
    new IRunnableWithProgress() {
      public void run(IProgressMonitor monitor) {
        //do UI work
      }
    },
    Platform.getWorkspace().getRoot());
效果:

这里最后一个参数可以是null,或者是这个操作的规则,在这里我们是设定运行这个UI操作时锁定工作台.
更加具体的可以参见:
http://help.eclipse.org/help30/index.jsp?topic=/org.eclipse.platform.doc.isv/guide/workbench_jobs.htm

另外,有少数时候,我们不想弹出一个进度条窗口,而是只在最底下的状态栏显示就可以了,很简单,写自己的Job类时,在构造方法里加上一句:
setUser(false);就可以了.
顶端 时间: 2006年01月12日 20:56 | [楼 主]
金鳞

状态: 离线
级别: CEC版主
精华: 2
发帖: 247
威望: 48 点
财富: 191 元
在线时间:50(小时)
注册时间:2005-11-24
最后登录:2007-05-14

好文章!学习了~~
---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
金鳞岂是池中物,一遇风云便化龙
顶端 时间: 2006年01月13日 09:29 | 1 楼
jetgeng

状态: 离线
级别: CEC版主
精华: 1
发帖: 199
威望: 31 点
财富: 94 元
在线时间:51(小时)
注册时间:2005-10-27
最后登录:2008-05-16

太帅了。不顶都不好意思了。
---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
爱软件,爱eclipse
我的blog:http://www.blogjava.net/JetGeng/
顶端 时间: 2006年02月18日 12:10 | 2 楼
coolying_1981

状态: 离线
级别: CEC高级程序员
精华: 0
发帖: 30
威望: 12 点
财富: 239 元
在线时间:8(小时)
注册时间:2005-12-06
最后登录:2008-12-02

很厉害 ,好东西 谢谢!
顶端 时间: 2006年02月28日 09:59 | 3 楼
red_pupa

状态: 离线
级别: CEC高级程序员
精华: 0
发帖: 5
威望: 11 点
财富: 205 元
在线时间:9(小时)
注册时间:2006-02-20
最后登录:2007-04-25

good archive~

一般写Eclipse插件时关于线程有些要特别注意的,给大家提个醒:

- Eclipse平台本身具有多线程特性。对于这点,大家写的所有类应该要仔细考虑线程安全。这个在通讯应用中很常见,包括所有通讯都在一个单线程信道内进行的情况。如果没注意同步的话,哼哼……

- 平台本身提供的一些UI操作是由Eclipse UI线程产生的,所以扩展这些操作时,绝大部分情况要保证“非阻塞”,这种情况远程通讯过程中不注意就会违反的,不过如果设计者甘愿冒冻结Eclipse UI的风险的话另当别论。一般设计时要权衡好采用同步还是异步。
[ 此贴被red_pupa在2006年02月28日 11:07重新编辑 ]
顶端 时间: 2006年02月28日 10:59 | 4 楼
czhcc

状态: 离线
级别: CEC高级程序员
精华: 0
发帖: 30
威望: 1 点
财富: 254 元
在线时间:4(小时)
注册时间:2005-11-09
最后登录:2007-12-27

我用楼主的代码试验了一下,但不显示进度显示条,请问为什么?
顶端 时间: 2006年03月06日 10:50 | 5 楼
relay

状态: 离线
级别: CEC程序员
精华: 0
发帖: 208
威望: 11 点
财富: 0 元
在线时间:33(小时)
注册时间:2006-03-11
最后登录:2008-10-13

写的很详细,非常好,支持!
---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
思想决定行动,
行动决定习惯,
习惯决定品德,
品德决定命运。
顶端 时间: 2006年03月13日 10:08 | 6 楼
wfb

状态: 离线
级别: CEC程序员
精华: 0
发帖: 4
威望: 11 点
财富: 2 元
在线时间:0(小时)
注册时间:2006-03-19
最后登录:2008-07-09

好象很不错啊
顶端 时间: 2006年03月19日 17:55 | 7 楼
wuyouwen

状态: 离线
级别: CEC高级程序员
精华: 0
发帖: 31
威望: 11 点
财富: 234 元
在线时间:9(小时)
注册时间:2006-03-05
最后登录:2006-06-16

值得顶一下
顶端 时间: 2006年03月20日 20:08 | 8 楼
yoyo8208

状态: 离线
级别: CEC高级程序员
精华: 0
发帖: 2
威望: 11 点
财富: 202 元
在线时间:1(小时)
注册时间:2006-04-05
最后登录:2006-04-06

大力支持楼主的贡献精神
顶端 时间: 2006年04月05日 21:07 | 9 楼
bmlinkai

状态: 离线
级别: CEC高级程序员
精华: 0
发帖: 41
威望: 1 点
财富: 253 元
在线时间:6(小时)
注册时间:2005-10-29
最后登录:2008-07-11

我也想问问,样才能显示那个进度条啊。。。。
顶端 时间: 2006年04月06日 01:36 | 10 楼
superstar

状态: 离线
级别: CEC高级程序员
精华: 0
发帖: 14
威望: 11 点
财富: 214 元
在线时间:2(小时)
注册时间:2006-04-06
最后登录:2006-09-04

thankyou!
顶端 时间: 2006年04月06日 13:29 | 11 楼
krf301

状态: 离线
级别: CEC程序员
精华: 0
发帖: 314
威望: 15 点
财富: 1 元
在线时间:30(小时)
注册时间:2005-11-28
最后登录:2008-12-02

今天才看这个贴子。真的很不错。
顶端 时间: 2006年05月17日 20:05 | 12 楼
混北民工

状态: 离线
级别: CEC总版
精华: 4
发帖: 112
威望: 53 点
财富: 313 元
在线时间:11(小时)
注册时间:2006-06-01
最后登录:2008-03-15

Job在以插件方式开发的程序才能显示进度条。它是利用平台本身提供的进度条服务。显示的位置是在右下方的一小进度条图标,点击后可以弹出大的进度框。效果见图:
[ 此贴被混北民工在2006年05月30日 19:00重新编辑 ]
图片:
顶端 时间: 2006年05月30日 18:44 | 13 楼
krf301

状态: 离线
级别: CEC程序员
精华: 0
发帖: 314
威望: 15 点
财富: 1 元
在线时间:30(小时)
注册时间:2005-11-28
最后登录:2008-12-02

你的什么抓图软件呀。这么牛X。
顶端 时间: 2006年05月30日 21:13 | 14 楼
混北民工

状态: 离线
级别: CEC总版
精华: 4
发帖: 112
威望: 53 点
财富: 313 元
在线时间:11(小时)
注册时间:2006-06-01
最后登录:2008-03-15

Quote:
引用第14楼krf3012006年05月30日 21:13发表的“”:
你的什么抓图软件呀。这么牛X。


SnagIt 8
顶端 时间: 2006年05月31日 11:32 | 15 楼
yz_zcg

状态: 离线
级别: CEC高级程序员
精华: 0
发帖: 15
威望: 1 点
财富: 115 元
在线时间:0(小时)
注册时间:2006-05-31
最后登录:2007-04-19

---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
努力就会成功
顶端 时间: 2006年06月01日 16:36 | 16 楼
笑脸男人

状态: 离线
级别: CEC高级程序员
精华: 0
发帖: 14
威望: 1 点
财富: 117 元
在线时间:0(小时)
注册时间:2006-05-11
最后登录:2007-06-20

好贴。要顶
顶端 时间: 2006年06月07日 01:04 | 17 楼
zf00110011

状态: 离线
级别: CEC高级程序员
精华: 0
发帖: 37
威望: 1 点
财富: 146 元
在线时间:0(小时)
注册时间:2006-05-18
最后登录:2006-07-19

org.eclipse.core.runtime这个包哪里有下载的?
顶端 时间: 2006年07月06日 11:22 | 18 楼
bluejerry

状态: 离线
级别: CEC高级程序员
精华: 0
发帖: 9
威望: 1 点
财富: 109 元
在线时间:0(小时)
注册时间:2006-07-07
最后登录:2006-07-09

写的很好!
顶端 时间: 2006年07月07日 16:35 | 19 楼
« 1 23» Pages: ( 1/3 total )
中国Eclipse社区 » 插件开发


辽ICP备05021625号