本文介紹了Spring Boot 2@Transaction批注使自動連接的字段為空的處理方法,對大家解決問題具有一定的參考價值,需要的朋友們下面隨著小編來一起學習吧!
問題描述
我正嘗試在我的服務的方法中使用@Transactional
注釋來延遲加載域。但是,在我的實現類上使用@Transactional
會使所有自動綁定的字段null
。
以下是我的實現:
@Service
public class UserServiceImpl implements UserService {
/**
* DefaultMapper.
*/
@Autowired
private DefaultMapper defaultMapper;
/**
* Resource service injection.
*/
@Autowired
private ResourceService resourceService;
/**
* UserRepository.
*/
@Autowired
private UserRepository userRepository;
/**
* Jwt Factory.
*/
@Autowired
private JwtService jwtService;
@Override
@Transactional
public final UserDto findByLogin(final String login) throws ResourceNotFoundException {
// user repository is null here when using @Transactional
User user = this.userRepository.findByLogin(login)
.orElseThrow(() -> new ResourceNotFoundException(
resourceService.getMessage(MessageBundle.EXCEPTION, "resource.notfound.user.login")
));
UserDto userDto = defaultMapper.asUserDtoWithRoles(user);
return userDto;
}
提前謝謝您。
推薦答案
事務是使用AOP應用的,Spring中的默認AOP機制是使用代理。使用Spring Boot時,代理模式設置為基于類的代理。
您可以使用以下兩種方法中的一種來解決此問題。
-
從您的方法中刪除
final
通過將
spring.aop.proxy-target-class=false
添加到application.properties
來禁用基于類的代理
現在,當您加載@Transactional
時,這將導致您的UserServiceImpl
的代理被創建,準確地說是基于類的代理。發生的情況是為UserServiceImpl
創建了一個子類,并且所有方法都被覆蓋以應用TransactionInterceptor
。但是,由于您的方法被標記為final
,所以動態創建的類不能覆蓋此方法。因此,該方法查看動態創建的代理類中的字段實例,該實例始終為null
。
當刪除final
方法時,可以重寫該方法,應用該行為,并且它將查看正確的字段實例(實際UserServiceImpl
而不是代理)。
禁用基于類的代理時,您將獲得一個JDK動態代理,它基本上是一個實現服務實現的所有接口的瘦包裝。它應用添加的行為(事務)并調用實際服務。不需要對實際類進行擴展,因此您可以代理最終方法(只要它是您的接口的一部分)。
這篇關于Spring Boot 2@Transaction批注使自動連接的字段為空的文章就介紹到這了,希望我們推薦的答案對大家有所幫助,