所谓的线程是指程序的运行流程,可以看做进程的一个执行路径。“多线程”的机制是指,可以同时运行多个程序块(进程的多条路径),可克服传统程序语言所无法解决的问题。
通过继承Thread类实现多线程
因为Thread类存放于java.lang类库中。所以我们可以直接使用Thread类,而无需显式加载。
由于在Thread类中,已经定义了run()方法。所以,如果用户想要实现多线程,必须定义继承于Thread的子类,同时覆写Thread中的run()方法。
class 类名称 extends Thread{ //从Thread类中扩展出
属性。。。
方法。。。
修饰符 run() //覆写Thread类中的run()方法
{
程序代码; //激活的线程,将从run方法开始执行
}
}
然后再使用用户自定义的线程类,生成对象,并调用该对象的start()方法,从而来激活一个新的线程。
通过实现Runnable接口实现多线程
Java中不允许多继承。如果一个类继承了其他的类,那么这个类就不能继承Thread类。但是它又想实现多线程技术,那么这是就要用到Runnable接口来创建线程。一个类可实现多个接口,这样就间接的实现了多继承。
class 类名称 implements Runnable{ //实现Runnable接口
属性。。。
方法。。。
@Override
public void run() { //实现Runnable接口的run()方法
程序代码 //激活的线程将从run方法开始执行
}
}
因为在Runnable接口中只有一个run方法,所以激活一个新的线程,需要使用Thread类的start()方法。
两种多线程机制的比较
Thread类实现了Runnable接口。实质上,Thread类是Runnable接口众多的实现子类中的一个,他的地位和我们自己写一个Runnable接口的实现类没有多大区别。
实现Runnable接口对于继承Thread类来说的优点:
- 避免了由于Java单继承特性带来的局限性
- 可使多个线程共享相同的资源,以达到资源共享的目的
Java8中创建线程的新方法
Java8中引入了Lambda表达式,使创建线程的形式改变了。
()->{程序语句} 是lambda表达式,这个语句等同于创建了Runnable()接口的一个匿名子类,并使用new操作创建这个子类的匿名对象,然后把这个匿名对象当做Thread类的构造方法中的一个参数。
Lambda表达式结构:
- 最前面的部分是一对括号,里面是参数
- 中间是 -> ,用来分割参数和主体部分
- 主体部分可以使一个表达式或者一个语句块,用花括号括起来。如果是表达式,表达式的值会被当做返回值返回。如果是语句块,则需要使用return语句指定返回值。
线程的状态
每个Java程序都有一个默认的主线程,对于Java应用程序,主方法是main方法执行的线程。要想实现多线程,必须在主线程中创建新的线程对象。
线程的状态
- NEW(创建态):初始状态,线程已经被创建,但尚未启动,即使还没有被调用start()方法。
- RUNNABLE(运行态):正在Java虚拟机中执行的状态处于该种状态。Java的线程概念中,就绪(REDAY)和运行(RUNNING)这两种状态,统称为可运行态(RUNNABLE)。
- BLOCKED(阻塞态):受阻塞,并等待某个监视锁。
- WATING(无限等待态):无限期的等待,表明当前线程需要等待其他线程执行某一个特定操作(通知或者中断等等)。
- TIMED_WATING(超时等待态):与WATING状态不同,他可以在指定的等待时间后,自行返回。
- TERMINATED(终止态):表示当前线程已经执行完毕
线程创建以后,需要调用start()方法启动运行,当线程执行wait()方法时,线程进入等待状态。进入等待状态的线程需要依赖其他线程的通知才能返回运行状态。对比而言,超时间等待状态,相当于在等待状态的基础上增加了超时限制,超时之后,线程会进入终止状态。
线程调用同步方法时,在没有获取到锁的情况下,线程会进入阻塞状态。
线程的操作方法
在Thread类中,可以通过getName()方法取得线程的名称,setName()方法设置线程的名称。如果程序没有给线程指定名称,系统会自动给线程分配名称。Thread类中的currentThread()方法,是个静态方法,返回值是执行该方法的线程实例。
Thread类中start()和run()方法的联系和不同
- start()方法
他的作用是启动一个新的线程,有了他的调用,才能真正实现多线程运行,这时无需等待run方法执行完毕,而是继续执行start()后的代码。
*start()方法不能被重复调用 - run()方法
run()只是类的一个普通的方法,如果直接调动run()方法,程序中依然只有主线程这一个线程,其程序的执行路径依然只有一条,也就是说,一旦run()方法被调用,程序还是按顺序执行,只有run()方法体执行完毕之后,才能继续执行后面的代码,这样并不能达到多线程并发的目的。
run()方法可以被重复调用,每次单独调用,就会在当前线程中执行run()方法,而不会启动新的线程。
判断线程是否启动
isAlive()方法可以测试线程是否已经启动而且仍然在运行。
守护线程与setDaemon方法
JVM中,线程被分为两类:用户线程与守护线程。对Java来说,只要有一个用户线程存在,程序就不会结束。
守护线程也称为后台线程,运行在后台,为用户提供一种通用服务的线程。当线程中只剩下守护线程时,JVM就会自动退出。
*设置某个线程为守护线程时,一定要在start()方法调用之前设置,也就是要在线程启动之前设置其属性。
线程的特点
- 同步代码块和同步对象锁的是对象,而不是代码。//如果某个对象呗同步代码块或同步方法锁住了,那么其他使用该对象的代码必须等待,直到该对象的锁被释放。
- 如果一个进程只有后台进程,这个进程就会结束。
- 每一个已经被创建的线程在结束后会处于就绪,运行,阻塞其一。