java同步机制:使用synchronize block,还是使用synchronize method

本文探讨了Java中同步代码块和同步方法在功能和性能上的区别。虽然两者都能确保原子性执行,但在字节码层面,同步方法的效率更高。通过分析javap命令输出的字节码,可以发现同步代码块的字节码数量多于同步方法。

今天在学习java原子类的时候,遇到了这篇博客,看到了同步代码块和同步方法的区别,之前没有意识到,这里记录下。

public class CP {

	private int i = 0;

	public synchronized int synchronizedMethodGet() {
		return i;
	}

	public int synchronizedBlockGet() {
		synchronized (this) {
			return i;
		}
	}
}

 

从功能角度来说,上面两种方式没有差别,都能保证方法执行时候的原子性。从性能上来看,同步方法比同步代码块更有优势。我们使用JDK提供的javap命令,查看生成的字节码。

E:\code_space\test\bin>javap -c CP
Compiled from "CP.java"
public class CP {
  public CP();
    Code:
       0: aload_0
       1: invokespecial #10                 // Method java/lang/Object."<init>":()V
       4: aload_0
       5: iconst_0
       6: putfield      #12                 // Field i:I
       9: return

  public synchronized int synchronizedMethodGet();
    Code:
       0: aload_0
       1: getfield      #12                 // Field i:I
       4: ireturn

  public int synchronizedBlockGet();
    Code:
       0: aload_0
       1: dup
       2: astore_1
       3: monitorenter
       4: aload_0
       5: getfield      #12                 // Field i:I
       8: aload_1
       9: monitorexit
      10: ireturn
      11: aload_1
      12: monitorexit
      13: athrow
    Exception table:
       from    to  target type
           4    10    11   any
          11    13    11   any
}

 

明显可以看到:使用同步代码块,产生的字节码数更多;而同步方法则很少。


 

 

Java 中,同步锁是一种用于控制多个线程对共享资源访问的机制。通过使用同步锁,可以确保在同一时刻只有一个线程能够访问特定的代码块或对象,从而避免数据竞争不一致的问题。 ### synchronized 关键字 最常用的同步锁是由 `synchronized` 关键字提供的。它可以作用于方法或者代码块上: - 当应用在一个实例方法前时,该方法变成对该对象加锁的方法。 - 若应用于静态方法,则整个类将被锁定。 - 对于代码段而言,开发者需指定某个对象作为监视器(lock),只有获得此对象锁才能继续执行内部内容。 #### 使用示例 这里有一个银行账户的例子来演示如何利用同步保护关键区域的数据一致性: ```java class BankAccount { private double balance; public BankAccount(double initialBalance){ this.balance=initialBalance; } // Synchronize whole method public synchronized void deposit(double amount){ if(amount >0 ){ balance +=amount; }else{ throw new IllegalArgumentException("Deposit must be positive"); } } // Or you can sync a specific part of code with block syntax /* * public void withdraw(double amount) { synchronized(this){ .... } }*/ } // Usage example in multithreading context... ``` 在这个例子当中,我们创建了一个包含余额属性以及存款操作的基本帐户模型,并通过对deposit函数添加了`sychronized`修饰符使得每一次仅允许单一线程修改balance变量值。这有助于保证并发场景下的事务完整性。 --- 另外值得注意的是,在JDK1.5以后引入了一些更先进的锁API位于java.util.concurrent.lock包内比如ReentrantLock等它们提供比内置monitor lock更多的灵活性包括尝试获取锁、设置公平策略等功能但是通常情况下简单地依靠synchronzied就足够满足大多数实际需求啦!
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值