java多线程(二)-Runnable和Thread

时间:2022-05-17 22:49:03

Java在顺序性语言的基础上提供了多线程的支持。Java的线程机制是抢占式的。这表示调度机制会周期的中断线程,将上下文切换到另一个线程,从而为每个线程都提供时间片。(与抢占式多线程对应的是 协作式多线程,每个子线程都自动的放弃控制,这就要求程序员需要在子线程的代码中插入某些让步语句)。

Java的多线程,最常见的是 Runnable和Thread类。

示例代码:src/thread_runnable/CountDown.java

 1 public  class CountDown implements Runnable{
2 protected int countDown = 5;
3 private static int myId= 0;
4 private final int taskId = myId++;
5 public CountDown() {
6 super();
7 }
8
9
10 public CountDown(int countDown) {
11 super();
12 this.countDown = countDown;
13 }
14
15 public String status() {
16 return "#" + taskId + "#(" + (countDown>0 ? countDown : "Over") + "), " + Thread.currentThread().getName();
17 }
18 public void run() {
19 while(countDown-- > 0){
20 System.out.println(status());
21 Thread.yield();
22 }
23 }
24

输出结果:

java多线程(二)-Runnable和Thread

 

通过输出可以看出来,Runnable的运行其实还是main线程。(当一个java程序运行时,默认会启动两个线程,一个main线程,一个gc线程)。
所以说,Runnable接口其实就是 描述任务的接口,要想定义某个任务,可以 实现Runnable接口并编写run()方法。
另外,代码当中的 Thread.yield() 方法是对线程调度器的一种建议。意思是“当前线程最重要的部分已经执行完毕了,可以把时间片切换给其他线程了”。当然,该方法也就是个建议而已,实际效果这个取决于具体的jvm。就像你跟领导提意见一样,建议可以提,但是领导是否采纳那就另一说了。因为java的线程机制是抢占式机制。

 

 

而实际上,真正实现启动新线程的是 Thread类,
示例代码:src/thread_runnable/DirectThread.java

 1 public class DirectThread extends Thread{
2 protected int countDown = 5;
3
4 public String status() {
5 return "###(" + (countDown>0 ? countDown : "Over") + "), " + Thread.currentThread().getName();
6 }
7 @Override
8 public void run() {
9 // TODO Auto-generated method stub
10 while(countDown-- > 0){
11 System.out.println(status());
12 }
13 }
14
15 public static void main(String[] args) {
16 System.out.println("DirectThread --- main start, " + Thread.currentThread().getName());
17 new DirectThread().start();
18 System.out.println("DirectThread --- main end, " + Thread.currentThread().getName());
19 }
20 }

输出结果:

java多线程(二)-Runnable和Thread

 

通过代码和输出结果,我们可以观察到Thread类,的确启动了一个新线程,并且可以看到main函数方法最后一条语句执行完,才开始输出Thread类中 run()方法当中的内容。这个run()方法就是新线程需要执行的任务。

不过在实际当中,我们经常并不是单独使用Thread类,而是在Thread类的构造器中传递类一个 Runnable()对象。然后调用Thread对象的start()方法来为该线程执行必须的初始化操作。然后系统会自动调用Runnable的run()方法。

 

示例代码:src/thread_runnable/ThreadRunnable.java

 1 public class ThreadRunnable{
2
3 private static final String TAG = "LiftOff";
4
5 public static void main(String[] args) {
6 // TODO Auto-generated method stub
7
8 System.out.println("main start, " + Thread.currentThread().getName());
9 for (int i=0; i<3; i++){
10 Thread thread = new Thread(new CountDown());
11 thread.start();
12 }
13 System.out.println("main end, " + Thread.currentThread().getName());
14 }
15
16 }

 

输出结果: (每次执行的结果都有差别)

java多线程(二)-Runnable和Thread

 

可以看出,新启动了三个新线程,来输出各自的任务,并且从结果,还可以看出,三个线程的执行顺序都是随机的,并且每次执行的结果都不同。那么这就说明线程运行的时候是被混在一起的,这种线程的切换就是线程调度器来进行自动控制的。具有不确定性。而对于多核处理器,也有可能在几个不同的核上同时执行着几个线程,但是对于输出信息到控制台,(就是 System.out.println()语句),这种输出资源是只存在一份的。

既然可以直接使用Thread()对象的start()方法来启动新线程,那为什么还要使用Runnable()对象来传递到Thread()对象的构造方法中,在Runnable对象的run()方法中来执行任务呢。原因如下:
(1)java是单根继承,而Runnable是interface。方便了任务的继承。
(2)解耦,任务本身与线程分离,两者相互独立。也方便任务代码被多个线程共享。

 

 

这几篇java多线程文章的demo代码下载地址 http://download.csdn.net/detail/yaowen369/9786452