我们看一个例子:

class Demo {
public synchronized void test() {
System.out.println("test方法开始执行,当前线程为:"+Thread.currentThread().getName());
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println("test方法执行完毕,当前线程为:"+Thread.currentThread().getName());
}
}

class MyThread implements Runnable {

@Override
public void run() {
Demo demo = new Demo();
demo.test();
}

}
public class Test {
public static void main(String[] args) {
MyThread myThread = new MyThread();
new Thread(myThread,"子线程A").start();
new Thread(myThread,"子线程B").start();
new Thread(myThread,"子线程C").start();
}
}

  运行结果:

  从运行结果我们可以看出,Demo类提供的test同步方法好像并没有起作用,这是怎么一回事。

  实际上,synchronized(this) 以及非 static 的 synchronized 方法,只能防止多个线程同时执行同一个对象的同步代码块。即 synchronized 锁住的是括号里的对象,而不是代码块

  所以说 synchronized 是一个对象锁。

  当 synchronized 锁住一个对象后,别的线程如果也想拿到这个对象的锁,就必须等待这个线程执行完成释放锁,才能再次给对象加锁,这样才能达到线程同步的目的。所以即使两个不同的代码块都要锁住同一个对象,那么这两个代码段也不能在多线程环境下同时运行,必须等其中一个现将对象锁释放掉,另一个才能给对象上锁。

  所以在上例中,MyThread线程类启动三次也创建了三个Demo类,并且对其调用,三个不同的对象进入了同步方法中,所以显示如上结果。

  当一个线程A 进入到同步方法所在的类中,其他线程不能进入该类中的其他类中,因为锁住的是对象。类比:厕所里有个电视机,某人上厕所时关上了锁,其他人也不能进来看电视。

  那我们如果想将一段代码锁住,使同时有且只有一个对象能访问该代码块应该如何操作。

  这种锁住代码块的的操作叫做全局锁,可以通过以下两种途径实现:

1.1 锁住同一个对象
class Demo {
public void test() {
// 锁住进入的方法的对象
synchronized(this) {
System.out.println("test方法开始执行,当前线程为:"+Thread.currentThread().getName());
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println("test方法执行完毕,当前线程为:"+Thread.currentThread().getName());
}
}
}

class MyThread implements Runnable {
// 为了防止多个线程创建多个对象,所以在类中自己创建一个对象
private Demo demo;
// 在构造方MyThread时将真正的对象传入
public MyThread(Demo demo) {
this.demo = demo;
}

@Override
public void run() {
this.demo.test();
}

}
public class Test {
public static void main(String[] args) {
// 实际上,整个程序只有这一个对象
// 锁住了该对象就相当于将 Demo类中的test方法代码锁住了,曲线救国实现全局锁
Demo demo = new Demo();
MyThread myThread = new MyThread(demo);
new Thread(myThread,"子线程A").start();
new Thread(myThread,"子线程B").start();
new Thread(myThread,"子线程C").start();
}
}

1.2 锁住整个类
class Demo {
public void test() {
// 将 Demo类 作为锁定的对象,每次只能有一个对象进入该类
synchronized(Demo.class) {
System.out.println("test方法开始执行,当前线程为:"+Thread.currentThread().getName());
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println("test方法执行完毕,当前线程为:"+Thread.currentThread().getName());
}
}
}

class MyThread implements Runnable {

@Override
public void run() {
// 虽然这里还是存在创建多个对象的问题
// 但是由于test方法这次锁住了整个类,所以同时有且仅有一个对象能够进入Demo类中
Demo demo = new Demo();
demo.test();
}

}
public class Test {
public static void main(String[] args) {
MyThread myThread = new MyThread();
new Thread(myThread,"子线程A").start();
new Thread(myThread,"子线程B").start();
new Thread(myThread,"子线程C").start();
}
}

  

  当然,使用静态同步方法也可以实现锁住整个类的效果。

public static synchronized test() {
// statement
}
————————————————
原文链接:https://blog.csdn.net/weixin_40739833/article/details/80293480

java内置锁实现锁住代码块方案(同一个对象或锁住整个类.class)的更多相关文章

  1. JAVA内置注解 基本注解

    温故而知新,可以为师矣! 每天复习,或者学习一点小东西,也能水滴石穿! 今天复习5个JAVA内置基本注解(贴代码胜过千言万语): package com.lf.test; import java.ut ...

  2. Java基础之线程——管理线程同步代码块(BankOperation4)

    控制台程序. 除了同步类对象的方法之外,还可以把程序中的语句或代码块制定为synchronized,这种方式更强大,因为可以指定哪个对象从语句或代码块的同步中获益,而不像同步方法那样仅仅是包含代码的对 ...

  3. Android(java)学习笔记68:同步代码块 和 同步方法 的应用

    1. 同步代码块 和 同步方法 代码示例: (1)目标类,如下: package cn.himi.text; public class SellTicket implements Runnable { ...

  4. Android(java)学习笔记8:同步代码块 和 同步方法 的应用

    1. 同步代码块 和 同步方法 代码示例: (1)目标类,如下: package cn.himi.text; public class SellTicket implements Runnable { ...

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

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

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

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

  7. Java内置锁和简单用法

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

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

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

  9. Java 性能分析工具 , 第 2 部分:Java 内置监控工具

    引言 本文为 Java 性能分析工具系列文章第二篇,第一篇:操作系统工具.在本文中将介绍如何使用 Java 内置监控工具更加深入的了解 Java 应用程序和 JVM 本身.在 JDK 中有许多内置的工 ...

随机推荐

  1. Sql注入--数字型手工测试

    Sql注入--数字型手工测试 漏洞原因:是在数据交互中,前端的数据传入到后台处理时,没有做严格的判断,导致其传入的"数据"拼接到SQL语句中后,被当作SQL语句的一部分执行. 从而 ...

  2. DDD(领域驱动设计)--战术设计

    前言 战术设计 战略设计为我们提供一种高层视野来审视我们的软件系统,主要包括领域/子域.通用语言.限界上下文和架构风格等概念, 而战术设计则将战略设计进行具体化和细节化,它主要关注的是技术层面的实施, ...

  3. PyQt(Python+Qt)学习随笔:toolButton的toolButtonStyle属性

    toolButtonStyle属性用于确认toolButton按钮显示文字.图标的方式,其类型为枚举类型 Qt.ToolButtonStyle,有如下值: ToolButtonIconOnly(值为0 ...

  4. 【题解】P3629 [APIO2010]巡逻

    link 题意 有 \(n\) 个村庄,编号为 \(1, 2, ..., n\) .有 \(n – 1\) 条道路连接着这些村 庄,从任何一个村庄都可以到达其他任一个村庄.道路长度均为 1. 巡警车每 ...

  5. 【题解】「UVA1149」装箱 Bin Packing

    做法显然:贪心,但是怎么贪? 首先从大到小或从小到大排序,窝这次是从大到小排,这样更容易理解(从小到大更方变) 然后设置两个变量 front 和 after 作为前指针和后指针. 循环判断: 当前后两 ...

  6. Linux 上安装 mysql

    1.通过 yum 命令安装 mysql 可以先通过 yum list |grep mysql 方式查看有哪些版本的 mysql 2.安装 mysql yum install mysql-server ...

  7. shell 编程 -- 条件判断

    1.按照文件类型进行判断(常用的)-b 判断该文件是否存在-d 判断是否存在,并且是否为目录(是目录就为真)-e 判断该文件是否存在(存在为真)-f 判断文件是否存在,并且是否为普通文件(是普通文件为 ...

  8. HDFS 操作命令

    本地文件系统创建一个文件,输入带姓名学号信息. 查看新建文件所在目录 查看新建文件内容 将此文件上传到HDFS文件系统上的用户目录 HDFS上查看相应目录 HDFS上查看文件内容 HDFS上查看文件大 ...

  9. .NET5下的三维应用程序开发

    终于等到了.NET5的发布,怀着激动的心情体验了一下:"香"就一个字. 如何基于.NET5开发工业软件,也广大三维应用开发者关心的问题.我们的Rapid SDK已经率先支持.NET ...

  10. antDesign中排序sorter的坑

    antd中sorter是写在columns中的一个配置,结果为一个回调函数 如图,这是我项目中使用sorter的小例子,参数a,b分别为列表第0项数据和第1项数据.回调函数中return一个值,按照什 ...