A, introducing
In the previous article, we analyzed the transaction related object components in Spring. With these basic components understood, it is much easier to look at the source code of the Spring transaction process. In the Spring transaction process, the in-depth analysis will only be to start the transaction part of the process. But don’t worry, if you can understand how a transaction is started, then it’s easy to look at the commit and rollback of a transaction, right
Second, transaction source code analysis
2.1. Transaction manager interface
public interface PlatformTransactionManager {
TransactionStatus getTransaction(TransactionDefinition definition);
void commit(TransactionStatus status) throws TransactionException;
void rollback(TransactionStatus status) throws TransactionException;
}
Copy the code
In the previous analysis, we know that the transaction object DataSourceTransactionObject will hold a rollback point connection and related operations, The transaction object is used to realize complete transaction features using PlatformTransactionManager affairs manager to complete, interface defines three methods:
-
getTransaction: Get the Transactional state according to the Transactional definition, or if no transaction exists, create a transaction and return the state. This method is used after the Transactional annotation is intercepted by AOP. Then use this information to create a transaction object. After creating a transaction object, use the TransactionStatus object TransactionStatus to save the current situation of the transaction. In the subsequent transaction process, the transaction state will be reversed, so the transaction state object is throughout the whole transaction processing process
-
Commit: Commit a transaction using a transaction state object
-
Rollback: use the transaction state object to rollback the transaction
PlatformTransactionManager transaction manager defines the function of the transaction, the abstract class AbstractPlatformTransactionManager need to implement these interfaces, and provide the template, the subclass to decide affairs open, close, and other functions, AbstractPlatformTransactionManager operating at the same time to provide the transaction propagation behavior, so AbstractPlatformTransactionManager implementation is common functions, Different transactions (such as JTA distributed transactions, ordinary JDBC transaction) opening and closing operation is different, different transaction inheritance AbstractPlatformTransactionManager to provide different implementation classes, and realize the common functions of the template
2.2 transaction interceptor
The Transactional transaction manager provides Transactional operations, such as creating, committing, and rolling back transactions. The use of a @Transactional annotation requires the integration of these operations. AOP functions are used to schedule Transactional functions using AOP. AOP involves two main classes: TransactionInterceptor extend TransactionAspectSupport, actually really achieve AOP operation is the latter, namely the parent class, subclass on the basis of the function of the parent class, provides the AOP object to obtain the function of the source object, Since AOP objects are simply other implementation classes of the @Transactional annotation class’s interface based on interface proxies, pointcut objects in the proxy class do not get Transactional annotation information until they get the source object. So the TransactionInterceptor provides the ability to get the source Class object. The real AOP operations are done by the parent Class:
public class TransactionInterceptor extends TransactionAspectSupport {
public Object invoke(MethodInvocation invocation) throws Throwable {
// Get the source Class objectClass<? > targetClass = (invocation.getThis() ! =null ? AopUtils.getTargetClass(invocation.getThis()) : null);
// Use the parent class invokeWithinTransaction to complete the function of the transaction interceptor
returninvokeWithinTransaction(invocation.getMethod(), targetClass, invocation::proceed); }}public abstract class TransactionAspectSupport {
protected Object invokeWithinTransaction(Method method, @NullableClass<? > targetClass,final InvocationCallback invocation) throws Throwable {
TransactionAttributeSource tas = getTransactionAttributeSource();
finalTransactionAttribute txAttr = (tas ! =null ? tas.getTransactionAttribute(method, targetClass) : null);
final PlatformTransactionManager tm = determineTransactionManager(txAttr);
final String joinpointIdentification = methodIdentification(method, targetClass, txAttr);
if (txAttr == null| |! (tminstanceof CallbackPreferringPlatformTransactionManager)) {
TransactionInfo txInfo = createTransactionIfNecessary(tm, txAttr, joinpointIdentification);
Object retVal;
try {
retVal = invocation.proceedWithInvocation();
}
catch (Throwable ex) {
completeTransactionAfterThrowing(txInfo, ex);
throw ex;
}
finally {
cleanupTransactionInfo(txInfo);
}
commitTransactionAfterReturning(txInfo);
return retVal;
}
else{... The transaction manager for CallbackPreferringPlatformTransactionManager cases operation... If you are interested in this section, you can explore it. In fact, it provides the function of callback during transaction operation. It is very easy to study this section once you understand the basic function of transaction manager.Copy the code
TransactionAttributeSource we described before, used to get the @ Transactional annotation information class, by calling its getTransactionAttribute method, Parsing @ Transactional annotation for TransactionAttribute object (actually RuleBasedTransactionAttribute)
Call determineTransactionManager method determine the transaction manager, parameters for TransactionAttribute, because @ Transactional annotation can specify the transaction manager, Here is the value of @ Transactional annotation/transactionmanager properties, obtain the corresponding transaction manager PlatformTransactionManager
If txAttr is empty (for execution is @ Transactional annotations of a class of transaction method) or the transaction manager not CallbackPreferringPlatformTransactionManager (see the code above description), Create a transaction is called createTransactionIfNecessary method, then using the invocation. ProceedWithInvocation perform business code in the method, After a successful call the commitTransactionAfterReturning method to commit the transaction, if there are abnormal call completeTransactionAfterThrowing method rollback transaction in the transaction method, Void txAttr is excluded (methods not annotated with the @Transactional annotation)
CreateTransactionIfNecessary method is called PlatformTransactionManager. GetTransaction method to create a transaction (if there is a communication behavior will in this processing). CommitTransactionAfterReturning method is called PlatformTransactionManager.com MIT to commit the transaction, Call the PlatformTransactionManager completeTransactionAfterThrowing method. The rollback transaction rollback method, so, The real function of the Transactional AOP interceptor and its parent class, TransactionAspectSupport, is to intercept methods where the @Transactional annotation exists, And using the transaction manager PlatformTransactionManager to complete the transaction scheduling function
2.3, AbstractPlatformTransactionManager transaction interceptor source analysis
2.3.1 getTransaction Obtains a transaction
public final TransactionStatus getTransaction(@Nullable TransactionDefinition definition) throws TransactionException {
Object transaction = doGetTransaction();
if (definition == null) {
definition = new DefaultTransactionDefinition();
}
if (isExistingTransaction(transaction)) {
return handleExistingTransaction(definition, transaction, debugEnabled);
}
if (definition.getPropagationBehavior() == TransactionDefinition.PROPAGATION_MANDATORY) {
throw new IllegalTransactionStateException(
"No existing transaction found for transaction marked with propagation 'mandatory'");
} else if (definition.getPropagationBehavior() == TransactionDefinition.PROPAGATION_REQUIRED ||
definition.getPropagationBehavior() == TransactionDefinition.PROPAGATION_REQUIRES_NEW ||
definition.getPropagationBehavior() == TransactionDefinition.PROPAGATION_NESTED) {
SuspendedResourcesHolder suspendedResources = suspend(null);
booleannewSynchronization = (getTransactionSynchronization() ! = SYNCHRONIZATION_NEVER); DefaultTransactionStatus status = newTransactionStatus( definition, transaction,true, newSynchronization, debugEnabled, suspendedResources);
doBegin(transaction, definition);
prepareSynchronization(status, definition);
return status;
} else {
boolean newSynchronization = (getTransactionSynchronization() == SYNCHRONIZATION_ALWAYS);
return prepareTransactionStatus(definition, null.true, newSynchronization, debugEnabled, null); }}Copy the code
The first is to call doGetTransaction to get a transaction object, and create one if one does not exist. This method is abstract and implemented by subclasses. Different transaction managers provide different transaction objects. We usually use the transaction manager is DataSourceTransactionManager, so its transaction object is: create DataSourceTransactionObject
After obtaining the current transaction object, began to judge, if the current existing transaction, then walk handleExistingTransaction to processing, according to the different transaction propagation behavior, will have different processing
If no transaction currently exists, the current transaction propagation behavior is MANDATORY, and an exception is thrown. Call the suspend transaction method if the propagation behavior is PROPAGATION_REQUIRED, PROPAGATION_REQUIRES_NEW, and PROPAGATION_NESTED. All of the logic that goes through here does nothing in the suspend method, it just returns NULL, and the point of calling suspend is to prevent potential synchronization, or illegal synchronization, which we’ll talk about later in the summary, Create a transaction state that will continue for the entire life of the transaction. Then call doBegin to start the transaction. DoBegin is an abstract method where subclasses decide how transactions are started. Okay
If it’s not one of these isolation levels, else logic is used to create an empty transaction, and the reason for doing that, according to the comments in the code, is to prevent potential synchronization behavior, which we’ll talk about later
2.3.2 doGetTransaction Creates a transaction object
Let’s take a look at DataSourceTransactionManager doGetTransaction method:
protected Object doGetTransaction(a) {
DataSourceTransactionObject txObject = new DataSourceTransactionObject();
txObject.setSavepointAllowed(isNestedTransactionAllowed());
ConnectionHolder conHolder =
(ConnectionHolder) TransactionSynchronizationManager.getResource(obtainDataSource());
txObject.setConnectionHolder(conHolder, false);
return txObject;
}
Copy the code
Create a DataSourceTransactionObject transaction object, and then get ConnectionHolder, and set to the transaction object, in the Spring, the database connection in the form of ThreadLocal is saved in the thread object, Normally TransactionSynchronizationManager getResource method is returns null, only in the case of nested transaction emerged, That is, one @Transactional method calls another @Transactional method. Connections can only be made from thread local variables, so the transaction object that was retrieved when doGetTransaction was first called There are no connection objects, so this method is compatible with nested transactions
In general, this code produces the following result:
DataSourceTransactionObject txObject = new DataSourceTransactionObject();
txObject.setSavepointAllowed(isNestedTransactionAllowed());
txObject.setConnectionHolder(null, false);
Copy the code
We will next to detail TransactionSynchronizationManager, wait to read the back of the analysis, will clearly understand the meaning of this paragraph above, the nested transaction to get ConnectionHolder would happen
2.3.3, TransactionSynchronizationManager
public abstract class TransactionSynchronizationManager {
private static final ThreadLocal<Map<Object, Object>> resources =
new NamedThreadLocal<>("Transactional resources");
private static final ThreadLocal<Set<TransactionSynchronization>> synchronizations =
new NamedThreadLocal<>("Transaction synchronizations");
private static final ThreadLocal<String> currentTransactionName =
new NamedThreadLocal<>("Current transaction name");
private static final ThreadLocal<Boolean> currentTransactionReadOnly =
new NamedThreadLocal<>("Current transaction read-only status");
private static final ThreadLocal<Integer> currentTransactionIsolationLevel =
new NamedThreadLocal<>("Current transaction isolation level");
private static final ThreadLocal<Boolean> actualTransactionActive =
new NamedThreadLocal<>("Actual transaction active");
}
Copy the code
-
resources: It saves the connection object in the current thread. Normally, it forms a Map<DataSource, ConnectionHolder>. In 2.3.2 summary, We call TransactionSynchronizationManager. GetResource method is obtained from the ThreadLocal connection object, using the data source in the obtainDataSource access to current transaction manager, This data source is then used to retrieve the saved connection object ConnectionHolder
-
synchronizations: TransactionSynchronization interface provides a transaction callback function, before and after complete the transaction invokes TransactionSynchronization in corresponding callback methods, It functions just like the HandlerInterceptor in SpringMVC, with callbacks like beforeCommit and afterCommit. If you know anything about springMVC interceptors, you’ll see that they implement the same thing. A Set is used to store all the callback functions of the current transaction. After the transaction is completed, the implementation class of the callback interface is iterated one by one and the corresponding callback methods are called
-
CurrentTransactionName: Uses ThreadLocal to hold the name of the transaction in the current thread
-
CurrentTransactionReadOnly: using ThreadLocal to save the current thread the transaction is read-only
-
CurrentTransactionIsolationLevel: using ThreadLocal to save the current thread transaction isolation level
-
ActualTransactionActive: Uses ThreadLocal to store whether the current thread has started transactions
Mybatis TransactionSynchronizationManager behind the SqlSession objects will be used for transmission of integration, here is to link object, realize the operation of a context, in the back of the operation can be obtained from the thread local variable connection
2.3.4 doBegin Starts the transaction
Let’s take a look at DataSourceTransactionManager doBegin method:
protected void doBegin(Object transaction, TransactionDefinition definition) {
DataSourceTransactionObject txObject = (DataSourceTransactionObject) transaction;
Connection con = null;
if(! txObject.hasConnectionHolder() || txObject.getConnectionHolder().isSynchronizedWithTransaction()) { Connection newCon = obtainDataSource().getConnection(); txObject.setConnectionHolder(new ConnectionHolder(newCon), true);
}
txObject.getConnectionHolder().setSynchronizedWithTransaction(true);
con = txObject.getConnectionHolder().getConnection();
Integer previousIsolationLevel = DataSourceUtils.prepareConnectionForTransaction(con, definition);
txObject.setPreviousIsolationLevel(previousIsolationLevel);
if (con.getAutoCommit()) {
txObject.setMustRestoreAutoCommit(true);
con.setAutoCommit(false);
}
prepareTransactionalConnection(con, definition);
txObject.getConnectionHolder().setTransactionActive(true);
int timeout = determineTimeout(definition);
if(timeout ! = TransactionDefinition.TIMEOUT_DEFAULT) { txObject.getConnectionHolder().setTimeoutInSeconds(timeout); }if(txObject.isNewConnectionHolder()) { TransactionSynchronizationManager.bindResource(obtainDataSource(), txObject.getConnectionHolder()); }}Copy the code
Reach doBegin has two cases, one is the first call @ Transactional method, in this time of the transaction object DataSourceTransactionObject haven’t ConnectionHolder connection object, The second is the case of transaction nesting, transaction nesting also need to use doBegin to open the transaction, this time the transaction object has been connected to the object, so we look at this piece of source code from two perspectives
If there is no connection object, the obtainDataSource method is called to get the data source and get the connection object, then a ConnectionHolder is created and placed into the transaction object
Transaction isolation level, before you call setPreviousIsolationLevel method save in the case of nested, layer to restore the data after the transaction has been completed
Call con. SetAutoCommit (false) open transaction (not automatically submit is open transactions), prepareTransactionalConnection method is to a certain configuration of the database connection, protected modified method, the default implementation, Subclasses can also customize implementations, followed by some initialization operations
If the current transaction is a new ConnectionHolder, the outermost layer of a nested transaction, or if there is no nested transaction at all, call the bindResource method with key as DataSource. Value for ConnectionHolder data binding to the current thread local variable, namely 2.3.3 subtotal TransactionSynchronizationManager we analysis the ThreadLocal resources
ThreadLocal is used to maintain the connection object, so it can be used to retrieve the connection object. Only need the step of the mybatis to get connection object to provide a custom implementation class, and then in the implementation class using TransactionSynchronizationManager to get connection object, this completes the integration
3, summarize
So far, we have analyzed the source code of the transaction, in the process of analysis, we learned that the transaction manager provides the function of the transaction, open the transaction, submit the transaction, roll back the transaction and other operations, the whole transaction life cycle will use TransactionStatus to change the state of the transaction, Each transaction manager has its own Datasource object based on which transactions are started and connections are created
The Transactional manager provides transaction management functions. Scheduling these functions to make the @Transactional annotation work is done using AOP, using the TransactionInterceptor to intercept annotations, and using the transaction manager to actually complete the transaction. Namely the try catch {} {}, call PlatformTransactionManager. GetTransaction ways to create a transaction, called PlatformTransactionManager.com MIT method to commit the transaction, Call the PlatformTransactionManager. Rollback transaction rollback method
GetTransaction methods are implemented AbstractPlatformTransactionManager, so provides the function of the public that the function of the transaction propagation behavior, and then provides a template to subclass implementation, subclasses decide transaction object access (different subclasses can transaction object is not the same, Upper using Object receives), subclasses decide how to open the transaction at the same time, at the same time, also provides transaction callback function, use of TransactionSynchronization interface to define the callback function, BeforeCommit, afterCommit, beforeCompletion, afterCompletion and other callback methods are provided, which are called when a transaction commits
Utilizing TransactionSynchronizationManager the entire transaction process, to achieve the transaction information related to the context of preservation, the inside is to use a thread local variable ThreadLocal to complete the save operation, The advantage of this is that the connection object can be fetched from anywhere in the current thread. Spring’s myBatis connection fetch is based on this capability
There are, of course, other methods for transactions, but with the ideas provided in this article, and a clear understanding of these components, it’s easy to explore the logic of these other methods without going into detail here