lock和synchronized的区别以及优点缺点

1.synchronized和lock 都有可重入性.

2.synchronized和lock 都可以知道线程是否持有了锁(看到很多文章说只有lock可以判断是否获取到锁,其实是错误的,synchronized也可以获取. synchronized 用Thread.holdsLock来判断).

3.synchronized在进入锁池后,会不停得等待直到获取锁对象,无法中断,是真爱.而lock可以设置等待时间停止锁竞争.

public boolean tryLock(long timeout, TimeUnit unit)
        throws InterruptedException {
    return sync.tryAcquireNanos(1, unit.toNanos(timeout));
}

还可以用lock.lockInterruptibly()方法:当无法获取到锁的时候,直接中断,进行异常处理.

4.synchronized是内置的,而lock是一个接口.

5.synchronized在运行结束后或者异常后会自动释放锁,而lock需要自己手动释放锁
lock.unlock();

6.synchronized是非公平锁,而lock可以是非公平锁也可以是公平锁.用布尔值来选择类型.

public ReentrantLock(boolean fair) {
    sync = fair ? new FairSync() : new NonfairSync();
}

7.lock还有读写锁,
读-读不互斥,
读-写互斥,
写-写互斥.
写锁还可以不用unlock降级为读锁

知识点: 读写锁经典案例,防缓存系统

import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.locks.ReentrantReadWriteLock;


public class ReadWriteLockTest {
	
	ReentrantReadWriteLock rwl = new ReentrantReadWriteLock();
	
	Map<String,Object> cache = new HashMap<String,Object>();
	
	public Object get(String key){
		Object value  = null;
		try {
			rwl.readLock().lock();
			value = cache.get(key);
			if(null==value){
				rwl.readLock().unlock();//释放都锁,获取写锁
				try {
					rwl.writeLock().lock();
					//获取写锁后再次判断对象是否为null,方式下一个等待的写线程进入后直接获取数据去
					value = cache.get(key);
					if(null==value){
						System.out.println(Thread.currentThread().getName());
						value="aaaaa";//实际操作代码从数据库中查询得到的对象
						cache.put(key, value);
					}
					//自身锁降级为都锁
					rwl.readLock().lock();
				} catch (Exception e) {
					// TODO: handle exception
				}finally{
					rwl.writeLock().unlock();//释放写锁
				}
			}
			
		} catch (Exception e) {
			// TODO: handle exception
		}finally{
			rwl.readLock().unlock();
		}
		return value;
	}
	
}

import java.util.HashMap;
import java.util.Map;


public class MainReadWritLockTest {

	Map<String,Object> cache = new HashMap<String,Object>();
	/**
	 * @param args
	 */
	public static void main(String[] args) {
		// TODO Auto-generated method stub
		
		final ReadWriteLockTest rwlt = new ReadWriteLockTest();
		for(int i=0; i<10;i++){
			new Thread(new Runnable() {
				@Override
				public void run() {
					// TODO Auto-generated method stub
					Object value= rwlt.get("id");
					System.out.println(Thread.currentThread().getName()+"--"+value);
				}
			}).start();
		}
		
	}

}

参考文章:https://blog.csdn.net/cdw8131197/article/details/52601559?utm_medium=distribute.pc_relevant.none-task-blog-2%7Edefault%7EBlogCommendFromMachineLearnPai2%7Edefault-1.control&depth_1-utm_source=distribute.pc_relevant.none-task-blog-2%7Edefault%7EBlogCommendFromMachineLearnPai2%7Edefault-1.control

### 回答1: synchronized和ReentrantLock都是Java中的线程同步机制。它们的作用都是为了保证线程在访问共享资源时的互斥性,避免出现线程安全问题。 synchronized是Java内置的关键字,使用起来比较简单,可以直接用于方法或代码块的同步。synchronized优点是使用方便,缺点是性能较低,无法进行细粒度的控制。 ReentrantLock是Java中的一个可重入锁,它可以实现更细粒度的控制,提供了很多灵活的功能,比如可以支持公平锁和非公平锁、可以设置超时时间等。ReentrantLock优点是灵活性高,可以进行更细粒度的控制,缺点是使用稍微复杂一些。 总的来说,如果只是简单的同步操作,使用synchronized就足够了。如果需要更细粒度的控制,或者需要一些高级功能,可以考虑使用ReentrantLock。 ### 回答2: synchronized和reentrantlock都是Java中用于实现线程同步的机制。 synchronized是一种内置的关键字,用于修饰方法或代码块。使用synchronized修饰的方法或代码块在同一时间只能被一个线程执行,其他线程需要等待当前线程执行完毕后才能继续执行。synchronized具有自动释放锁的功能,当线程执行完毕或出现异常时,会自动释放当前线程所持有的锁。synchronized简单易用,但只能实现基本的线程同步,无法实现更复杂的同步需求。 reentrantlock是Java中的一个类,可以通过实例化一个reentrantlock对象来使用。reentrantlock具备比synchronized更高的灵活性和扩展性。它提供了多种高级特性,例如可重入性、可中断的锁、公平性、条件变量等。reentrantlock需要手动锁定和解锁,所以需要显示地在代码中编写lock()和unlock()方法来控制锁的获取和释放。相比之下,reentrantlock更适合复杂的同步问题,因为它可以在代码执行过程中灵活地锁定和释放锁,并且可以更好地控制线程的访问顺序。 总结来说,synchronized是一种简单易用的线程同步机制,适用于简单的同步需求;而reentrantlock则是一种更灵活、可扩展的线程同步机制,适用于复杂的同步需求。对于大多数普通的线程同步问题,synchronized已经足够胜任,但是对于需要更高级特性的同步需求,reentrantlock是一个更好的选择。 ### 回答3: synchronized和reentrant lock是Java中用于实现线程同步的两种机制。 首先,synchronized是Java中内置的一个关键字,用于实现对象级别的同步。它可以修饰方法或者代码块,实现对共享资源的互斥访问。当一个线程进入synchronized修饰的方法或者代码块时,会尝试获取对象的锁,如果锁被其他线程占用,则该线程会处于等待状态。当锁被释放时,等待的线程会争夺锁的拥有权,获取锁进入执行。synchronized机制简单易用,且可以自动释放锁,但是只能实现基本的互斥同步,如果需要更多的控制,不能满足需求。 而reentrant lock是Java.util.concurrent包中提供的一个可重入锁,实现了Lock接口。它是一个更加灵活和强大的同步机制,可以实现更复杂的线程同步。与synchronized不同的是,reentrant lock需要手动加锁和解锁。它可以实现公平锁和非公平锁,提供了更精确的线程同步控制,可以实现尝试获取锁、获取锁超时和可中断等功能。此外,reentrant lock还可以创建多个条件对象,可以通过条件对象来实现更加灵活的线程协作。 总结来说,synchronized适用于简单的互斥同步,使用方便但功能相对较少。而reentrant lock虽然使用稍复杂,但提供了更高级别的同步控制和更多的特性,可以满足更复杂的线程同步需求。选择使用哪种机制要根据具体的需求来决定。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值