多线程

一、程序、进程与多任务

  • 程序是对数据描述与操作的代码的集合,是应用程序执行的脚本。
  • 进程是程序的一次执行过程,是系统运行程序的基本单位。程序是静态的,进程是动态的。系统运行一个程序即是一个进程从创建、运行到消亡的过程。
  • 多任务在一个系统中可以同时运行多个程序,即有多个独立运行的任务,每个任务对应一个进程。

二、线程

  • 线程:比进程更小的运行单位,是程序中单个顺序的流控制。一个进程中可以包含多和线程。
  • 简单来讲,线程是一个独立的执行流,是进程内部的一个独立执行单位,相当于一个子程序。
  • 一个进程中的所有线程都在该进程的虚拟地址空间中,使用该进程的全局变量和系统资源。
  • 操作系统给每个线程分配不同的CPU时间片,在某一时刻,CPU只执行一个时间片内的线程,多个时间片中的相应线程在CPU内轮流执行。

三、创建多线程

  • 每个Java程序启动后,虚拟机将自动创建一个主线程。
  • 可以通过一下两种方式自定义线程类:
    -创建Java.lang.Thread类的子类,重写该类的run方法
    -创建java.lang.Runnable接口的实现类,实现接口中的run方法

四、继承Thread类

  • Thread:代表一个线程类
构造方法含义
Thread创建一个新的线程对象
Thread(Runnable target)基于Runnable接口实现类的实例创建一个线程对象
Thread(Runnable t,String name)基于给定的Runnbale接口实现类的实例和指定名字创建一个线程对象
Thred(String name)基于给定的名称创建一个县城对象

Thread类

-run方法:包括线程运行是执行的代码,通常在子类中重写它。
-start方法:启动一个新的线程,然后虚拟机调用新线程的run方法。

Thread类代码示例

//MyThread类是一个线程类
class MyThread extends Thread{
	//重写父类的run方法
	public void run(){
		for(int i = 1 ; i < = 100; i++)
			System.out.println("线程"+getName()+"正在打印"+i);
		}
	}	
	public static void main(String[] args){
		//获取当前线程的名字
		String name = Thread.currentThread().getName();
		System.out.println(name + "开始执行。。。");
		//创建线程对象
		MyThread thread = new MyThread();
		//启动线程
		thread.start();
		for(int i = 0 ; i <= 100 ; i++){
			System.out.println("线程"+ name +"正在打印"+i);
			System.out.println(name + "执行完毕");
		}
	}

}

创建多线程

一、Runnable接口
  • Runnable接口中只有一个为实现的run方法,实现该类接口的类必须重写该方法。
  • Runnable接口与Thread类之间的区别
    —Runnable接口必须实现run方法,而Thread类中的run方法是一个空方法,可以不重写
    —Runnable接口的实现类并不是真正的线程类,只是线程的目标类。要想以线程的方式执行run方法,必须依靠Thread类
    —Runnable接口适合于资源的共享

在这里插入图片描述

线程的生命周期

  • 线程的生命周期:
    —指线程从创建到启动,直至运行结束
    —可以通过调用Thread类的相关方法影响线程的运行状态
  • 线程的运行状态
    -新建(New)
    -可执行(Runnable)
    -运行(Running)
    -阻塞 (Blocking)
    -死亡 (Dead)

在这里插入图片描述

  • 新建状态(New)
    —当创建了一个Thread对象是,该对象就处于“新建状态”
    —没有启动,因此无法运行
  • 可执行状态(Runnable)
    —其他线程调用了处于新建状态线程的start方法,该线程对象将转换到“可执行状态”
    —线程拥有获得CPU控制权的机会,处在等待调度阶段。
  • 运行状态(Running)
    —处在“可执行状态”的线程对象一旦获得了CPU控制权,就会转换到“执行状态”
    —在“执行状态”下,线程转台占用CPU时间片段,执行run方法中的代码
    —处在“执行状态”下的线程可以调用yield方法,该方法用于主动出让CPU控制权。线程对象出让控制权后回到“可执行状态”,重新等待调度。

在这里插入图片描述

  • 阻塞状态(Blocking)
    —线程在“执行状态”下由于受某种条件的影响会被迫出让CPU控制权,进入“阻塞状态”。
  • 进入阻塞状态的三种情况
    —调用sleep方法
    —调用join方法
    —执行I/O操作

-调用sleep方法
public void sleep(long millis)
—Thread类的sleep方法用于让当前线程暂时休眠一段时间
—参数millis的单位是毫秒

在这里插入图片描述

-调用join方法(合并某个线程)
—处在“执行状态”的线程如果调用了娶她线程的join方法,将被挂起进入“阻塞状态”
—目标线程执行完毕后才会解除阻塞,回到“可执行状态”

  • 执行I/O操作
    —线程在执行过程中如果因为访问外部资源(等待用户键盘输入、访问网络)时发生了阻塞,也会导致当前线程进入“阻塞状态”。

在这里插入图片描述

在这里插入图片描述

  • 解除阻塞
    —睡眠状态超时
    —调用join后等待其他线程执行完毕
    —I/O操作执行完毕
    —调用阻塞线程的interrupt方法(线程睡眠啊是,调用该线程的interrupt方法会跑出InterruptedException)

在这里插入图片描述

  • 死亡状态(Dead):处于“执行状态”的线程一旦从run方法返回(无论是正常退出还是抛出异常)就会进入“死亡状态”。
  • 已经“死亡”的线程不能重新运行,否则会抛出IllegalThreadStateException
  • 可以使用Thread类的isAlive方法判断线程是否活着

在这里插入图片描述
在这里插入图片描述

  • 线程调度
    —按照特定机制为线程分配CPU时间片段的行为
    —Java程序运行是,有Java虚拟机负责线程的调度

  • 线程调度的实现方式
    —分时调度模型:让所有线程轮流获得CPU的控制权,并且为每个线程平均分配CPU时间片段
    —抢占式调度模式:选择优先级响度较高的线程执行,如果所有线程的优先级相同,则随机选择一个线程执行。
    Java虚拟机采用此种调度模型。

  • 线程的优先级
    -Thread类提供了获取和设置线程优先级的方法
    —getPriority:获取当前线程的优先级
    —setPriority:设置当前线程的优先级

  • Java语言为线程类设置了10个优先级,分别使用1~10内的整数表示,整数值越大代表优先级越高。每个线程都有一个默认的优先级,主线程的默认优先级是5。

  • Thread类定义的三个常量分别代表了几个常用的优先级:
    —MAX_PRIORITY::代表了最高优先级10
    —MIN_PRIORITY::代表了最低优先级1
    —NORM_PRIORITY::代表了正常优先级5

  • setPriority不一定起作用,在不同的操作系统、不同的JVM上,效果也可能不同。操作系统也不能保证设置了优先级的线程就一定会先运行或得到更多的CPU时间。

  • 在实际使用中,不建议使用该方法
    -在这里插入图片描述

在这里插入图片描述

线程同步

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

测试方法

在这里插入图片描述

线程同步

  • 为了得到两个线程并发操作的结果,下面稍微修改一下run()方法
    在这里插入图片描述

线程安全

  • 多线程应用程序同时访问共享对象是,由于线程间相互抢占CPU的控制权,造成一个线程夹在另一个线程的执行过程中运行,所以可能导致错误的执行结果。
    在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

Synchronized关键字

  • 为了防止共享对象在并发访问是出现错误,Java中提供了“synchronized”关键字。
  • synchronized关键字
    —确保共享对象在同一时刻只能被一个线程访问,这种处理机制称为“线程同步”或“线程互斥”。
    Java中的“线程同步”基于“对象锁”的概念
  • 使用synchronized关键字
    —修饰方法:被“synchronized”关键字修饰的放大称为“同步方法”
    在这里插入图片描述

—当一个线程访问对象的同步方法时,被访问对象就处于"锁定"状态,访问该方法的其他线程只能等待,对象中的其他同步方法也不能访问,但非同步方法则可以访问

在这里插入图片描述

  • 重新修改小朋友分苹果的例子,将getApple方法改为同步方法
    在这里插入图片描述

  • 使用“synchronized”关键字:修饰部分代码,如果只希望同步部分代码行,可以使用“同步快”
    在这里插入图片描述

  • 同步块的作用于同步方法一样,只是控制范围有所区别
    在这里插入图片描述

线程通信

  • 当一个线程正在使用同步方法是,其他线程就不能使用这个同步方法,而有时涉及一些特殊情况:
    —当一个人在一个售票窗口排队买电影票时,如果她给售票员的不是零钱,而售票员有没有售票员找她,那么她必须等待,并允许后面的人买票,以便售票员获取零钱找她,如果第2个人也没有零钱,那么她俩必须同时等待。

  • 当一个线程使用的同步方法中用到某个变量,因此变量有需要其他线程修改后才能符合本线程的需要,那么可以再同步方法中使用wait()方法

  • wait()方法
    —中断方法的执行,使本线程等待,暂时让出cpu的使用权,并允许其他线程使用这个同步方法。

  • notify()方法:
    —唤醒由于使用这个同步方法而处于等待线程的某一个结束等待

  • notifyall()方法
    —唤醒所有由于使用这个同步方法而处于等待的线程结束等待

  • 线程通讯示例

  • 模拟3个人,张飞、关羽和刘备,来买电影票,售票员只有一张5元的钱,电影票5元钱一张:张飞拿20元一张的人民币排在关羽和刘备的前面,关羽和刘备各拿了一张5元的人民币买票。因此张飞必须等待(还是关羽和刘备先买了票)
    在这里插入图片描述
    在这里插入图片描述

在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值