1 线程控制

t.join():让主线程进入线程池,等待t执行完才执行。

t.sleep():让线程阻塞,休眠一段时间,休眠结束后进入就绪状态。不会释放锁。

t.yield():让线程让出CPU,从运行态进入就绪态。可能会接着进入运行态。

t.setDaemon():设置为守护线程,非守护线程都死了的时候自动终止。

2 线程的分类

线程分为:守护线程 用户线程

  • 守护线程和用户线程基本上是相同的,唯一区别就是判断JVM何时离开

  • 守护线程是用来服务用户线程的。通过在start方法之前调用thread.setDaemon(true) 可以将一个用户线程变成守护线程

  • java的垃圾回收 他是一个典型的守护线程

  • 如果JVM中都是守护线程,JVM将退出(用户线程执行结束 守护线程无论是否结束 都将终止执行)

3 线程的生命周期

jdk中用Thread State定义了线程的状态:

  • 线程状态。线程可以处于以下状态之一:

    • NEW 尚未启动的线程处于此状态。

    • RUNNABLE 在Java虚拟机中执行的线程处于此状态。

    • BLOCKED 被阻塞等待监视器锁定的线程处于此状态。

    • WAITING 正在等待另一个线程执行特定动作的线程处于此状态。

    • TIMED_WAITING 正在等待另一个线程执行动作达到指定等待时间的线程处于此状态。

    • TERMINATED 已退出的线程处于此状态。

    线程的状态通常分为5种状态:

    新建:当一个Thread类及其子类的对象被声明并创建时,此时的线程对象就处于新建状态

    就绪 : 处于新建状态的线程被start后,线程将进入CPU的执行队列等待获得CPU的执行权,此时的线程已经具备了运行的条件,只是还有获得CPU的执行权

    运行:当就绪的线程获得CPU的执行权 ,处于运行状态

    阻塞:在某种特殊的情况下,被人为挂起或执行输入输出操作时,让出CPU并临时中止自己的执行。此时线程就进入阻塞状态

    死亡:线程完成了他的全部工作或被线程被提前强制性的终止 或出现异常导致线程异常结束。

    线程状态之间的相互转换

一个线程一旦死亡 是不可以在重新启动的。

4.线程同步

  1. public class SellTicketDemo {
  2. public static void main(String[] args) {
  3. SellTicket st = new SellTicket();
  4. //创建三个线程
  5. Thread t1 = new Thread(st,"1号窗口");
  6. Thread t2 = new Thread(st,"2号窗口");
  7. Thread t3 = new Thread(st,"3号窗口");
  8. //启动线程
  9. t1.start();
  10. t2.start();
  11. t3.start();

  12. }
  13. }
  14. public class SellTicket implements Runnable{
  15. private int ticktes = 100;
  16. @Override
  17. public void run() {
  18. while(true){
  19. if(ticktes > 0 ){
  20. try {
  21. Thread.sleep(100);
  22. } catch (InterruptedException e) {
  23. e.printStackTrace();
  24. }
  25. System.out.println(Thread.currentThread().getName()+"正在出售第" + ticktes+"张票");//会出现数据不同步的现象,与实际需求不符
  26. ticktes--;
  27. }
  28. }
  29. }
  30. }

4.1 解决数据安全问题--同步代码块

出现问题的条件:

1 多线程环境

2 有共享数据

3 有多条语句操作共享数据

如何解决线程安全问题:

基本的思想:让程序没有安全问题的环境

怎么实现呢? 把多条语句操作共享数据的代码给锁起来,让任意时刻只能有一个线程执行。

java提供了解决的方式是使用同步代码块或同步方法:synchronized 相当于给代码加锁

可以用在代码块和方法上 分别称为同步代码块和同步方法:

代码块:synchronized(obj/this){

共享代码;

}

在同步代码块中 谁来担当这个所对象呢?

任意对象都可以充当所对象 一般情况下使用this

4.2 解决数据安全问题--同步方法

在方法的声明上添加synchronized关键字

同步成员方法

静态同步方法:static synchronized

静态方法的同步代码块:使用类名.class作为锁对象

静态的同步方法或者静态方法中的同步代码块的所对象是类名.class对象

单例设计模式的懒汉式的线程安全问题

  1. public class Singleton {
  2. private static Singleton instance;
  3. private Singleton(){
  4. }
  5. public static Singleton getInstance(){
  6. if( instance == null){
  7. synchronized (Singleton.class){
  8. if(instance ==null){
  9. instance = new Singleton();
  10. }
  11. }
  12. }
  13. return instance;
  14. }
  15. }

4.6 线程安全的类

StringBuffer 线程安全的可变字符序列

StringBuilder 线程不安全的可变字符序列

Vector 线程同步的

HashTable 线程同步的 是一个哈希表的实现

在实际使用时,如果不需要线程安全的实现,推荐使用与之功能相同的 但是线程不同步的实现

5 线程的死锁的演示

死锁是我们需要规避的问题

不同的线程分别占用对方所需要的同步资源不放弃,都在等待对方放弃自己需要的同步资源,就行成了线程死锁

出现死锁 不会出现异常,不会出现提示,只是所有的线程都处于阻塞状态 无法继续

死锁问题的出现是一个概率时间。

死锁问题的解决:

1 减少同步代码块或同步方法的嵌套

2 尽量减少同步资源的定义

3 使用专门的算法

提示:

1 明确那些代码是多线程运行的代码,就是需要写入run方法

2 明确那些数据是共享数据

3 明确多下称运行代码中的那些语句操作了共享数据

6 线程通信

6.1 什么时候需要线程通信

多个线程并发执行,在默认情况下CPU随机切换线程,如果我们希望他们有规律的执行,就需要使用线程通信。

6.2 线程间如何通信

如果线程需要等待 就调用的wait().wait()方法可以等待时间结束或者被唤醒。如果要唤醒一个等待的线程 那么就使用notify() /notifyAll()

6.3 互斥锁

互斥锁 依次最多只能有一个线程持有锁

锁是用于通过多个线程控制对共享资源的访问的工具。 通常,锁提供对共享资源的独占访问:一次只能有一个线程可以获取锁,并且对共享资源的所有访问都要求首先获取锁

lock.lock():加锁 lock.unlock()解锁

Interface Lock 可以使用的实现类 ReentrantLock

  • 一个可重入互斥Lock具有与使用synchronized方法和语句访问的隐式监视锁相同的基本行为和语义,但具有扩展功能。

Condition

c1.await():使当前线程等待,直到c1.singal()或者interrupt()打断阻塞状态,或者到await()指定的时间。

c1.signal():唤醒一个等待线程

c1.signalAll():唤醒所有等待进程

不同的线程需要使用不同的 Condition 这样就能区分唤醒额时候唤醒的是那个线程

  1. public class ThreadLock {
  2. public static void main(String[] args) {
  3. ThreadLock tw = new ThreadLock();
  4. new Thread(new Runnable() {
  5. @Override
  6. public void run() {
  7. while(true){
  8. tw.print1();
  9. }

  10. }
  11. }).start();
  12. new Thread(){
  13. @Override
  14. public void run() {
  15. while(true){
  16. tw.print2() ;
  17. }
  18. }
  19. }.start();
  20. new Thread(){
  21. @Override
  22. public void run() {
  23. while(true){
  24. tw.print3() ;
  25. }
  26. }
  27. }.start();
  28. }
  29. // 创建Lock锁对象
  30. Lock lock = new ReentrantLock();
  31. //创建锁使用的条件
  32. Condition c1 = lock.newCondition();
  33. Condition c2 = lock.newCondition();
  34. Condition c3 = lock.newCondition();
  35. // 创建一个唤醒标志
  36. int flag = 1;
  37. public void print1(){
  38. lock.lock();//给当前代码上锁
  39. if(flag != 1){
  40. try {
  41. c1.await();//当前线程处于等待
  42. } catch (InterruptedException e) {
  43. e.printStackTrace();
  44. }
  45. }
  46. System.out.print("中");
  47. System.out.print("北");
  48. System.out.print("大");
  49. System.out.print("学");
  50. System.out.println();
  51. flag = 2;
  52. c2.signal();//唤醒c2
  53. lock.unlock();//释放锁
  54. }
  55. public void print2(){
  56. lock.lock();
  57. if(flag != 2){
  58. try {
  59. c2.await();
  60. } catch (InterruptedException e) {
  61. e.printStackTrace();
  62. }
  63. }
  64. System.out.print("塔");
  65. System.out.print("里");
  66. System.out.print("木");
  67. System.out.print("大");
  68. System.out.print("学");
  69. System.out.println();
  70. flag =3;
  71. c3.signal();
  72. lock.unlock();
  73. }
  74. public void print3(){
  75. lock.lock();
  76. if(flag != 3){
  77. try {
  78. c3.await();
  79. } catch (InterruptedException e) {
  80. e.printStackTrace();
  81. }
  82. }
  83. System.out.print("青");
  84. System.out.print("岛");
  85. System.out.print("大");
  86. System.out.print("学");
  87. System.out.println();
  88. flag =1;
  89. c1.signal();
  90. lock.unlock();
  91. }
  92. }

6.4生产者—消费者模型

生产者消费者模型的作用是什么?

1通过平衡生产者的生产能力和消费者消费能力来提升整个系统的运行效率

2 解耦

多线程的学习重点:

1 线程的创建方式

2线程的生命周期(线程的五种状态的转换)

3 线程同步

4 线程通信

15Java进阶 进程的更多相关文章

  1. Python进阶----进程之间通信(互斥锁,队列(参数:timeout和block),), ***生产消费者模型

    Python进阶----进程之间通信(互斥锁,队列(参数:timeout和block),), ***生产消费者模型 一丶互斥锁 含义: ​ ​ ​ 每个对象都对应于一个可称为" 互斥锁&qu ...

  2. Python进阶----进程间数据隔离, join阻塞等待, 进程属性, 僵尸进程和孤儿进程, 守护进程

    Python进阶----进程间数据隔离, join阻塞等待, 进程属性, 僵尸进程和孤儿进程, 守护进程 一丶获取进程以及父进程的pid 含义:    进程在内存中开启多个,操作系统如何区分这些进程, ...

  3. python进阶------进程线程(五)

    Python中的IO模型 同步(synchronous) IO和异步(asynchronous) IO,阻塞(blocking) IO和非阻塞(non-blocking)IO分别是什么,到底有什么区别 ...

  4. python进阶------进程线程(四)

    Python中的协程 协程,又称微线程,纤程.英文名Coroutine.一句话说明什么是线程:协程是一种用户态的轻量级线程. 协程拥有自己的寄存器上下文和栈.协程调度切换时,将寄存器上下文和栈保存到其 ...

  5. python进阶------进程线程(三)

    python中的进程 1.multiprocessing模块 由于GIL的存在,python中的多线程其实并不是真正的多线程,如果想要充分地使用多核CPU的资源,在python中大部分情况需要使用多进 ...

  6. python进阶-------进程线程(二)

    Python中的进程线程(二) 一.python中的"锁" 1.GIL锁(全局解释锁) 含义: Python中的线程是操作系统的原生线程,Python虚拟机使用一个全局解释器锁(G ...

  7. python进阶------进程线程(一)

    Python中的进程线程 一.进程线程的概念 1.1进程: 进程就是一个程序在一个数据集上的一次动态执行过程.进程一般由程序.数据集.进程控制块三部分组成.我们编写的程序用来描述进程要完成哪些功能以及 ...

  8. python进阶——进程/线程/协程

    1 python线程 python中Threading模块用于提供线程相关的操作,线程是应用程序中执行的最小单元. #!/usr/bin/env python # -*- coding:utf-8 - ...

  9. Java进阶(四十三)线程与进程的区别

    Java进阶(四十三)线程与进程的区别 1.线程的基本概念   概念:线程是进程中执行运算的最小单位,是进程中的一个实体,是被系统独立调度和分派的基本单位,线程自己不拥有系统资源,只拥有一点在运行中必 ...

随机推荐

  1. 类编程的WAF(上)

    一.复杂的需求 WAF (WEB 应用防火墙) 用来保护 WEB 应用免受来自应用层的攻击.作为防护对象的 WEB 应用,其功能和运行环境往往是复杂且千差万别的,这导致即便防御某个特定的攻击方式时,用 ...

  2. 01 . Varnish简介,原理,配置缓存

    简介 Varnish是高性能开源的反向代理服务器和HTTP缓存服务器,其功能与Squid服务器相似,都可以用来做HTTP缓存.可以安装 varnish 在任何web前端,同时配置它缓存内容.与传统的 ...

  3. grep过滤空行和注释行

    1)grep的排除选项为 -v排除空行的命令是:grep -v '^$' filename 2)排除以#注释的命令是:grep -v '^#' filename 3)结合起来就是,既排除空行又排除注释 ...

  4. JavaScript的介绍概括

    1.js是一种轻型的解释性的脚本语言,称为web脚本语言. 2.js的执行原理:当客户端向服务器端请求某个页面时,浏览器端将整个页面中包含JavaScript的脚本代码作为响应内容,发送到客户端的机器 ...

  5. 自定义Nuget包的技巧一二

    背景: 在项目中, 通常会拆分成核心库(Core)和应用(App)两个部分.核心库由专人维护, 不同的App是不同的团队,但都引用了核心库.当核心库需要升级更新时,有的应用会更新,有的不会--可能是没 ...

  6. docker4-docker网络,容器编排,集群部署

    1,docker网络 1.1,docker0 有三个网络环境,那么docker是如何处理容器网络访问的? 1.2,测试 docker run -d -p 80:8080 --name tomcat01 ...

  7. App免责声明

    一切移动客户端用户在下载并浏览xxxAPP软件时均被视为已经仔细阅读本条款并完全同意.凡以任何方式使用本APP,或直接.间接使用本APP资料者,均被视为自愿接受本网页相关声明和用户服务协议的约束. x ...

  8. ActiveMq 之JMS 看这一篇就够了

    什么是JMS MQ 全称:Java MessageService 中文:Java 消息服务. JMS 是 Java 的一套 API 标准,最初的目的是为了使应用程序能够访问现有的 MOM 系 统(MO ...

  9. CentOS-Docker搭建Nextcloud

    下载镜像 $ docker pull nextcloud 运行镜像 $ docker run -d --restart=unless-stopped --name nextcloud -v /home ...

  10. CentOS-配置JDK(压缩包)

    卸载openjdk $ rpm -qa | grep jdk 以上命令用来检查linux上是否安装openjdk,如果安装需要将其全部卸载掉,卸载命令: $ rpm -e --nodeps java- ...