一、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方法中定义需要执行的任务。
class MyThread extends Thread {
private static int num = 0; public MyThread() {
num++; } @Override
public void run() {
System.out.println("主动创建的第" + num + "个线程");
}
}
创建好了自己的线程类之后,就可以创建线程对象了,然后通过start()方法去启动线程。注意,不是调用run方法启动线程,run方法只是定义需要执行的任务。如果调用run方法,即相当于在主线程中执行run方法,跟普通的方法调用没有任何区别,此时并不会创建一个新的线程来执行定义的任务。
package com.meng.javalanguage.thread.test; public class MyThreadTest {
public static void main(String[] args) {
MyThread thread = new MyThread();
thread.start();
}
} class MyThread extends Thread {
private static int num = 0; public MyThread() {
num++;
} @Override
public void run() {
System.out.println("主动创建的第" + num + "个线程");
}
}
在上面的代码中,通过调用start()方法,就会创建一个新的线程。为了分清start()方法调用和run()方法调用的区别,请看下面一个例子:
package com.meng.javalanguage.thread.test; public class MyThreadTest {
public static void main(String[] args) {
System.out.println("主线程ID:" + Thread.currentThread().getId());
MyThread thread1 = new MyThread("thread1");
thread1.start();
MyThread thread2 = new MyThread("thread2");
thread2.run();
}
} class MyThread extends Thread {
private String name;
public MyThread(String name) {
this.name = name;
} @Override
public void run() {
System.out.println("name:" + name + " 子线程ID:" + Thread.currentThread().getId());
}
}
运行结果:
从输出结果可以得出以下结论:
1)thread1和thread2的线程ID不同,thread2和主线程ID相同,说明通过run方法调用并不会创建新的线程,而是在主线程中直接运行run方法,跟普通的方法调用没有任何区别;
2)虽然thread1的start方法调用在thread2的run方法前面调用,但是先输出的是thread2的run方法调用的相关信息,说明新线程创建的过程不会阻塞主线程的后续执行。
2、实现Runnable接口
在Java中创建线程除了继承Thread类之外,还可以通过实现Runnable接口来实现类似的功能。实现Runnable接口必须重写其run方法。
下面是一个例子:
package com.meng.javalanguage.thread.test; public class MyThreadTest {
public static void main(String[] args) {
System.out.println("主线程ID:" + Thread.currentThread().getId());
MyRunnable runnable = new MyRunnable();
Thread thread = new Thread(runnable);
thread.start();
}
} class MyRunnable implements Runnable { public MyRunnable() { } @Override
public void run() { System.out.println("子线程ID:" + Thread.currentThread().getId());
}
}
Runnable的中文意思是“任务”,顾名思义,通过实现Runnable接口,我们定义了一个子任务,然后将子任务交由Thread去执行。注意:这种方式必须将Runnable作为Thread类的参数,然后通过Thread的start方法创建一个新线程来执行该子任务。如果调用Runnable的run方法的话,是不会创建新线程的,这跟普通的方法调用没有任何区别。
事实上,查看Thread类的实现源代码会发现Thread类是实现了Runnable接口的。
在Java中,这两种方式都可以用来创建线程去执行子任务,具体选择哪一种方式要看自己的需求。直接继承Thread类的话,可能比实现Runnable接口看起来更加简洁,但是由于Java只允许单继承,所以如果自定义类需要继承其它类,则只能选择实现Runnable接口。