热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

开发笔记:JUC一

警醒自己不断学习和成长

警醒自己不断学习和成长

正式学习的准备工作

JUC并发编程

1.什么是JUC

JUC就是java.util .concurrent工具包的简称。这是一个处理线程的工具包,JDK 1.5开始出现的。

技术图片

 

 

2.进程和线程回顾


什么是进程和线程?

进程:是一个具有一定独立功能的程序在一个数据集上的一次动态执行的过程,是操作系统进行资源分配和调度的一个独立单位,是应用,例如:QQ.exe

线程:是拥有资源和独立运行的最小单位,也是程序执行的最小单位.

一个进程可以包含多个线程,一个进程至少有一个线程!Java程序至少有两个线程:GC,main

并发、并行

并发:多个线程操作同一个资源,交替执行的过程!

并行:多个线程同时执行!只有在多核CPU下才能完成!

使用多线程或者并发编程的目的:提高效率,让CPU一直工作,达到最高处理性能!

线程有几种状态

线程有6种状态

public enum State {
   // java能够创建线程吗? 不能!
// 新建
   NEW,
   // 运行
   RUNNABLE,
   // 阻塞
   BLOCKED,
   // 等待
   WAITING,
   // 延时等待
   TIMED_WAITING,
   // 终止!
   TERMINATED;
}

Java不能够创建线程!!!(线程是操作系统的资源)

线程的状态切换

技术图片

 

 

wait/sleep的区别

1、类不同!

wait——Object类 sleep——Thread类
在juc编程中,线程休眠怎么实现?
//时间单位
TimeUnit.SECONDS.sleep(3);
?

2、会不会释放资源

sleep:抱着锁睡得,不会释放锁;wait会释放锁。

3、使用的范围是不同的

wait和notify是一组,一般在线程通信的时候使用;

sleep就是一个单独的方法,在哪里都可以调用。

4、关于异常

sleep需要捕获异常

3.Lock锁


synchronized 传统的方式

代码:

package com.rudd.demo;
?
import java.util.concurrent.TimeUnit;
?
/**
* 传统的synchronized
* 企业级开发:
* 1.架构:高内聚,低耦合
* 2.套路:线程操作资源类,资源类是单独的。
*/
public class Test1 {
?
   public static void main(String[] args) {
       //1.新建资源类
       Ticket ticket = new Ticket();
       //2.线程操纵资源类
       new Thread(()->{
           for (int i = 1; i <41; i++) {
               ticket.saleTicket();
          }
      },"A").start();
       new Thread(()->{
           for (int i = 1; i <41; i++) {
               ticket.saleTicket();
          }
      },"B").start();
       new Thread(()->{
           for (int i = 1; i <41; i++) {
               ticket.saleTicket();
          }
      },"C").start();
  }
}
//单独的资源类应该只有:属性和方法
class Ticket{
?
   private int number = 30;
?
   //synchronized 关键字
   public synchronized void saleTicket(){
       if(number>0){
           System.out.println(Thread.currentThread().getName()+"卖出第"+(number--)+"票,还剩:"+number);
      }
  }
}

Lock锁

代码:

package com.rudd.demo;
?
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
?
public class Test02 {
?
   public static void main(String[] args) {
       //1.新建资源类
       Ticket2 ticket = new Ticket2();
       //2.线程操纵资源类,所有的函数式接口都可以用lambda表达式简化
       //lambda表达式:(参数)->{具体的代码}
       new Thread(()->{
           for (int i = 1; i <41; i++) {
               ticket.saleTicket();
          }
      },"A").start();
       new Thread(()->{
           for (int i = 1; i <41; i++) {
               ticket.saleTicket();
          }
      },"B").start();
       new Thread(()->{
           for (int i = 1; i <41; i++) {
               ticket.saleTicket();
          }
      },"C").start();
  }
}
?
class Ticket2{
   /*
    * 使用Lock,它是一个对象
    * ReentrantLock 可重入锁
    * ReentrantLock 默认是非公平锁
    * 非公平锁:不公平(插队,后面的线程可以插队)
    * 公平锁:公平(只能排队,后面的线程无法插队)
    */
   private Lock lock = new ReentrantLock();
   private int number = 30;
?
   public void saleTicket(){
       //加锁
       lock.lock();
       try {
           if(number>0){
               System.out.println(Thread.currentThread().getName()+"卖出第"+(number--)+"票,还剩:"+number);
          }
      }catch (Exception e){
           e.printStackTrace();
      }finally {
           //解锁
           lock.unlock();
      }
  }
}

synchronized和Lock的区别



  1. synchronized是一个关键字;Lock是一个对象


  2. synchronized无法尝试获取锁,Lock可以尝试获取锁,会进行判断;


  3. synchronized会自动释放锁(a线程执行完毕,b如果异常了,也会释放锁),Lock锁是手动释放锁,如果不释放就会死锁


  4. synchronizedsynchronizedsynchronized(线程A(获得锁,如果阻塞),线程B(等待,一直等待));Lock可以尝试获取锁,失败了之后就放弃。

    技术图片

     

     


  5. synchronized一定是非公平的锁,但是Lock锁可以是公平的,通过参数设置;


  6. 代码量特别大的时候,我们一般使用Lock实现精准控制,synchronized适合代码量较小的同步问题。



4.生产者和消费者

线程和线程之间本来是不能通信的,但是有时候我们需要线程之间可以协调操作:等待唤醒机制

synchronized

package com.rudd.demo;
/*
   synchronized
?
   目的:有两个线程:A B,还有一个初始值为0的变量
   实现两个线程交替执行,对该变量进行+1和-1操作,交替10次。
?
   传统的wait和notify方法不能实现精准唤醒。
*/
public class Test03 {
   public static void main(String[] args){
       Data data = new Data();
       //负责+1操作
       new Thread(()->{
           for (int i = 0; i <10; i++) {
               try {
                   data.increment();
              }catch (Exception e){
                   e.printStackTrace();
              }
          }
      },"A").start();
       //负责-1操作
       new Thread(()->{
           for (int i = 0; i <10; i++) {
               try {
                   data.decrement();
              }catch (Exception e){
                   e.printStackTrace();
              }
          }
      },"B").start();
       new Thread(()->{
           for (int i = 0; i <10; i++) {
               try {
                   data.increment();
              }catch (Exception e){
                   e.printStackTrace();
              }
          }
      },"C").start();
       new Thread(()->{
           for (int i = 0; i <10; i++) {
               try {
                   data.decrement();
              }catch (Exception e){
                   e.printStackTrace();
              }
          }
      },"D").start();
  }
}
?
//资源类
//线程之间的通信:判断   执行 通知
class Data{
   private int number = 0;
?
   public synchronized void increment() throws Exception{
       //不要用if,会导致需要唤醒
       while(number!=0){//1.判断是否需要等待
           this.wait();
      }
       number++;//2.执行
       System.out.println(Thread.currentThread().getName()+":::"+number);
       this.notifyAll();//3.通知:唤醒所有线程
  }
?
   public synchronized void decrement() throws Exception{
       //不要用if,会导致需要唤醒
       while(number!=1){//1.判断是否需要等待
           this.wait();
      }
       number--;//2.执行
       System.out.println(Thread.currentThread().getName()+":::"+number);
       this.notifyAll();//3.通知:唤醒所有线程
  }
}

用if判断可能会导致虚假唤醒

技术图片

 

 

Lock和Condition实现精准唤醒

传统的监视器:Object,JUC的监视器:Condition

技术图片

 

 

技术图片

 

 

package com.rudd.demo;
?
import sun.awt.windows.ThemeReader;
?
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
?
/*
   实现线程交替执行:
       主要的实现目标:精准的唤醒线程!
       三个线程:A,B,C
       三个方法:A p5,B p10,C p15一次循环。
?
*/
public class Test04 {
   public static void main(String[] args) {
       Data2 data = new Data2();
       new Thread(() -> {
           for (int i = 0; i <10; i++) {
               data.print5();
          }
      }, "A").start();
       new Thread(() -> {
           for (int i = 0; i <10; i++) {
               data.print10();
          }
      }, "B").start();
       new Thread(() -> {
           for (int i = 0; i <10; i++) {
               data.print15();
          }
      }, "C").start();
  }
}
?
?
class Data2 {
   Lock lock = new ReentrantLock();
   Condition condition1 = lock.newCondition();
   Condition condition2 = lock.newCondition();
   Condition condition3 = lock.newCondition();
   private int number = 1;//1-A线程;2—B线程;3—C线程
?
   public void print5() {
       lock.lock();
       try {
           //1.判断
           while (number != 1) {
               condition1.await();//等待
          }
           //2.执行
           for (int i = 0; i <5; i++) {
               System.out.println(Thread.currentThread().getName() + " " + i);
          }
           //3.通知第二个线程干活
           number = 2;
           condition2.signal();//唤醒
      } catch (Exception e) {
           e.printStackTrace();
      } finally {
           lock.unlock();
      }
  }
?
   public void print10() {
       lock.lock();
       try {
           //1.判断
           while (number != 2) {
               condition2.await();
          }
           //2.执行
           for (int i = 0; i <10; i++) {
               System.out.println(Thread.currentThread().getName() + " " + i);
          }
           //3.通知第三个线程干活
           number = 3;
           condition3.signal();
      } catch (Exception e) {
           e.printStackTrace();
      } finally {
           lock.unlock();
      }
  }
?
   public void print15() {
       lock.lock();
       try {
           //1.判断
           while (number != 3) {
               condition3.await();
          }
           //2.执行
           for (int i = 0; i <15; i++) {
               System.out.println(Thread.currentThread().getName() + " " + i);
          }
           //3.通知第一个线程干活
           number = 1;
           condition1.signal();
      } catch (Exception e) {
           e.printStackTrace();
      } finally {
           lock.unlock();
      }
  }
}

 

5.八锁现象,搞懂锁


synchronized实现同步的基础:

1、普通同步方法(实例方法),锁是当前实例对象 ,进入同步代码前要获得当前实例的锁

2、静态同步方法,锁是当前类的class对象 ,进入同步代码前要获得当前类对象的锁

3、同步代码块,锁是括号里面的对象,对给定对象加锁,进入同步代码块前要获得给定对象的锁。

 

6.集合类不安全


List

安全的List解决方案:

List<String> list1 = new Vector<String>();
List<String> list2 = Collections.synchronizedList(new ArrayList<String>());
/*
CopyOnWrite:写入时复制(COW思想)  
CopyOnWriteArrayList,是一个写入时复制的容器,它是如何工作的呢?简单来说,就是平时查询的时候,都不需要加锁,随便访问,只有在写入/删除的时候,才会从原来的数据复制一个副本出来,然后修改这个副本,最后把原数据替换成当前的副本。修改操作的同时,读操作不会被阻塞,而是继续读取旧的数据。
*/
List<String> list3 = new CopyOnWriteArrayList<String>();
//class CopyOnWriteArrayList;
public boolean add(E e) {
   final ReentrantLock lock = this.lock;
   lock.lock();
   try {
       Object[] elements = getArray();
       int len = elements.length;
       Object[] newElements = Arrays.copyOf(elements, len + 1);
       newElements[len] = e;
       setArray(newElements);
       return true;
  } finally {
       lock.unlock();
  }
}
?

Set

安全的Set解决方案:

Set set1 = Collections.synchronizedSet(new HashSet());
Set set2 = new CopyOnWriteArraySet();

HashSet的底层就是:HashMap

技术图片

 

Map

安全的Map解决方案:

Map<String,String> map1 = new Hashtable<String,String>();
       Map<String,String> map2 = Collections.synchronizedMap(new HashMap<String,String>());
       Map<String,String> map3 = new ConcurrentHashMap<String,String>();

HashMap的底层数据结构:链表+红黑树

 


推荐阅读
  • 开发日志:201521044091 《Java编程基础》第11周学习心得与总结
    开发日志:201521044091 《Java编程基础》第11周学习心得与总结 ... [详细]
  • 如何利用Java 5 Executor框架高效构建和管理线程池
    Java 5 引入了 Executor 框架,为开发人员提供了一种高效管理和构建线程池的方法。该框架通过将任务提交与任务执行分离,简化了多线程编程的复杂性。利用 Executor 框架,开发人员可以更灵活地控制线程的创建、分配和管理,从而提高服务器端应用的性能和响应能力。此外,该框架还提供了多种线程池实现,如固定线程池、缓存线程池和单线程池,以适应不同的应用场景和需求。 ... [详细]
  • 本文深入解析了Java面向对象编程的核心概念及其应用,重点探讨了面向对象的三大特性:封装、继承和多态。封装确保了数据的安全性和代码的可维护性;继承支持代码的重用和扩展;多态则增强了程序的灵活性和可扩展性。通过具体示例,文章详细阐述了这些特性在实际开发中的应用和优势。 ... [详细]
  • Java学习第10天:深入理解Map接口及其应用 ... [详细]
  • 线程能否先以安全方式获取对象,再进行非安全发布? ... [详细]
  • 分享一款基于Java开发的经典贪吃蛇游戏实现
    本文介绍了一款使用Java语言开发的经典贪吃蛇游戏的实现。游戏主要由两个核心类组成:`GameFrame` 和 `GamePanel`。`GameFrame` 类负责设置游戏窗口的标题、关闭按钮以及是否允许调整窗口大小,并初始化数据模型以支持绘制操作。`GamePanel` 类则负责管理游戏中的蛇和苹果的逻辑与渲染,确保游戏的流畅运行和良好的用户体验。 ... [详细]
  • Java排序算法详解:选择排序、插入排序、冒泡排序与递归实现
    本文详细解析了Java中的几种基础排序算法,包括选择排序、插入排序和冒泡排序,并探讨了递归在这些算法中的应用。选择排序通过每次找到未排序部分的最小值并将其置于已排序部分的末尾来实现;插入排序则通过逐步将每个元素插入到已排序序列的正确位置;而冒泡排序则是通过多次遍历数组,两两比较并交换相邻的元素,最终使较大的元素逐渐“冒”到数组末尾。文章还提供了具体的代码示例,帮助读者更好地理解和掌握这些算法的实现细节。 ... [详细]
  • 在本文中,我们将深入探讨C#中的构造函数及其应用场景。通过引入构造函数,可以有效解决在访问类属性时反复赋值导致的代码冗余问题,提高代码的可读性和维护性。此外,还将介绍构造函数的不同类型及其在实际开发中的最佳实践。 ... [详细]
  • 本文深入探讨了Java多线程环境下的同步机制及其应用,重点介绍了`synchronized`关键字的使用方法和原理。`synchronized`关键字主要用于确保多个线程在访问共享资源时的互斥性和原子性。通过具体示例,如在一个类中使用`synchronized`修饰方法,展示了如何实现线程安全的代码块。此外,文章还讨论了`ReentrantLock`等其他同步工具的优缺点,并提供了实际应用场景中的最佳实践。 ... [详细]
  • 在当前的软件开发领域,Lua 作为一种轻量级脚本语言,在 .NET 生态系统中的应用逐渐受到关注。本文探讨了 Lua 在 .NET 环境下的集成方法及其面临的挑战,包括性能优化、互操作性和生态支持等方面。尽管存在一定的技术障碍,但通过不断的学习和实践,开发者能够克服这些困难,拓展 Lua 在 .NET 中的应用场景。 ... [详细]
  • 本文介绍了如何利用ObjectMapper实现JSON与JavaBean之间的高效转换。ObjectMapper是Jackson库的核心组件,能够便捷地将Java对象序列化为JSON格式,并支持从JSON、XML以及文件等多种数据源反序列化为Java对象。此外,还探讨了在实际应用中如何优化转换性能,以提升系统整体效率。 ... [详细]
  • Java中不同类型的常量池(字符串常量池、Class常量池和运行时常量池)的对比与关联分析
    在研究Java虚拟机的过程中,笔者发现存在多种类型的常量池,包括字符串常量池、Class常量池和运行时常量池。通过查阅CSDN、博客园等相关资料,对这些常量池的特性、用途及其相互关系进行了详细探讨。本文将深入分析这三种常量池的差异与联系,帮助读者更好地理解Java虚拟机的内部机制。 ... [详细]
  • 在Java编程中,`AbstractClassTest.java` 文件详细解析了抽象类的使用方法。该文件通过导入 `java.util.*` 包中的 `Date` 和 `GregorianCalendar` 类,展示了如何在主方法 `main` 中实例化和操作抽象类。此外,还介绍了抽象类的基本概念及其在实际开发中的应用场景,帮助开发者更好地理解和运用抽象类的特性。 ... [详细]
  • 第六章:枚举类型与switch结构的应用分析
    第六章深入探讨了枚举类型与 `switch` 结构在编程中的应用。枚举类型(`enum`)是一种将一组相关常量组织在一起的数据类型,广泛存在于多种编程语言中。例如,在 Cocoa 框架中,处理文本对齐时常用 `NSTextAlignment` 枚举来表示不同的对齐方式。通过结合 `switch` 结构,可以更清晰、高效地实现基于枚举值的逻辑分支,提高代码的可读性和维护性。 ... [详细]
  • 在 Linux 环境下,多线程编程是实现高效并发处理的重要技术。本文通过具体的实战案例,详细分析了多线程编程的关键技术和常见问题。文章首先介绍了多线程的基本概念和创建方法,然后通过实例代码展示了如何使用 pthreads 库进行线程同步和通信。此外,还探讨了多线程程序中的性能优化技巧和调试方法,为开发者提供了宝贵的实践经验。 ... [详细]
author-avatar
mobiledu2502855247
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有