1:什么是多线程?
讲到多线程,就不可避免要提到程序和进程了,他们之间的大小顺序是: 程序>进程>线程 ,在我的理解中一个程序最少会包含一个进程,一个进程最少会包含一个线程,而超过一个线程的话就叫做“多线程”,就比如我点开了某易云,这个时候就等于计算机运行了一个程序,而当我点击播放的时候,就相当于开启了一个进程,而这个进程中有播放,暂停,上一曲,下一曲,评论等等…行为状态,当我点击上一曲/下一曲的时候线程是同步 的,因为业务只需要同时播放一首歌,而线程之间默认是异步的,所以我可以边看MV边听歌边写评论,这个 行为就叫做多线程
2:线程从创建、运行到结束总是处于下面五个状态之一:新建状态、就绪状态、运行状态、阻塞状态及死亡状态。
-
新建状态(New)
当用new操作符创建一个线程时, 例如new Thread®,线程还没有开始运行,此时线程处在新建状态。 当一个线程处于新生状态时,程序还没有开始运行线程中的代码
-
就绪状态(Runnable)
一个新创建的线程并不自动开始运行,要执行线程,必须调用线程的start()方法。当线程对象调用start()方法即启动了线程,start()方法创建线程运行的系统资源,并调度线程运行run()方法。当start()方法返回后,线程就处于就绪状态。
处于就绪状态的线程并不一定立即运行run()方法,线程还必须同其他线程竞争CPU时间,只有获得CPU时间才可以运行线程。因为在单CPU的计算机系统中,不可能同时运行多个线程,一个时刻仅有一个线程处于运行状态。因此此时可能有多个线程处于就绪状态。对多个处于就绪状态的线程是由Java运行时系统的线程调度程序(thread scheduler)来调度的。 -
运行状态(Running)
当线程获得CPU时间后,它才进入运行状态,真正开始执行run()方法.
-
阻塞状态(Blocked)
线程运行过程中,可能由于各种原因进入阻塞状态:
1>线程通过调用sleep方法进入睡眠状态
2>线程调用一个在I/O上被阻塞的操作,即该操作在输入输出操作完成之前不会返回到它的调用者
3>线程试图得到一个锁,而该锁正被其他线程持有;
4>线程在等待某个触发条件;
… 所谓阻塞状态是正在运行的线程没有运行结束,暂时让出CPU,这时其他处于就绪状态的线程就可以获得CPU时间,进入运行状态。
-
死亡状态(Dead)
有两个原因会导致线程死亡:
- run方法正常退出而自然死亡,
- 一个未捕获的异常终止了run方法而使线程猝死。
为了确定线程在当前是否存活着(就是要么是可运行的,要么是被阻塞了),需要使用isAlive方法。如果是可运行或被阻塞,这个方法返回true; 如果线程仍旧是new状态且不是可运行的, 或者线程死亡了,则返回false.
3:多线程的三种实现
-
继承Thread类:重写run()方法
public class MyThread extends Thread { public void start(){ System.out.println("执行start"); } public void run(){ System.out.println("执行run"); } public static void main(String[] args) { MyThread m = new MyThread();//调用MyThread m.start();//启动线程 } }
-
实现Runnable接口,实现run()方法
package io; public class MyRunnable implements Runnable{ @Override public void run() { System.out.println("启动线程"); } }
-
实现Callable接口
public class MyCallable implements Callable<Integer>{ public static void main(String[] args) throws ExecutionException, InterruptedException { Test_2_Callable t=new Test_2_Callable(); //FutureTask类,接收返回结果 FutureTask<Integer> res=new FutureTask<>(t); //启动线程,FutureTask实现了Runnable接口 new Thread(res).start(); //接收返回结果 int i=res.get(); System.out.println(i); } @Override public Integer call() throws Exception { //继承Callable接口,重写call()方法,该方法返回<V>或者抛出异常 int sum=0; for (int i = 1; i <=100; i++) { sum+=i; } return sum; } }