📄 hibernatetransactionmanager.java
字号:
txObject.getSessionHolder().setSynchronizedWithTransaction(true);
Session session = txObject.getSessionHolder().getSession();
if (debugEnabled) {
logger.debug("Beginning Hibernate transaction on session [" + session + "]");
}
// apply read-only
if (definition.isReadOnly()) {
if (txObject.isNewSessionHolder()) {
// just set to NEVER in case of a new Session for this transaction
session.setFlushMode(FlushMode.NEVER);
}
try {
Connection con = session.connection();
if (debugEnabled) {
logger.debug("Setting JDBC connection [" + con + "] read-only");
}
con.setReadOnly(true);
}
catch (Exception ex) {
// SQLException or UnsupportedOperationException
logger.warn("Could not set JDBC connection read-only", ex);
}
}
else if (!txObject.isNewSessionHolder()) {
// we need AUTO or COMMIT for a non-read-only transaction
FlushMode flushMode = session.getFlushMode();
if (FlushMode.NEVER.equals(flushMode)) {
txObject.setPreviousFlushMode(flushMode);
session.setFlushMode(FlushMode.AUTO);
}
}
// apply isolation level
if (definition.getIsolationLevel() != TransactionDefinition.ISOLATION_DEFAULT) {
Connection con = session.connection();
if (debugEnabled) {
logger.debug("Changing isolation level of JDBC connection [" + con + "] to " +
definition.getIsolationLevel());
}
txObject.setPreviousIsolationLevel(new Integer(con.getTransactionIsolation()));
session.connection().setTransactionIsolation(definition.getIsolationLevel());
}
// add the Hibernate transaction to the session holder
txObject.getSessionHolder().setTransaction(session.beginTransaction());
// register transaction timeout
if (definition.getTimeout() != TransactionDefinition.TIMEOUT_DEFAULT) {
txObject.getSessionHolder().setTimeoutInSeconds(definition.getTimeout());
}
// bind the session holder to the thread
if (txObject.isNewSessionHolder()) {
TransactionSynchronizationManager.bindResource(this.sessionFactory, txObject.getSessionHolder());
}
// register the Hibernate Session's JDBC Connection for the DataSource, if set
if (this.dataSource != null) {
ConnectionHolder conHolder = new ConnectionHolder(session.connection());
if (definition.getTimeout() != TransactionDefinition.TIMEOUT_DEFAULT) {
conHolder.setTimeoutInSeconds(definition.getTimeout());
}
TransactionSynchronizationManager.bindResource(this.dataSource, conHolder);
}
}
catch (SQLException ex) {
throw new CannotCreateTransactionException("Could not set transaction isolation", ex);
}
catch (HibernateException ex) {
throw new CannotCreateTransactionException("Could not create Hibernate transaction", ex);
}
}
protected Object doSuspend(Object transaction) {
HibernateTransactionObject txObject = (HibernateTransactionObject) transaction;
txObject.setSessionHolder(null);
SessionHolder sessionHolder = (SessionHolder) TransactionSynchronizationManager.unbindResource(this.sessionFactory);
ConnectionHolder connectionHolder = null;
if (this.dataSource != null) {
connectionHolder = (ConnectionHolder) TransactionSynchronizationManager.unbindResource(this.dataSource);
}
return new SuspendedResourcesHolder(sessionHolder, connectionHolder);
}
protected void doResume(Object transaction, Object suspendedResources) {
SuspendedResourcesHolder resourcesHolder = (SuspendedResourcesHolder) suspendedResources;
if (TransactionSynchronizationManager.hasResource(this.sessionFactory)) {
// from non-transactional code running in active transaction synchronization
// -> can be safely removed, will be closed on transaction completion
TransactionSynchronizationManager.unbindResource(this.sessionFactory);
}
TransactionSynchronizationManager.bindResource(this.sessionFactory, resourcesHolder.getSessionHolder());
if (this.dataSource != null) {
TransactionSynchronizationManager.bindResource(this.dataSource, resourcesHolder.getConnectionHolder());
}
}
protected boolean isRollbackOnly(Object transaction) {
return ((HibernateTransactionObject) transaction).getSessionHolder().isRollbackOnly();
}
protected void doCommit(DefaultTransactionStatus status) {
HibernateTransactionObject txObject = (HibernateTransactionObject) status.getTransaction();
if (status.isDebug()) {
logger.debug("Committing Hibernate transaction on session [" +
txObject.getSessionHolder().getSession() + "]");
}
try {
txObject.getSessionHolder().getTransaction().commit();
}
catch (net.sf.hibernate.TransactionException ex) {
// assumably from commit call to underlying JDBC connection
throw new TransactionSystemException("Could not commit Hibernate transaction", ex);
}
catch (JDBCException ex) {
// assumably failed to flush changes to database
throw convertJdbcAccessException(ex.getSQLException());
}
catch (HibernateException ex) {
// assumably failed to flush changes to database
throw convertHibernateAccessException(ex);
}
}
protected void doRollback(DefaultTransactionStatus status) {
HibernateTransactionObject txObject = (HibernateTransactionObject) status.getTransaction();
if (status.isDebug()) {
logger.debug("Rolling back Hibernate transaction on session [" +
txObject.getSessionHolder().getSession() + "]");
}
try {
txObject.getSessionHolder().getTransaction().rollback();
}
catch (net.sf.hibernate.TransactionException ex) {
throw new TransactionSystemException("Could not rollback Hibernate transaction", ex);
}
catch (JDBCException ex) {
// shouldn't really happen, as a rollback doesn't cause a flush
throw convertJdbcAccessException(ex.getSQLException());
}
catch (HibernateException ex) {
// shouldn't really happen, as a rollback doesn't cause a flush
throw convertHibernateAccessException(ex);
}
}
protected void doSetRollbackOnly(DefaultTransactionStatus status) {
HibernateTransactionObject txObject = (HibernateTransactionObject) status.getTransaction();
if (status.isDebug()) {
logger.debug("Setting Hibernate transaction on Session [" +
txObject.getSessionHolder().getSession() + "] rollback-only");
}
txObject.getSessionHolder().setRollbackOnly();
}
protected void doCleanupAfterCompletion(Object transaction) {
HibernateTransactionObject txObject = (HibernateTransactionObject) transaction;
// remove the JDBC connection holder from the thread, if set
if (this.dataSource != null) {
TransactionSynchronizationManager.unbindResource(this.dataSource);
}
// remove the session holder from the thread
if (txObject.isNewSessionHolder()) {
TransactionSynchronizationManager.unbindResource(this.sessionFactory);
}
try {
Connection con = txObject.getSessionHolder().getSession().connection();
// reset transaction isolation to previous value, if changed for the transaction
if (txObject.getPreviousIsolationLevel() != null) {
if (logger.isDebugEnabled()) {
logger.debug("Resetting isolation level of connection [" + con + "] to " +
txObject.getPreviousIsolationLevel());
}
con.setTransactionIsolation(txObject.getPreviousIsolationLevel().intValue());
}
// reset read-only
if (con.isReadOnly()) {
if (logger.isDebugEnabled()) {
logger.debug("Resetting read-only flag of connection [" + con + "]");
}
con.setReadOnly(false);
}
}
catch (Exception ex) {
// HibernateException, SQLException, or UnsupportedOperationException
// typically not something to worry about, can be ignored
logger.info("Could not reset JDBC connection of Hibernate session", ex);
}
Session session = txObject.getSessionHolder().getSession();
if (txObject.isNewSessionHolder()) {
if (logger.isDebugEnabled()) {
logger.debug("Closing Hibernate session [" + session + "] after transaction");
}
try {
SessionFactoryUtils.closeSessionIfNecessary(session, this.sessionFactory);
}
catch (CleanupFailureDataAccessException ex) {
// just log it, to keep a transaction-related exception
logger.error("Count not close Hibernate session after transaction", ex);
}
}
else {
if (logger.isDebugEnabled()) {
logger.debug("Not closing pre-bound Hibernate session [" + session + "] after transaction");
}
txObject.getSessionHolder().setTransaction(null);
if (txObject.getPreviousFlushMode() != null) {
session.setFlushMode(txObject.getPreviousFlushMode());
}
}
}
/**
* Convert the given HibernateException to an appropriate exception from
* the org.springframework.dao hierarchy. Can be overridden in subclasses.
* @param ex HibernateException that occured
* @return the corresponding DataAccessException instance
*/
protected DataAccessException convertHibernateAccessException(HibernateException ex) {
return SessionFactoryUtils.convertHibernateAccessException(ex);
}
/**
* Convert the given SQLException to an appropriate exception from the
* org.springframework.dao hierarchy. Uses a JDBC exception translater if set,
* and a generic HibernateJdbcException else. Can be overridden in subclasses.
* @param ex SQLException that occured
* @return the corresponding DataAccessException instance
* @see #setJdbcExceptionTranslator
*/
protected DataAccessException convertJdbcAccessException(SQLException ex) {
if (this.jdbcExceptionTranslator != null) {
return this.jdbcExceptionTranslator.translate("HibernateTemplate", null, ex);
}
else {
return new HibernateJdbcException(ex);
}
}
/**
* Holder for suspended resources.
* Used internally by doSuspend and doResume.
* @see #doSuspend
* @see #doResume
*/
private static class SuspendedResourcesHolder {
private final SessionHolder sessionHolder;
private final ConnectionHolder connectionHolder;
private SuspendedResourcesHolder(SessionHolder sessionHolder, ConnectionHolder connectionHolder) {
this.sessionHolder = sessionHolder;
this.connectionHolder = connectionHolder;
}
private SessionHolder getSessionHolder() {
return sessionHolder;
}
private ConnectionHolder getConnectionHolder() {
return connectionHolder;
}
}
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -