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();
}
}
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());
}
}
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());
}
}
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处理