让星星⭐月亮告诉你,Java synchronized(*.class) synchronized 方法 synchronized(this)分析

简介: 本文通过Java代码示例,介绍了`synchronized`关键字在类和实例方法上的使用。总结了三种情况:1) 类级别的锁,多个实例对象在同一时刻只能有一个获取锁;2) 实例方法级别的锁,多个实例对象可以同时执行;3) 同一实例对象的多个线程,同一时刻只能有一个线程执行同步方法。

⭐⭐⭐结论🌙🌙🌙:

  1. synchronized(SynchronizedTest.class)锁加在类上,若有多个类的实例对象,则同一时刻只能由一个类的实例对象(拥有t1的线程th1)获取到该类上的锁,其他类的实例对象(拥有t2的线程th2)需要等待
  2. synchronized void synchronizedMethod() 锁加在普通方法上,同一时刻多个实例对象访问到该方法时,每个实例对象都可以进行方法的执行
  3. synchronized void synchronizedMethod() 锁加在当前实例对象上,多个线程拥有同一个实例对象,同一时刻只能有一个拥有该实例对象的线程获得该方法锁进而执行方法,其他拥有该实例对象的线程需要等待

⭐⭐⭐代码🌙🌙🌙:

package unittest.thread;
public class SynchronizedTest {
   
 //锁住class对象
// public static  void synchronizedStatic(){
   
    public void synchronizedStatic(){
   
     synchronized(SynchronizedTest.class){
   
         for(int i = 0 ; i < 10 ; i ++){
   
             System.out.println(Thread.currentThread().getName() + i + ":synchronizedStatic");
             try {
   Thread.sleep(100);} catch (InterruptedException e) {
    e.printStackTrace();}
         }
     }
 }
 //锁住方法,lock标记打在该实例上
 public synchronized void synchronizedMethod(){
   
     for(int i = 0 ; i < 10 ; i ++){
   
         System.out.println(Thread.currentThread().getName() + i + ":synchronizedMethod");
         try {
   Thread.sleep(100);} catch (InterruptedException e) {
    e.printStackTrace();}
     }
 }
 //不会有影响,正常调用
 public  void synchronizedMethod2WithNosynchronized(){
   System.out.println("synchronizedMethod2WithNosynchronized"); }
 //synchronizedMethod 已经锁住实例,再加锁不成功
 public void synchronizedMethod2(){
   
     synchronized( this ){
   
         for(int i = 0 ; i < 10 ; i ++){
   
             System.out.println(Thread.currentThread().getName() + i + ":synchronizedMethod2");
             try {
   Thread.sleep(100);} catch (InterruptedException e) {
    e.printStackTrace();}
         }
     }
 }
 //synchronizedMethod 已经锁住实例, 再加锁不成功
 public void synchronizedMethod3(){
   synchronized( this ){
     System.out.println("synchronizedMethod3"); } }
 public static void main(String[] args) throws InterruptedException {
   
    final SynchronizedTest t= new SynchronizedTest();
    final SynchronizedTest t1= new SynchronizedTest();
    final SynchronizedTest t2= new SynchronizedTest();
//synchronized(SynchronizedTest.class)锁加在类上,若有多个类的实例对象,则同一时刻只能由一个类的实例对象(拥有t1的线程th1)获取到该类上的锁,其他类的实例对象(拥有t2的线程th2)需要等待
//Thread th1 = new Thread( new Runnable(){ @Override   public void run() {  SynchronizedTest.synchronizedStatic(); } } ,"线程th1-");  th1.start();
//Thread th2 = new Thread( new Runnable(){ @Override   public void run() {  SynchronizedTest.synchronizedStatic(); } } ,"线程th2-");  th2.start();
//Thread th1 = new Thread( new Runnable(){ @Override   public void run() {  t1.synchronizedStatic(); } } ,"线程th1-");  th1.start();
//Thread th2 = new Thread( new Runnable(){ @Override   public void run() {  t2.synchronizedStatic(); } } ,"线程th2-");  th2.start();

//synchronized void synchronizedMethod() 锁加在普通方法上,同一时刻多个实例对象访问到该方法时,每个实例对象都可以进行方法的执行
//Thread th3 = new Thread( new Runnable(){ @Override public void run() {t1.synchronizedMethod();} } ,"线程th3-"); th3.start();
//Thread th4 = new Thread( new Runnable(){ @Override public void run() { t2.synchronizedMethod();} } ,"线程th4-");  th4.start();
//synchronized void synchronizedMethod() 锁加在普通方法上,多个线程拥有同一个实例对象,同一时刻只能有一个拥有该实例对象的线程获得该方法锁进而执行方法,其他拥有该实例对象的线程需要等待
//Thread th3 = new Thread( new Runnable(){ @Override public void run() {t.synchronizedMethod();} } ,"线程th3-"); th3.start();
//Thread th4 = new Thread( new Runnable(){ @Override public void run() { t.synchronizedMethod();} } ,"线程th4-");  th4.start();

  //synchronized void synchronizedMethod() 锁加在当前实例对象上,多个线程拥有同一个实例对象,同一时刻只能有一个拥有该实例对象的线程获得该方法锁进而执行方法,其他拥有该实例对象的线程需要等待
//Thread th5 = new Thread( new Runnable(){ @Override public void run() { t.synchronizedMethod2();} } ,"线程th5-");  th5.start();
//Thread th6 = new Thread( new Runnable(){ @Override public void run() { t.synchronizedMethod2();} } ,"线程th6-");  th6.start(); 
  /*
  t.synchronizedMethod2WithNosynchronized();
  t.synchronizedMethod3();
  */
 }
}
AI 代码解读

```

目录
打赏
0
1
1
0
87
分享
相关文章
【原理】【Java并发】【synchronized】适合中学者体质的synchronized原理
本文深入解析了Java中`synchronized`关键字的底层原理,从代码块与方法修饰的区别到锁升级机制,内容详尽。通过`monitorenter`和`monitorexit`指令,阐述了`synchronized`实现原子性、有序性和可见性的原理。同时,详细分析了锁升级流程:无锁 → 偏向锁 → 轻量级锁 → 重量级锁,结合对象头`MarkWord`的变化,揭示JVM优化锁性能的策略。此外,还探讨了Monitor的内部结构及线程竞争锁的过程,并介绍了锁消除与锁粗化等优化手段。最后,结合实际案例,帮助读者全面理解`synchronized`在并发编程中的作用与细节。
48 8
【原理】【Java并发】【synchronized】适合中学者体质的synchronized原理
|
17天前
|
【Java并发】【synchronized】适合初学者体质入门的synchronized
欢迎来到我的Java线程同步入门指南!我不是外包员工,梦想是写高端CRUD。2025年我正在沉淀中,博客更新速度加快,欢迎点赞、收藏、关注。 本文介绍Java中的`synchronized`关键字,适合初学者。`synchronized`用于确保多个线程访问共享资源时不会发生冲突,避免竞态条件、保证内存可见性、防止原子性破坏及协调多线程有序访问。
51 8
【Java并发】【synchronized】适合初学者体质入门的synchronized
|
15天前
|
《从头开始学java,一天一个知识点》之:方法定义与参数传递机制
**你是否也经历过这些崩溃瞬间?** - 看了三天教程,连`i++`和`++i`的区别都说不清 - 面试时被追问&quot;`a==b`和`equals()`的区别&quot;,大脑突然空白 - 写出的代码总是莫名报NPE,却不知道问题出在哪个运算符 🚀 这个系列就是为你打造的Java「速效救心丸」!我们承诺:每天1分钟,地铁通勤、午休间隙即可完成学习;直击痛点,只讲高频考点和实际开发中的「坑位」;拒绝臃肿,没有冗长概念堆砌,每篇都有可运行的代码标本。上篇:《输入与输出:Scanner与System类》 | 下篇剧透:《方法重载与可变参数》。
46 25
|
9天前
|
重学Java基础篇—Java Object类常用方法深度解析
Java中,Object类作为所有类的超类,提供了多个核心方法以支持对象的基本行为。其中,`toString()`用于对象的字符串表示,重写时应包含关键信息;`equals()`与`hashCode()`需成对重写,确保对象等价判断的一致性;`getClass()`用于运行时类型识别;`clone()`实现对象复制,需区分浅拷贝与深拷贝;`wait()/notify()`支持线程协作。此外,`finalize()`已过时,建议使用更安全的资源管理方式。合理运用这些方法,并遵循最佳实践,可提升代码质量与健壮性。
20 1
Java中的异常处理方法
本文深入剖析Java异常处理机制,介绍可检查异常、运行时异常和错误的区别与处理方式。通过最佳实践方法,如使用合适的异常类型、声明精确异常、try-with-resources语句块、记录异常信息等,帮助开发者提高代码的可靠性、可读性和可维护性。良好的异常处理能保证程序稳定运行,避免资源泄漏和潜在问题。
|
23天前
|
Java代码结构解析:类、方法、主函数(1分钟解剖室)
### Java代码结构简介 掌握Java代码结构如同拥有程序世界的建筑蓝图,类、方法和主函数构成“黄金三角”。类是独立的容器,承载成员变量和方法;方法实现特定功能,参数控制输入环境;主函数是程序入口。常见错误包括类名与文件名不匹配、忘记static修饰符和花括号未闭合。通过实战案例学习电商系统、游戏角色控制和物联网设备监控,理解类的作用、方法类型和主函数任务,避免典型错误,逐步提升编程能力。 **脑图速记法**:类如太空站,方法即舱段;main是发射台,static不能换;文件名对仗,括号要成双;参数是坐标,void不返航。
45 5
JAVA方法的定义
JAVA方法的定义
124 0
杭州 【Java基础知识 11】java泛型方法的定义和使用(学习+改进+自己理解,想法) (借鉴-侵-删)
杭州 【Java基础知识 11】java泛型方法的定义和使用(学习+改进+自己理解,想法) (借鉴-侵-删)
68 1
|
10月前
|
Java数组与带参数方法:定义、调用及实践
Java数组与带参数方法:定义、调用及实践
103 1
|
10月前
|
Java中带返回值方法的定义与调用技术
Java中带返回值方法的定义与调用技术
138 1