Lock接口

Synchronized关键字回顾:

多线程编程步骤(上):

  1. 创建资源类,在资源类创建属性和操作方法
  2. 创建多个线程,调用资源类的操作方法

创建线程的四种方式:

  1. 继承Thread
  2. 实现Runnable接口
  3. 使用Callable接口
  4. 使用线程池

使用synchronized同步实现售票问题:

只有当资源是空闲的时候,线程才能访问。

 /**
  * 创建资源
  */
 class ticket{
     private  int number = 30;
     public synchronized void sell(){
         if(number >0){
             System.out.println(Thread.currentThread().getName()+":卖出"+number--+"剩余:"+number);
        }
 ​
    }
 }
 ​
 /**
  * 创建多个线程,调用资源类的操作方法
  */
 public class sellTicket {
     public static void main(String[] args) {
         /**
          * 创建资源
          */
         ticket ticket = new ticket();
         /**
          * 创建三个线程(售票员)
          */
         new Thread(new Runnable() {
             public void run() {
                 for (int i = 0; i < 40; i++) {
                     ticket.sell();
                }
            }
        }, "aaa").start();
         new Thread(new Runnable() {
             public void run() {
                 for (int i = 0; i < 40; i++) {
                     ticket.sell();
                }
            }
        }, "bbb").start();
         new Thread(new Runnable() {
             public void run() {
                 for (int i = 0; i < 40; i++) {
                     ticket.sell();
                }
            }
        }, "ccc").start();
    }
 }
 ​

学习《Java并发编程的艺术》一节是synchronized的实现原理与应用

Java中的每一个对象都可以作为锁:

  1. 对于同步方法,锁的是当前的对象。
  2. 对于静态方法,锁的是当前类的class对象。
  3. 对于静态代码块,Synchonized括号里配置的对象。

当一个线程试图访问同步代码块时,它首先必须得到锁,退出或抛出异常时必须释放锁。

那么锁到底存在哪里呢?锁里面会存储什么信息呢?

这个问题是该书中的一个问题,对个人来说有点抽象,主要点是:JVM基于进入和退出Monitor(监视器)对象来实现方法同步和代码块同步,但两者的实现细节不一样。

代码块同步是使用monitorenter 和monitorexit指令实现的,而方法同步是使用另外一种方式实现的,细节在JVM规范里并没有 详细说明。但是,方法的同步同样可以使用这两个指令来实现。

monitorenter指令是在编译后插入到同步代码块的开始位置,而monitorexit是插入到方法结 束处和异常处,JVM要保证每个monitorenter必须有对应的monitorexit与之配对。任何对象都有 一个monitor与之关联,当且一个monitor被持有后,它将处于锁定状态。线程执行到monitorenter 指令时,将会尝试获取对象所对应的monitor的所有权,即尝试获得对象的锁。

其中monitorenter主要是获取监视器锁,monitorexit主要是释放监视器锁。

synchonized中的一个概念是Java对象头:

synchronized用的锁是存放在Java对象头里面的。

非数组类型 数组类型
虚拟机 3个字宽 2个字宽

需要注意的是:在32位虚拟机中:1字宽=4字节,即:32bit

Java对象头里的Mark Word里面默认存储对象HashCode、分代年龄和锁标记位。

在运行期间,Mark Word里面存储的数据会随着锁标志位的变化而变化。

32位JVM的Mark Word的默认存储结构与在64位JVM不相同

Lock接口并创建实例:

Java并发编程的艺术

在JavaSE5之前使用的是synchronized关键字实现锁功能,5v之后并发包中新增Lock接口以及相关实现类用来实现锁功能,提供与synchronized类似的同步关系,但是比其更具有灵活性和扩展性(拥有了锁获取与释放的可操作性、可中断的获取锁以及超时获取锁)。

查看Lock的文档得知:

使用 lock 块来调用 try,在之前/之后的构造中,最典型的代码如下:

  class X {
    private final ReentrantLock lock = new ReentrantLock();
    // ...
 ​
    public void m() {
      lock.lock();  // block until condition holds
      try {
        // ... method body
      } finally {
        lock.unlock()
      }
    }
  }
 

根据上述的代码,实现买票代码:

 import java.util.concurrent.locks.ReentrantLock;
 class ticket2{
     private int number = 30;
 ​
     //创建一个重入锁
     private ReentrantLock lock = new ReentrantLock();
 ​
     public void sale(){
         //在内容之前加锁
         lock.lock();
         if(number >0){
             System.out.println(Thread.currentThread().getName()+":卖出"+number--+"剩余:"+number);
        }
         //在内容之后解锁
         lock.unlock();
    }
 }
 ​
 public class lockCase {
     public static void main(String[] args) {
         ticket2 tk = new ticket2();
         //创建线程
         new Thread(()->{
             for (int i = 0; i < 40; i++) {
                 tk.sale();
            }
        },"aaa").start();
         new Thread(()->{
             for (int i = 0; i < 40; i++) {
                 tk.sale();
            }
        },"bbb").start();
         new Thread(()->{
             for (int i = 0; i < 40; i++) {
                 tk.sale();
            }
        },"bbb").start();
    }
 }

上述代码中,在ticket2类中存在一个问题就是,当lock中间内容如果出现报错,那么后面的代码无法执行,也就是锁无法释放。所以我们需要将unlock()放到finally中。

目的是保证在获取到锁之后,最终能够被释放。

 public void sale(){
         //在内容之前加锁
         lock.lock();
         try{
             if(number >0){
                 System.out.println(Thread.currentThread().getName()+":卖出"+number--+"剩余:"+number);
            }
             //在内容之后解锁
        }finally{
             lock.unlock();
        }
 ​
    }

JUC之Lock接口以及Synchronized回顾的更多相关文章

  1. Java并发之synchronized关键字和Lock接口

    欢迎点赞阅读,一同学习交流,有疑问请留言 . GitHub上也有开源 JavaHouse,欢迎star 引用 当开发过程中,我们遇到并发问题.怎么解决? 一种解决方式,简单粗暴:上锁.将千军万马都给拦 ...

  2. synchronized关键字,Lock接口以及可重入锁ReentrantLock

    多线程环境下,必须考虑线程同步的问题,这是因为多个线程同时访问变量或者资源时会有线程争用,比如A线程读取了一个变量,B线程也读取了这个变量,然后他们同时对这个变量做了修改,写回到内存中,由于是同时做修 ...

  3. Java基础知识强化之多线程笔记06:Lock接口 (区别于Synchronized块)

    1. 简介 我们讲到了如何使用关键字synchronized来实现同步访问.本文我们继续来探讨这个问题,从Java 5之后,在java.util.concurrent.locks包下提供了另外一种方式 ...

  4. 多线程里面的关键字,wait, notfiy, 锁(synchronized), lock接口

    多线程环境下,必须考虑线程同步的问题,这是因为多个线程同时访问变量或者资源时会有线程争用,比如A线程读取了一个变量,B线程也读取了这个变量,然后他们同时对这个变量做了修改,写回到内存中,由于是同时做修 ...

  5. java多线程Lock接口简介使用与synchronized对比 多线程下篇(三)

    前面的介绍中,对于显式锁的概念进行了简单介绍 显式锁的概念,是基于JDK层面的实现,是接口,通过这个接口可以实现同步访问 而不同于synchronized关键字,他是Java的内置特性,是基于JVM的 ...

  6. 线程中synchronized关键字和lock接口的异同

    一.synchronized关键字 1.可以用来修饰代码块 synchronized (this) { // 同步的关键字 this 表示当前线程对象 if (num == 0) { break; } ...

  7. Synchronized和Lock接口

    关于synchronized字段,不管该关键字是修饰方法还是修饰同步代码块,synchronzed拿到的都是对象. 当synchronized修饰的是方法时,synchronized所拿到的是调用该方 ...

  8. JUC之Callable接口回顾和JUC辅助类

    Callable接口和JUC辅助类 Callable接口: 回顾: 创建线程的四种方式: 继承Thread 实现runnable接口 实现callable接口 使用线程池 之前的文章:多线程编程1-定 ...

  9. 5.Lock接口及其实现ReentrantLock

    jdk1.7.0_79 在java.util.concurrent.locks这个包中定义了和synchronized不一样的锁,重入锁——ReentrantLock,读写锁——ReadWriteLo ...

随机推荐

  1. [atAGC055B]ABC Supremacy

    将第$i$个字符在$A->C->B->A$这个环上操作$i$次,而此时的操作也即将$AAA,BBB$或$CCC$变为其中的另一个字符串 通过操作$XXXY->YYYY-> ...

  2. 一文带你吃透CLR垃圾回收机制

    前言 今天我们来共同学习一下CLR的垃圾回收机制,这对我们写出健壮性的代码很有帮助,也许有人会认为多此一举,认为垃圾回收交给CLR就行,我不用关心这个,诚然,大多数情况下是这样的,但是,我们今天讨论的 ...

  3. 规格模式(Specification Pattern)

    本文节选自<设计模式就该这样学> 1 规格模式的定义 规格模式(Specification Pattern)可以认为是组合模式的一种扩展.很多时候程序中的某些条件决定了业务逻辑,这些条件就 ...

  4. 第三方登陆-qq互联

    看到很多网站都有第三方登陆,使用业余时间自己也要实现一个第三方登陆的功能: 1.登陆qq互联的网站:https://connect.qq.com/index.html 2.点击头像进行资料申请 --- ...

  5. Pickle的简单用法

    Python中pickle的用法 pickle存在的意义 在python的文件操作里面,我们常常需要将python容器里面的一些东西把它写成一个二进制文件存放在硬盘里面来永久保存. 在不借助pickl ...

  6. Java计算器的简易实现(+-*/)

    java计算器的简易实现(+-*/) import java.util.Scanner; /* 写一个计算器 实现加减乘除四个功能 并且能够用循环接收新的数据,通过用户交互实现 写四个方法 + - * ...

  7. Promise(resolve,reject)的基本使用

    什么是Promise? Promise是一个构造函数,其原型上有 then.catch方法,还有reslove,reject等静态方法.通过创建Promise实例,可以调用Promise.protot ...

  8. ES2020新特性链操作符 '?.'和'??'

    ES2020新特性,js中的可选链操作符?. 概述 回想一下,我们是如何访问可能含有空值(null或undefined)属性的嵌套对象,比如访问web api 返回结果的user详情,可以使用嵌套的三 ...

  9. NOI2020 同步赛划水记

    因为太菜了没去现场参加 NOI 就算去了估计也只能混个Fe(雾) "两天都会各有一道签到题,争取拿到70分.剩下的题每道题打30分暴力.每天130分,就能稳拿Ag了."--ls D ...

  10. 贪心/构造/DP 杂题选做Ⅱ

    由于换了台电脑,而我的贪心 & 构造能力依然很拉跨,所以决定再开一个坑( 前传: 贪心/构造/DP 杂题选做 u1s1 我预感还有Ⅲ(欸,这不是我在多项式Ⅱ中说过的原话吗) 24. P5912 ...