我有一个非事务性和异步方法,它调用同一服务的其他事务性方法。
如果发生了什么事情,我会捕获一个异常,如果发生了什么事情,我会保存错误,但回滚不起作用。
@Service
@Transactional(readOnly = true)
public class PlayerServiceImpl implements PlayerService {
@Inject
PlayerRepository playerRepository;
@Override
@Transactional(readOnly = false, propagation = Propagation.REQUIRED)
public void save(PlayerEntity player) {
//more code here
playerRepository.save(player);
}
//other methods
}
@Service
public class TeamServiceImpl implements TeamService {
@Inject
TeamRepository teamRepository;
@Inject
MessageRepository messageRepository;
@Inject
ErrorRepository errorRepository;
@Inject
PlayerService playerService;
@Async("asyncExecutor")
@Override
public void saveWithPlayersAsync(TeamEntity team, User user) {
MessageEntity message = new MessageEntity();
try {
//I want rollback this if something wrong happens
this.savePlayersA(team);
this.savePlayersB(team);
message.setContent("Ok !");
} catch (TeamException e) {
//e.printStackTrace();
message.setContent("Fail !");
message.setUser(user)
//I save the error for audit
errorRepository.save(new Error("Fail", user.getCode()));
} finally {
//always save message for user than execute de function
messageRepository.save(message);
}
}
@Transactional(readOnly = false, rollbackFor = TeamException.class)
private void savePlayersA(TeamEntity team) throws TeamException {
PlayerEntity p1 = new PlayerEntity();
p1.setName("Name 1");
p1.setSurname("Surname 1");
p1.setTeam(team);
playerService.save(p1);
PlayerEntity p2 = new PlayerEntity();
p2.setName("Name 2");
p2.setSurname("Surname 2");
p2.setTeam(team);
playerService.save(p2);
}
@Transactional(readOnly = false, rollbackFor = TeamException.class)
private void savePlayersB(TeamEntity team) throws TeamException {
PlayerEntity p3 = new PlayerEntity();
p3.setName("Name 3");
p3.setSurname("Surname 3");
p3.setTeam(team);
playerService.save(p3);
PlayerEntity p4 = new PlayerEntity();
p4.setName("Name 4");
p4.setSurname("Surname 4");
p4.setTeam(team);
playerService.save(p4);
// here something happens and throw my custom exception
if (true) {
throw new TeamException("Fail!");
}
}
}
这不回滚的原因是因为调用事务性方法的异步方法在同一个类中。当您用@transaction或@Async注释方法时,注释代码由Spring创建的代理处理。当被注释的方法像这样在同一个类中时,对它们的调用不能被代理捕获以通过spring处理额外的请求,因此对方法的注释是多余的。您可以通过将savePlayer方法放在一个单独的类中,并在其中使用事务性方法来实现这一点。
问题内容: 我使用的是Spring 4,我注意到了一个奇怪的行为……如果我从普通实例方法多次调用异步方法,那么它们都将在不同的线程中调用,并在随机时间完成。但是,如果我多次从另一个异步方法中调用一个异步方法,那么它们将按顺序完成。我有这样的事情: 我正在使用默认的异步执行器。我应该换一个吗?但是,该执行程序不会重用任何线程,而是每次都启动另一个线程,因此应该没问题……这仅仅是巧合吗?但是我尝试了十
我正在尝试创建一个事务方法,该方法调用其他几个事务方法以保存一些相互依赖的db实体。如果任何调用失败,我希望事务完全回滚。但是,这不是观察到的行为。这是我的代码: 也有and,但是当事务在第二次调用时失败时,第一个被提交。
问题内容: 在Spring中,带有注释的方法 将获得一个新的事务(如果还没有),但是我注意到,如果从非事务处理方法调用事务方法,则不会获得任何事务。这是代码。 是一个常规方法,该方法调用是事务性的,但不会保留任何更改。 我正在使用Spring 3和Hibernate3。在这里我做错了什么?谢谢。 问题答案: 这是Springs AOP的局限性之一。因为dao bean实际上是在spring时创建的
问题内容: 在Bruce Eckel的“ Thinking In Java,第四版”的第428页(有关类型信息的章节)中,具有以下示例: 也许我有点累,但是我看不到add()方法中对add()的调用是如何工作的。我一直认为它应该有一个引用,或者是一个静态方法(并且我在ArrayList或List中找不到静态add())。我想念什么? 我只是为自己测试,发现这可行: 问题答案: Java为这样的方法
没有await并且在非异步方法中调用异步方法的行为是什么?我这样问是因为我看到Visual Studio在调用异步方法时没有显示任何警告,好像这是一件非常正常的事情。在这种情况下,异步方法的行为是否像是同步的?