场景
真实业务场景中往往存在高并发情况的情况,解决高并发中数据重复异常的手段主要是通过加锁实现,但是加锁的前提是牺牲效率,如何细化锁的影响范围,减少对程序效率的影响。
解决方案
1,利用synchronized关键字可以对对象和代码块加锁的特性,避免对方法全部加锁。
2,利用String.intern()方法保证字面值一样的字符串是同一个对象的特性,通过Interners.newWeakInterner()构建对象池。
示例
例如现在需要根据业务信息生成交易,同一个账户不能生成重复交易数据。
常规写法如下:
public synchronized void createTrade(CreateDTO createDTO) {
//具体生成交易数据业务方法
createDraftTradeLock(createDTO);
}
}
}
以上这种写法将生成交易方法整个加锁,导致同一时间只能执行一条数据,大大影响并发效率。
优化写法:
private static Interner<String> pool = Interners.newWeakInterner();
public void createTrade(CreateDTO createDTO) {
synchronized (pool.intern(createDTO.getReqAcctId())) {
createDraftTradeLock(createDTO);
}
}
以上写法通过Interner构建账户对象池,只会对同一时间相同账户的同一笔交易进行加锁,而同一时间不同账户的交易生成不受影响,从而实现在满足安全的情况下细化了锁的范围,大大提高了程序效率。
注意
此种写法只能满足多线程情况下,对于多进程情况下锁会失效。