我有一个程序的一部分,与另一个系统并行实现同步。每个同步过程都会打开新事务。同步对象将映射到某些数据并保持不变。 有关哪些对象的映射数据可以在之前保留,并且可以在同步时间内保留。
我的持久层提供级联操作,所以我不关心所有依赖关系。
在映射对象之前没有持久化并且在同步时间它将同时保存在不同的线程中出现hibernate异常:
org.springframework.dao.DataIntegrityViolationException: A different object with the same identifier value was already associated with the session : [my.project.entity.CrmCaseLink#27];
nested exception is javax.persistence.EntityExistsException: A different object with the same identifier value was already associated with the session : [my.project.entity.CrmCaseLink#27]
1
我尝试使用缓存,我可以为不同的线程获得相同的CrmCaseLink
,但它并没有解决我的问题。
2
我再试一次
@CacheConfig(cacheNames = Constants.CRM_CASE_LINK_CACHE, cacheManager = Constants.GUAVA_CACHE_MANAGER)
@Service
public class CrmCaseLinkService {
private static final Logger LOGGER = LogManager.getLogger(CrmCaseLinkService.class);
@Autowired
private CrmCaseLinkRepository crmCaseLinkRepository;
@Cacheable(key = "#id")
@Transactional(propagation = Propagation.REQUIRES_NEW)
public synchronized CrmCaseLink findOne(Long id) {
LOGGER.debug("trace loading crm case link by specified id: {}", id);
return Optional.ofNullable(crmCaseLinkRepository.findOne(id))
.orElseGet(() -> crmCaseLinkRepository.save(new CrmCaseLink(id)));
}
}
为什么缓存是错误的?我希望hiberate
能够在不同的事务中管理相同的非持久化实体。我确信在单独的spring bean中使用新事务的缓存解决方案是无可争议的解决方案,但它也对我没有帮助。
java8,spring,spring data jpa,postgres,hibernate5,jetty container
答案 0 :(得分:1)
我认为问题出现是因为在提供的解决方案中使用了分离的实体对象。如您所见,我为同步单例方法创建了新事务
@Transactional(propagation = Propagation.REQUIRES_NEW)
public synchronized CrmCaseLink findOne(Long id) {
LOGGER.debug("trace loading crm case link by specified id: {}", id);
return Optional.ofNullable(crmCaseLinkRepository.findOne(id))
.orElseGet(() -> crmCaseLinkRepository.save(new CrmCaseLink(id)));
}
如果数据库存储中缺少新对象,则保留新对象。但是public synchronized CrmCaseLink findOne
用于从另一个事务中获取CrmCaseLink
并将其映射到其他对象上。要成功映射和持久保存其他实体,我必须使用附加的CrmCaseLink
。
我已经用下一个方式更改了代码:
CrmCaseLink
。代码示例:
@Service
public class CrmCaseLinkManager implements CrmLinkManager {
@Autowired
private CrmCaseLinkRepository crmCaseLinkRepository;
@Override
public CrmCaseLink findOne(Long id) {
return crmCaseLinkRepository.findOne(id);
}
@Override
@Transactional(propagation = Propagation.REQUIRES_NEW)
public synchronized void save(Long id) {
if (crmCaseLinkRepository.exists(id)) {
return;
}
crmCaseLinkRepository.save(new CrmCaseLink(id));
}
}
使用它的代码部分演示了如何从已实现的服务中获取必要的附加实体:
T getOrCreateCrmLink(Long id, CrmLinkManager<T> crmLinkManager) {
T crmLink = crmLinkManager.findOne(id);
if (crmLink == null) {
crmLinkManager.save(id);
crmLink = crmLinkManager.findOne(id);
}
return crmLink;
}