JavaSE——线程—死锁,线程通信

  • 回顾和提问
  1. 多线程概念
  2. 多线程创建及使用
  3. 多线程优先级
  4. 多线程生命周期
  5. 多线程同步
  6. Lock

二、本章任务

  1. 完成死锁的理解
  2. 完成线程通信的理解及使用
  3. 完成线程池的创建及使用

三、本章目标

  1. 掌握死锁的产生原因及避免
  2. 掌握线程通信方法使用
  3. 掌握线程通信
  4. 掌握线程池的使用
  • 知识点
  1. 死锁
  1. 死锁
    • 不同的线程分别占用对方需要的同步资源不放弃,都在等待对方放弃

自己需要的同步资源,就形成了线程的死锁

    • 出现死锁后,不会出现异常,不会出现提示,只是所有的线程都处于 阻塞状态,无法继续
  1. 解决方法
    • 专门的算法、原则
    • 尽量减少同步资源的定义
    • 尽量避免嵌套同步

死锁案例:见课堂练习死锁

  1. 线程通信

wait() 与 notify() 和 notifyAll()

    1. wait():令当前线程挂起并放弃CPU、同步资源并等待,使别的线程可访问并修改共享资源,而当 前线程排队等候其他线程调用notify()或notifyAll()方法唤醒,唤醒后等待重新获得对监视器的所有 权后才能继续执行。
    2. notify():唤醒正在排队等待同步资源的线程中优先级最高者结束等待
    3. notifyAll ():唤醒正在排队等待资源的所有线程结束等待.

注意:这三个方法只有在synchronized方法或synchronized代码块中才能使用,否则会报

java.lang.IllegalMonitorStateException异常

因为这三个方法必须有锁对象调用,而任意对象都可以作为synchronized的同步锁,  因此这三个方法只能在Object类中声

Wait()
  1. 在当前线程中调用方法: 对象名.wait()
  2. 使当前线程进入等待(某对象)状态 ,直到另一线程对该对象发出 notify

(或notifyAll) 为止。

  1. 调用方法的必要条件:当前线程必须具有对该对象的监控权(加锁)
  2. 调用此方法后,当前线程将释放对象监控权      ,然后进入等待

在当前线程被notify后,要重新获得监控权,然后从断点处继续代码的执行

Notify/notifyAll()
  1. 在当前线程中调用方法: 对象名.notify()
  2. 功能:唤醒等待该对象监控权的一个/所有线程。
  3. 调用方法的必要条件:当前线程必须具有对该对象的监控权(加锁)

通信案例:生产者与消费者
  1. 生产者(Productor)将产品交给店员(Clerk),而消费者(Customer)从店员处 取走产品,店员一次只能持有固定数量的产品(比如:20),如果生产者试图 生产更多的产品,店员会叫生产者停一下,如果店中有空位放产品了再通 知生产者继续生产;如果店中没有产品了,店员会告诉消费者等一下,如 果店中有产品了再通知消费者来取走产品。
  2. 这里可能出现两个问题:
  • 生产者比消费者快时,消费者会漏掉一些数据没有取到。
  • 消费者比生产者快时,消费者会取相同的数据。
  1. 线程池

背景:经常创建和销毁、使用量特别大的资源,比如并发情况下的线程, 对性能影响很大。

思路:提前创建好多个线程,放入线程池中,使用时直接获取,使用完 放回池中。可以避免频繁创建销毁、实现重复利用。类似生活中的公共交 通工具。

好处:

    1. 提高响应速度(减少了创建新线程的时间)
    2. 降低资源消耗(重复利用线程池中线程,不需要每次都创建)
    3. 便于线程管理
      1. corePoolSize:核心池的大小
      2. maximumPoolSize:最大线程数
      3. keepAliveTime:线程没有任务时最多保持多长时间后会终止

线程池API
  1. JDK 5.0起提供了线程池相关API:ExecutorService Executors
  2. ExecutorService:真正的线程池接口。常见子类ThreadPoolExecutor
    • void execute(Runnable command) :执行任务/命令,没有返回值,一般用来执行Runnable
    • <T> Future<T> submit(Callable<T> task):执行任务,有返回值,一般又来执行Callable
    • void shutdown() :关闭连接池
  1. Executors:工具类、线程池的工厂类,用于创建并返回不同类型的线程池
    • Executors.newCachedThreadPool():创建一个可根据需要创建新线程的线程池
    • Executors.newFixedThreadPool(n); 创建一个可重用固定线程数的线程池
    • Executors.newSingleThreadExecutor() :创建一个只有一个线程的线程池
    • Executors.newScheduledThreadPool(n):创建一个线程池,它可安排在给定延迟后运行命令或者定期地执行。

五、总结

见:JavaSE-死锁-线程通信-线程池.xmind

六、作业

1、完成死锁概念理解,实现死锁案例

2、完成线程通信(生产者消费者)(课上案例)

3、完成线程池创建线程(课上案例)

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值