一、Java中关于应用程序和进程相关的概念
在Java中,一个应用程序对应着一个JVM实例(也有地方称为JVM进程),一般来说名字默认是java.exe或者javaw.exe(windows下可以通过任务管理器查看)。Java采用的是单线程编程模型,即在我们自己的程序中如果没有主动创建线程的话,只会创建一个线程,通常称为主线程。但是要注意,虽然只有一个线程来执行任务,不代表JVM中只有一个线程,JVM实例在创建的时候,同时会创建很多其它的线程(比如垃圾收集器线程)。
由于Java采用的是单线程编程模型,因此在进行UI编程时要注意将耗时的操作放在子线程中进行,以避免阻塞主线程(在UI编程时,主线程即UI线程,用来处理用户的交互事件)。
二、Java中如何创建线程
在Java中如果要创建线程的话,一般有两种方式:1)继承Thread类;2)实现Runnable接口。
1、继承Thread类
继承Thread类的话,必须重写run方法,在run方法中定义需要执行的任务。
1 class MyThread extends Thread { 2 private static int num = 0; 3 4 public MyThread() { 5 num++; 6 7 } 8 9 @Override 10 public void run() { 11 System.out.println("主动创建的第" + num + "个线程"); 12 } 13 }
创建好了自己的线程类之后,就可以创建线程对象了,然后通过start()方法去启动线程。注意,不是调用run方法启动线程,run方法只是定义需要执行的任务。如果调用run方法,即相当于在主线程中执行run方法,跟普通的方法调用没有任何区别,此时并不会创建一个新的线程来执行定义的任务。
1 package com.meng.javalanguage.thread.test; 2 3 public class MyThreadTest { 4 public static void main(String[] args) { 5 MyThread thread = new MyThread(); 6 thread.start(); 7 } 8 } 9 10 class MyThread extends Thread { 11 private static int num = 0; 12 13 public MyThread() { 14 num++; 15 } 16 17 @Override 18 public void run() { 19 System.out.println("主动创建的第" + num + "个线程"); 20 } 21 }
在上面的代码中,通过调用start()方法,就会创建一个新的线程。为了分清start()方法调用和run()方法调用的区别,请看下面一个例子:
1 package com.meng.javalanguage.thread.test; 2 3 public class MyThreadTest { 4 public static void main(String[] args) { 5 System.out.println("主线程ID:" + Thread.currentThread().getId()); 6 MyThread thread1 = new MyThread("thread1"); 7 thread1.start(); 8 MyThread thread2 = new MyThread("thread2"); 9 thread2.run(); 10 } 11 } 12 13 class MyThread extends Thread { 14 private String name; 15 public MyThread(String name) { 16 this.name = name; 17 } 18 19 @Override 20 public void run() { 21 System.out.println("name:" + name + " 子线程ID:" + Thread.currentThread().getId()); 22 } 23 }
运行结果:
从输出结果可以得出以下结论:
1)thread1和thread2的线程ID不同,thread2和主线程ID相同,说明通过run方法调用并不会创建新的线程,而是在主线程中直接运行run方法,跟普通的方法调用没有任何区别;
2)虽然thread1的start方法调用在thread2的run方法前面调用,但是先输出的是thread2的run方法调用的相关信息,说明新线程创建的过程不会阻塞主线程的后续执行。
2、实现Runnable接口
在Java中创建线程除了继承Thread类之外,还可以通过实现Runnable接口来实现类似的功能。实现Runnable接口必须重写其run方法。
下面是一个例子:
1 package com.meng.javalanguage.thread.test; 2 3 public class MyThreadTest { 4 public static void main(String[] args) { 5 System.out.println("主线程ID:" + Thread.currentThread().getId()); 6 MyRunnable runnable = new MyRunnable(); 7 Thread thread = new Thread(runnable); 8 thread.start(); 9 } 10 } 11 12 class MyRunnable implements Runnable { 13 14 public MyRunnable() { 15 16 } 17 18 @Override 19 public void run() { 20 21 System.out.println("子线程ID:" + Thread.currentThread().getId()); 22 } 23 }
Runnable的中文意思是“任务”,顾名思义,通过实现Runnable接口,我们定义了一个子任务,然后将子任务交由Thread去执行。注意:这种方式必须将Runnable作为Thread类的参数,然后通过Thread的start方法创建一个新线程来执行该子任务。如果调用Runnable的run方法的话,是不会创建新线程的,这跟普通的方法调用没有任何区别。
事实上,查看Thread类的实现源代码会发现Thread类是实现了Runnable接口的。
在Java中,这两种方式都可以用来创建线程去执行子任务,具体选择哪一种方式要看自己的需求。直接继承Thread类的话,可能比实现Runnable接口看起来更加简洁,但是由于Java只允许单继承,所以如果自定义类需要继承其它类,则只能选择实现Runnable接口。