spring transaction不生效的一些原因


原文:spring transaction不生效的一些原因

同一個類中一個方法調用的另一個有事務的方法

在spring的聲明式注解中,如果直接調用的方法不包含Transactional注解,那么即使這個方法里面調用的其他方法包含注解,那么事務也是不會生效的,例如下面的這段代碼:

class AImpl implements A {
	@Override
	public void update() {
		doSomething();
	}
	@Transactional
	@Override
	public void doSomething() {
		//...
	}
}

這里如果調用的是instanceA.update()方法,doSomething()的事務是不生效的,只有調用instanceA.doSomething()事務才會生效。

為什么

spring默認基於動態代理技術生成一個類來提供事務相關的功能:

public class TxHandler implements InvocationHandler {
	private Object originalObject;

	public Object bind(Object obj) {
		this.originalObject = obj;
		return Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj.getClass().getInterfaces(), this);
	}

	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
		Object result = null;
		if (!method.getName().startsWith("save")) {
			UserTransaction tx = null;
			try {
				tx = (UserTransaction) (new InitialContext().lookup("java/tx"));
				result = method.invoke(originalObject, args);
				tx.commit();
			} catch (Exception ex) {
				if (null != tx) {
					try {
						tx.rollback();
					} catch (Exception e) {
					}
				}
			}
		} else {
			result = method.invoke(originalObject, args);
		}
		return result;
	}
}

無論開始調用的方法又沒有事務注解,一旦執行invoke()后,里面的方法調用就跟代理沒有關系了,都是內部方法調用,也不會享受到事務的功能。

如果非要方法內部調用事務方法,只要保證調用的是被spring增強過的方法就可以了,雖然可用,但估計很少人用:

class AImpl implements A {
	@Autowire
	private ApplicationContext applicationContext;

	@Override
	public void update() {
		applicationContext.getBean(A.getClass()).doSomething();//從spring容器獲取代理后的對象
	}
	@Transactional
	@Override
	public void doSomething() {
		//...
	}
}

參考

http://mojito515.github.io/blog/2016/08/31/transactionalinspring/
https://spring.io/blog/2012/05/23/transactions-caching-and-aop-understanding-proxy-usage-in-spring


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM