首页 > 代码库 > 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类)