A conclusion
A Transational annotation does not take effect when a non-transactional method calls a Transational annotation method (@transational) in the same class. 3. In different classes, a method with a transactional annotation method calls a method with a transactional annotation method (@transational), and both methods have transactions. 4. Annotated transactions are valid when a non-transactional method calls another method with a Transational annotated method (@transational)
The second test
All our tests use the default propagation mechanism!!
First let’s look at the test data for the first case
code
expected
The expected effect is that a new data entry will be added, and the value of name with ID 46 and 47 will be changed. Note that when testing, be sure to restart the project if you uncomment @transational! Restart the project! Restart the project! Since Spring uses AOP dynamic proxies, it creates proxy classes dynamically, so don’t be lazy about testing with JRebel!
The actual effect
Ok, let’s take a look at the actual effect (the remarks on the screenshot can be ignored).
Sure enough, it worked as we expected
Let’s test the second case. First, let’s look at the test data
code
Save () and Update () now have @transational transaction annotations
expected
Update () method transactions do not take effect, so changes in update() methods are not rolled back
The actual effect
No surprise, no surprise, the data did not change!! You might think that as long as you annotate all the methods in this class with @transational annotations there are no transactions that will fail, so you’re in a big hole. In the same class a Transational method calls another Transational annotated method (@transational). Only the outer transaction is valid, that is, the update() method throws an exception that is sensed by the save() method, and that causes the transaction to roll back. If THIS doesn’t make sense, Watch the next test and you may find out. I’m going to show you the power of business!! Update () = update(); update() = update(); update() = update();
Console output
Exceptions are eaten, so there is no exception log
The update() method is not rolled back. The catch block in the save() method does not throw an exception, so the data is added. But some of you might ask, why don’t you just throw a try{}catch{} and let the outer layer feel it and roll back your operation. If you want to think about it another way, if the update has a transaction, the data must be rolled back if there is an error in this method, and now there is an error but it is not rolled back because of a catch.
Let’s test the third case. First, let’s take a look at the test data
code
In different classes, one method with a transaction calls another method with a transaction annotation method (@transational)
expected
The data doesn’t change
The actual situation
Again, it’s an error, so let’s look at the data in the database
Okay, so nothing’s changed, so let’s do the same try{}catch{}
Let’s see what happens
See, now that’s business!!
In a multi-layer nested transaction, if the default transaction propagation mode is used, a rollback-only exception will be reported when the inner transaction throws an exception and the outer transaction catches and executes normally. The Spring framework uses AOP to manage transactions. If a transaction-managed method completes its normal execution, Spring commits the SQL in the method at the end of the method. If an exception occurs during method execution, it is rolled back. The spring framework’s default transaction propagation method is PROPAGATION_REQUIRED: If there is no transaction, create a new one, and if one already exists, add it to it. In projects, we usually use the default propagation mode, so that no SQL is executed regardless of an exception in either the outer or inner transaction. In a nested transaction scenario, the SQL for the inner and outer transactions is committed or rolled back at the end of the outer transaction. If the inner transaction throws exception E, spring marks the transaction as “rollback-only” at the end of the inner transaction. If the outer transaction catches exception E, the outer transaction method will continue to execute the code until spring finds that the transaction has been marked “rollbacks-only”, but the method has executed normally. Then the spring will be thrown “org. Springframework. Transaction. UnexpectedRollbackException: Transaction rolled back because it has been marked as rollback-only “. Look at the data in the database
Nothing has changed
Next, let’s test the fourth case. First, let’s look at the test data
Directly on the code
code
This time I will remove the save() annotation and see the result directly.
Transaction rolled back did not appear because it has been marked as rollback-only, because there is no Transaction in the outer layer, let’s take a look at the data in the database
Name (id 48) is not changed, so the transaction is valid!!
Now it is Beijing time, 2021 May 14 00:05:11 tired, today’s use of experience is over, I hope to help you with your work and study
Here is A scenario where A method with transaction A calls A method with transaction annotation (@transational) B, and then B calls A method with transaction annotation (@transational) C. Take a look at the code
The result is that the data from the B method is also committed and not rolled back!! What if C method throws an exception? The answer is all rollback!! Because it joins A transaction in A method, C will cause A method to roll back the transaction!!