Java的synchronized加在方法上或者对象上有什么区别?

Java的synchronized可以加在方法上,也可以直接加在对象上,从而保证一段代码只能有一个线程在运行,保证线程的同步。

那么这两者究竟有啥区别呢?我们可以看下面的示例代码。

public class SyncTest {

	public static synchronized void testSyncOnStaticMethod() {
		System.out.println("testSyncOnStaticMethod");
		try {
			Thread.sleep(10000);
		} catch (InterruptedException e) {
		}
	}

	public static void testSyncOnClass() {
		synchronized (SyncTest.class) {
			System.out.println("testSyncOnClass");
			try {
				Thread.sleep(10000);
			} catch (InterruptedException e) {
			}
		}
	}

	public synchronized void testSyncOnMethod() {
		System.out.println("testSyncOnMethod");
		try {
			Thread.sleep(10000);
		} catch (InterruptedException e) {
		}
	}

	public void testSyncOnThis() {
		synchronized (this) {
			System.out.println("testSyncOnThis");
			try {
				Thread.sleep(10000);
			} catch (InterruptedException e) {
			}
		}
	}

	public static void case1() {
		// case1
		// 先输出testSyncOnThis或者testSyncOnMethod
		// 然后停顿10秒,再输出另一个
		// 这个现象表明了
		
		// public synchronized void func() {
		// }
		
		// 等价于
		
		// public void func() {
		//     synchronized (this) {
		//     }
		// }
		final SyncTest t1 = new SyncTest();
		new Thread(new Runnable() {

			@Override
			public void run() {
				t1.testSyncOnThis();
			}
		}).start();

		new Thread(new Runnable() {

			@Override
			public void run() {
				t1.testSyncOnMethod();
			}
		}).start();
	}

	public static void case2() {
		// case2
		// 先输出testSyncOnClass或者testSyncOnStaticMethod
		// 然后停顿10秒,再输出另一个
		// 这个现象表明了
		
		// public synchronized static void staticFunc() {
		// }
		
		// 等价于
		
		// public static void staticFunc() {
		//     synchronized (SyncTest.class) {
		//     }
		// }
		new Thread(new Runnable() {

			@Override
			public void run() {
				SyncTest.testSyncOnClass();
			}
		}).start();

		new Thread(new Runnable() {

			@Override
			public void run() {
				SyncTest.testSyncOnStaticMethod();
			}
		}).start();
	}

	public static void main(String[] args) {
		case1();
		case2();
	}
}

从上面的代码我们可以看出synchronized加在方法上本质上还是等价于加在对象上的。

如果synchronized加在一个类的普通方法上,那么相当于synchronized(this)。

如果synchronized加载一个类的静态方法上,那么相当于synchronized(Class对象)。

在使用多线程的时候,知道这个是很关键的,因为synchronized的两种不用用法可能导致两段不相干的代码是互斥的,增加了同步的开销(例如这里的函数testSyncOnThis和testSyncOnMethod,他们在同一个对象this上加了锁),更严重的是可能导致死锁。

注:如果要试验,上面的case1和case2请分开运行(可以跑两次,每次注释掉其中一个语句),这样子可以看得比较清楚。

  • 9
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
`synchronized (lock)`和`synchronized`方法Java中的使用有明显的区别。下面我们来详细分析: 1. `synchronized (lock)`:这是一种更广义的synchronized(同步)关键字,它可以与一个特定的对象或者变量相关联,用作一种锁定机制,用于确保多个线程在同一时间只执行某一代码段的一部分。这允许你在对象级别实现线程同步。这种形式的同步通常与对象锁(Object Lock)相关联,它是Java内存模型的一部分,确保了不同线程之间的内存一致性。 2. `synchronized`方法:`synchronized`方法Java中的一个特定语法,它与一个特定的方法相关联,用于同步方法调用。当一个线程正在执行一个同步方法时,其他线程必须等待直到该方法完成执行并释放锁,然后才能开始执行。这就确保了同一时间只有一个线程可以访问被同步的方法。`synchronized`方法适用于同步块,它可以包含任意数量的代码块,只需在块前面加上`synchronized`关键字即可。 两者主要的区别在于使用的上下文和灵活性。`synchronized`方法是一种更直接的方法,因为它直接针对一个特定的方法进行同步。然而,如果需要在更大的代码块(例如整个类或整个应用)上实现同步,`synchronized (lock)`更适用。而且,对于小的同步需求,`synchronized`方法的语法可能看起来过于复杂和冗余。 值得注意的是,无论是使用`synchronized`还是`synchronized (lock)`关键字,它们的主要目的是控制并发性以防止数据冲突和损坏。这通过防止多个线程同时执行相同的代码段来实现,从而避免了数据的不一致性。然而,过度使用同步可能会降低程序的性能,因此需要谨慎使用。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值