首页 > 代码库 > JAVA学习第二十二课(多线程(二))- (多线程的创建方式一 :继承Thread类)
JAVA学习第二十二课(多线程(二))- (多线程的创建方式一 :继承Thread类)
线程是程序中的执行线程。Java 虚拟机允许应用程序并发地运行多个执行线程。
创建新执行线程有两种方法。
一种方法是将类声明为 Thread
的子类。该子类应重写Thread
类的run
方法。另一种方法是声明实现 Runnable
接口的类。该类然后实现run
方法。
创建线程方式一:继承Thread类
一、创建线程的步骤:
1.定义一个类继承Thread类
2.覆盖Thread中的run()方法
3.直接创建Thread类子类的对象
4.调用start方法开启线程,并调用线程的任务run方法的执行
/*创建线程的目的是为了开启一条执行路径,去运行指定的代码和其他的代码实现同时运行
*
* 运行的指定代码就是这个执行路径的任务
*
* JVM创建的主线程的任务都定义在了主函数main中
* 而自定义的线程它的任务在哪?
* Thread类是描述线程的,线程是需要任务的,所以Thread类也对任务进行描述
* 这个任务通过Thread类中的run方法来体现,run方法就是封装自定义线程运行任务的函数
*
* 简而言之:run方法中定义的就是线程要运行的任务代码
*
* 开启线程就是为了运行指定代码,所以只有继承Thread类,并复写run方法
* 将运行的代码定义在run方法中即可,这就是为什么要继承Thread类的原因
*
*/
二、调用start和run的区别:
调用start方法实现多线程,而调用run方法没有实现多线程
用start方法来启动线程,真正实现了多线程运行,这时无需等待run方法体代码执行完毕而直接继续执行下面的代码。
调用run()方法只是类的一个普通方法而已,如果直接调用Run方法,程序中只有主线程这一个线程,其程序执行路径还是只有一条,所以要按顺序执行,还是要等待run方法体执行完毕后才可继续执行下面的代码
class Demo extends Thread { private String name; public Demo(String name) { this.name = name; } public void run() { for(int i = 0;i<10;i++) { System.out.println(name+"i = "+i); } } } public class Main { public static void main(String[] args) { Demo aDemo = new Demo("A"); Demo bDemo = new Demo("B"); //aDemo.run(); //bDemo.run(); //这样只会是主线程在跑,创建线程调用方法和以前我们创建对象调用方法是一样的 //所以使用start使该线程开始执行;Java 虚拟机调用该线程的 run 方法。 aDemo.start();System.out.println("sd");//sd的打印就是随机的了 bDemo.start(); } }
可以通过getName()获取线程的名称,编号方式:Thread-0(从0开始)
而线程一创建 就有编号了
public void run() { for(int i = 0;i<10;i++) { System.out.println(name+"i = "+i+"name="+getName()); } }
如何获取当前运行线程的名字呢?
Thread类中提供了该方法
class Demo extends Thread { private String name; public Demo(String name) { super(name);//当然线程的名字也可以自定义 this.name = name; } public void run() { for(int i = 0;i<10;i++) { System.out.println(name+"i = "+i+"name="+Thread.currentThread().getName()); } } } public class Main { public static void main(String[] args) { Demo aDemo = new Demo("A"); Demo bDemo = new Demo("B"); aDemo.start(); bDemo.start(); System.out.println("OVER "+Thread.currentThread().getName()); } }
所以主线程的名称就是main
三、线程的运行图解
PS:(假如主函数中存在System.out.println(1/0);)主线程出现异常,并不影响其他线程的执行,所以谁发生异常谁结束,其他的不影响
四、线程的状态
线程有4种状态:被创建、运行、冻结(让线程停一会)、消亡
其中sleep方法需要指定睡眠时间,单位是毫秒
wait方法,没有参数,线程会一直冻结,但是不消亡,所以用notify()方法来进行唤醒
有一个特殊的状态:就绪,具备了执行资格,但是还没有获取资源,正在等待执行权,当调用了线程对象的start方法之后,该线程就进入了就绪状态,但是此时线程调度程序还没有把该线程设置为当前线程,此时处于就绪状态。在线程运行之后,从等待或者睡眠中回来之后,也会处于就绪状态。
临时阻塞状态:线程正在运行的时候,被暂停,通常是为了等待某个时间的发生(比如说某项资源就绪)之后再继续运行。sleep,suspend,wait等方法都可以导致线程阻塞。
所以:运行、冻结、临时阻塞,三种状态的关系图
一个线程想冻结,必须先具备执行资格和执行权。
运行和冻结,运行和就绪可以相互转化,但是冻结和就绪不性
一个正在处于运行状态的线程,具备着CPU的执行资格,同时具备着CPU的执行权。
一个正在处于冻结状态的线程,释放CPU的执行权的同时,也在释放CPU的执行资格。冻结状态一结束,就具备了执行资格了
PS:
CPU执行资格:可以被CPU处理,在处理队列中进行排队的
CPU的执行权:正在被CPU处理
JAVA学习第二十二课(多线程(二))- (多线程的创建方式一 :继承Thread类)