多线程开发离不开锁机制,现在的Java语言中,提供了2种锁,一种是语言特性提供的内置锁,还有一种是 java.util.concurrent.locks 包中的锁,这篇文章简单整理一下内置锁的知识点。

内置锁在Java语言中的表现:

多线程的锁,其实本质上就是给一块内存空间的访问添加访问权限,因为Java中是没有办法直接对某一块内存进行操作的,又因为Java是面向对象的语言,一切皆对象,所以具体的表现就是某一个对象承担锁的功能,每一个对象都可以是一个锁。内置锁,使用方式就是使用 synchronized 关键字,synchronized 方法或者 synchronized 代码块。

每一种 synchronized 写法的锁是哪个对象:

1、指定当前对象加锁:

    private synchronized void function() {
//TODO execute something
}

2、指定当前类的Class对象加锁:

    private static synchronized void function() {
//TODO execute something
}

注意此处的 static 关键字。

3、指定任意对象加锁:

private void function() {
synchronized (object) {
//TODO execute something
}
}

此时,这段同步代码块的锁加在object对象上面。该对象可以是当前对象(object ==
this),也可以是当前类的Class对象(object == MyClass.class)。

简单验证一下:

现有如下的类:

public class SynchronizedTest {
private Object lock = new Object(); public void synchronizedBlockOnObject(long executeTime) {
synchronized (lock) {
System.out.println(Thread.currentThread().getName() + " -> start synchronizedBlockOnObject");
doSomething(executeTime);
System.out.println(Thread.currentThread().getName() + " -> end synchronizedBlockOnObject");
}
} public void synchronizedBlockOnThis(long executeTime) {
synchronized (this) {
System.out.println(Thread.currentThread().getName() + " -> start synchronizedBlockOnThis");
doSomething(executeTime);
System.out.println(Thread.currentThread().getName() + " -> end synchronizedBlockOnThis");
}
} public void synchronizedBlockOnClass(long executeTime) {
synchronized (SynchronizedTest.class) {
System.out.println(Thread.currentThread().getName() + " -> start synchronizedBlockOnClass");
doSomething(executeTime);
System.out.println(Thread.currentThread().getName() + " -> end synchronizedBlockOnClass");
}
} public synchronized void synchronizedMethodOnThis(long executeTime) {
System.out.println(Thread.currentThread().getName() + " -> start synchronizedMethodOnThis");
doSomething(executeTime);
System.out.println(Thread.currentThread().getName() + " -> end synchronizedMethodOnThis");
} public static synchronized void synchronizedMethodOnClass(long executeTime) {
System.out.println(Thread.currentThread().getName() + " -> start synchronizedMethodOnClass");
doSomething(executeTime);
System.out.println(Thread.currentThread().getName() + " -> end synchronizedMethodOnClass");
} private static void doSomething(long executeTime) {
try {
Thread.sleep(executeTime);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

1、static synchronized 方法 和 synchronized (MyClass.class) {} 同步代码块的锁都加在
MyClass.class 对象上面:

public static void main(String[] args) {
SynchronizedTest synchronizedTest = new SynchronizedTest(); new Thread(new Runnable() {
@Override
public void run() {
SynchronizedTest.synchronizedMethodOnClass(3000);
}
}, "Thread static synchronized method").start();
new Thread(new Runnable() {
@Override
public void run() {
synchronizedTest.synchronizedBlockOnClass(2000);
}
}, "Thread synchronized block on Class").start();
}

运行结果如下:

Thread static synchronized method -> start synchronizedMethodOnClass
Thread static synchronized method -> end synchronizedMethodOnClass
Thread synchronized block on Class -> start synchronizedBlockOnClass
Thread synchronized block on Class -> end synchronizedBlockOnClass

说明当线程 Thread static synchronized method 进入方法 synchronizedMethodOnClass
的时候,线程Thread synchronized block on Class 是不能进入synchronizedBlockOnClass 代码块的。

2、非 static 的 synchronized 方法和 synchronized (this) {} 同步代码块的锁都加在当前对象上面:

public static void main(String[] args) {
SynchronizedTest synchronizedTest = new SynchronizedTest(); new Thread(new Runnable() {
@Override
public void run() {
synchronizedTest.synchronizedMethodOnThis(3000);
}
}, "Thread non-static synchronized method").start();
new Thread(new Runnable() {
@Override
public void run() {
synchronizedTest.synchronizedBlockOnThis(2000);
}
}, "Thread synchronized block on this").start();
}

运行结果如下:

Thread non-static synchronized method -> start synchronizedMethodOnThis
Thread non-static synchronized method -> end synchronizedMethodOnThis
Thread synchronized block on this -> start synchronizedBlockOnThis
Thread synchronized block on this -> end synchronizedBlockOnThis

说明当线程 Thread non-static synchronized method 进入方法 synchronizedMethodOnThis
的时候,线程Thread synchronized block on this 是不能进入synchronizedBlockOnThis 代码块的。

3、当锁加在 MyClass.class 、 this 、 任意对象,这三种情况,起不到任何同步作用:

    public static void main(String[] args) {
SynchronizedTest synchronizedTest = new SynchronizedTest(); new Thread(new Runnable() {
@Override
public void run() {
synchronizedTest.synchronizedMethodOnThis(3000);
}
}, "Thread non-static synchronized method").start();
new Thread(new Runnable() {
@Override
public void run() {
SynchronizedTest.synchronizedMethodOnClass(2000);
}
}, "Thread static sybchronized method").start();
new Thread(new Runnable() {
@Override
public void run() {
synchronizedTest.synchronizedBlockOnObject(4000);
}
}, "Thread sybchronized block on other Object").start();
}

运行结果如下:

Thread non-static synchronized method -> start synchronizedMethodOnThis
Thread static sybchronized method -> start synchronizedMethodOnClass
Thread sybchronized block on other Object -> start synchronizedBlockOnObject
Thread static sybchronized method -> end synchronizedMethodOnClass
Thread non-static synchronized method -> end synchronizedMethodOnThis
Thread sybchronized block on other Object -> end synchronizedBlockOnObject

说明当锁没有加在同一个对象上的时候,起不到线程间的同步作用。

Object中对内置锁进行操作的一些方法:

wait()系列:

wait()系列方法的作用是:使当前已经获得该对象锁的线程进入等待状态,并且释放该对象的锁。

notify()系列:

notify()系列方法的作用是:唤醒那些正在等待该对象锁的线程,使其继续运行。

基于wait() notify()机制,我们可以实现一个简易的生产者-消费者模型。

大体思路如下,一个生产者线程负责向一个仓库中存放(put)物品,一个消费者线程负责从仓库中取出(get)物品。

代码如下:

public class Warehouse {

    private Queue<Integer> queue;
private int capacity; public Warehouse(int capacity) {
this.capacity = capacity;
queue = new LinkedList();
} public synchronized void put(int num) {
if (queue.size() >= capacity) {
try {
System.out.println(Thread.currentThread().getName() + " , put full wait");
wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
queue.add(num);
System.out.println(Thread.currentThread().getName() + " , put : " + num + " , queue -> " + queue);
notifyAll();
} public synchronized int get() {
if (queue.isEmpty()) {
try {
System.out.println(Thread.currentThread().getName() + " , get empty wait");
wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
int num = queue.poll();
System.out.println(Thread.currentThread().getName() + " , get : " + num + " , queue -> " + queue);
notifyAll();
return num;
}
}
    public static void main(String[] args) {
Warehouse warehouse = new Warehouse(4);
Random random = new Random(); new Thread(new Runnable() {
@Override
public void run() {
while (true) {
warehouse.put(random.nextInt(10));
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}, "生产者-01").start(); new Thread(new Runnable() {
@Override
public void run() {
while (true) {
warehouse.get();
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}, "消费者-01").start();
}

运行结果如下:

生产者-01 , put : 5  , queue -> [5]
消费者-01 , get : 5 , queue -> []
生产者-01 , put : 7 , queue -> [7]
消费者-01 , get : 7 , queue -> []
生产者-01 , put : 9 , queue -> [9]
生产者-01 , put : 7 , queue -> [9, 7]
消费者-01 , get : 9 , queue -> [7]
生产者-01 , put : 0 , queue -> [7, 0]
生产者-01 , put : 5 , queue -> [7, 0, 5]
消费者-01 , get : 7 , queue -> [0, 5]
生产者-01 , put : 9 , queue -> [0, 5, 9]
生产者-01 , put : 6 , queue -> [0, 5, 9, 6]
消费者-01 , get : 0 , queue -> [5, 9, 6]
生产者-01 , put : 4 , queue -> [5, 9, 6, 4]
生产者-01 , put full wait
消费者-01 , get : 5 , queue -> [9, 6, 4]
生产者-01 , put : 6 , queue -> [9, 6, 4, 6]
生产者-01 , put full wait
消费者-01 , get : 9 , queue -> [6, 4, 6]
生产者-01 , put : 2 , queue -> [6, 4, 6, 2]
生产者-01 , put full wait
消费者-01 , get : 6 , queue -> [4, 6, 2]
生产者-01 , put : 9 , queue -> [4, 6, 2, 9]
生产者-01 , put full wait
消费者-01 , get : 4 , queue -> [6, 2, 9]
生产者-01 , put : 7 , queue -> [6, 2, 9, 7]
生产者-01 , put full wait
消费者-01 , get : 6 , queue -> [2, 9, 7]
生产者-01 , put : 2 , queue -> [2, 9, 7, 2]

Java内置锁的简单认识的更多相关文章

  1. Java内置锁和简单用法

    一.简单的锁知识 关于内置锁 Java具有通过synchronized关键字实现的内置锁,内置锁获得锁和释放锁是隐式的,进入synchronized修饰的代码就获得锁,走出相应的代码就释放锁. jav ...

  2. 深入理解Java内置锁和显式锁

    synchronized and Reentrantlock 多线程编程中,当代码需要同步时我们会用到锁.Java为我们提供了内置锁(synchronized)和显式锁(ReentrantLock)两 ...

  3. Java内置锁synchronized的实现原理

    简述Java中每个对象都可以用来实现一个同步的锁,这些锁被称为内置锁(Intrinsic Lock)或监视器锁(Monitor Lock). 具体表现形式如下: 1.普通同步方法,锁的是当前实例对象 ...

  4. 深入理解java内置锁(synchronized)和显式锁(ReentrantLock)

    多线程编程中,当代码需要同步时我们会用到锁.Java为我们提供了内置锁(synchronized)和显式锁(ReentrantLock)两种同步方式.显式锁是JDK1.5引入的,这两种锁有什么异同呢? ...

  5. Java内置锁synchronized的实现原理及应用(三)

    简述 Java中每个对象都可以用来实现一个同步的锁,这些锁被称为内置锁(Intrinsic Lock)或监视器锁(Monitor Lock). 具体表现形式如下: 1.普通同步方法,锁的是当前实例对象 ...

  6. Java 内置锁 重入问题

    阅读<Java并发编程实战>,遇到了一个问题 代码如下 /** * @auther draymonder */ public class Widget { public synchroni ...

  7. java内置锁实现锁住代码块方案(同一个对象或锁住整个类.class)

    我们看一个例子: class Demo { public synchronized void test() { System.out.println("test方法开始执行,当前线程为:&q ...

  8. Java内置锁synchronized的可重入性

    学习自 https://blog.csdn.net/aigoogle/article/details/29893667 对我很有帮助 感谢作者

  9. 七 内置锁 wait notify notifyall; 显示锁 ReentrantLock

    Object中对内置锁进行操作的一些方法: Java内置锁通过synchronized关键字使用,使用其修饰方法或者代码块,就能保证方法或者代码块以同步方式执行. 内置锁使用起来非常方便,不需要显式的 ...

随机推荐

  1. 《自拍教程6》打开Windows文件后缀

    如果你用的是Windows操作系统, 请把文件后缀名打开,千万别隐藏后缀名, 后续有各类 .py, .sh, .bat, .exe等不同文件, 需要你时刻关注文件后缀名. 如果是我看到我手下的测试人员 ...

  2. ELK 记录 java log4j 类型日志

    ELK 记载  java log4j 时,一个报错会生成很多行,阅读起来很不方便. 类似这样 解决这个问题的方法 1.使用多行合并 合并多行数据(Multiline) 有些时候,应用程序调试日志会包含 ...

  3. c++ bool

    bool 就两个值,真或者假,通常用来存储关系表达式或者逻辑表达式的结果. 以前是用 int 来表示真假,大 int 有多个值,所以才规定 0 为假,非零为真,导致对应关系比较麻烦,有了 bool 就 ...

  4. 08.JS单词整理

    以下为按照文章顺序简单整理的JS单词, 注意:是JS单词注释,部分与英文不符 01.JS语法规范.变量与常量 console——控制台 log——日志 var——变量 variable变量,变化 co ...

  5. 网页延迟加载动画的实现-WOW.js

    网页内容一开始不显示,随着鼠标下拉延迟显示,还有时间差 一开始觉得好难好复杂好高大上,直到我发现 wow.js …… 首先是演示地址:https://www.delac.io/wow/ 嗯,狗子确实很 ...

  6. HTML连载70-相片墙、盒子阴影和文字阴影

    一. 制作一个相片墙 二. <!DOCTYPE html> <html lang="en"> <head> <meta charset=& ...

  7. 获取Data和Log默认路径

    使用SERVERPROPERTY()来得到Data和Log的默认路径: InstanceDefaultDataPath和InstanceDefaultLogPath分别返回默认数据和日志目录. DEC ...

  8. 从零开始一个个人博客 by asp.net core and angular(一)

    这是一个个人叙述自己建设博客的帖子,既然是第一篇那肯定是不牵扯代码了,主要讲一下大体的东西,微软最新的web框架应该就数asp.net core 3.1了这是一个长期支持版,而且是跨平台又开源版本,所 ...

  9. 关于js获取元素在屏幕中的位置的方法

    针对我们获取元素在页面中的位置的问题,我们还是用老师一峰老师的方法来解决吧 下面上HTML代码 <div class="left_footer"> <p data ...

  10. R语言入门:向量初探

    R语言主要用于统计,因此引入了向量这个概念将更好地进行统计计算,在其他无法引入向量的语言当中则会使用循环来计算一些大规模的数据,在R语言当中则不需要,下面我们来看看R语言当中向量的具体用法吧! 首先, ...