spring aop探寻之Before Advice

	/**
	 * Return the AopProxyFactory that this ProxyConfig uses.
	 */
	public AopProxyFactory getAopProxyFactory() {
		return this.aopProxyFactory;
	}

 1 接口的声明

 

/**
 * 应用服务管理
 * 
 * @author hehaibo
 * 
 */
public interface BiapAppService {

	public void insertApp(BiapApp biapApp);

	public void updateApp(BiapApp biapApp);

	public void deleteApp(BiapApp biapApp);

	public Paginator<BiapApp, BiapApp> getPaginatorApp(
			Paginator<BiapApp, BiapApp> biapApp);
	
	public BiapApp getBiapAppById(String appId);

}

 2 接口的实现

 

public class BiapAppServiceImpl implements BiapAppService {

	@Autowired
	private BiapAppDAO biapAppDAO;
	
	public void insertApp(BiapApp biapApp) {
		this.biapAppDAO.insertApp(biapApp);
	}

	public void updateApp(BiapApp biapApp) {
		this.biapAppDAO.updateApp(biapApp);
	}

	public void deleteApp(BiapApp biapApp) {
		this.biapAppDAO.deleteAppById(biapApp.getId());
	}
	
	public Paginator<BiapApp, BiapApp> getPaginatorApp(
			Paginator<BiapApp, BiapApp> biapApp) {
		biapApp.setData(this.biapAppDAO.getObjectList(biapApp));
		biapApp.setCount(this.biapAppDAO.getCount(biapApp));
		return biapApp;
	}

	public BiapApp getBiapAppById(String appId) {
		return this.biapAppDAO.getBiapAppById(appId);
	}
}

 3 BeforeAdvice 实现,简单打印调用方法,模拟记录

 

import java.lang.reflect.Method;

import org.springframework.aop.MethodBeforeAdvice;

/**
 * 方法调用记录
 * 
 * @author hehaibo
 * 
 */
public class InvokeRecordBeforeAdvice implements MethodBeforeAdvice {

	public void before(Method method, Object[] args, Object target)
			throws Throwable {
		System.out.println("invock method:" + target.getClass().getName() + "."
				+ method.getName());
	}

}

 4 测试代码

 

import org.springframework.aop.framework.ProxyFactory;

import com.hhb.biap.base.test.BiapBizBaseTest;
import com.hhb.biap.interceptor.before.InvokeRecordBeforeAdvice;
import com.hhb.biap.model.BiapApp;
import com.hhb.biap.service.BiapAppService;
import com.hhb.biap.service.impl.BiapAppServiceImpl;

public class InvokeRecordBeforeAdviceTest extends BiapBizBaseTest {
	@Test
	public void testBeforeInterceptor() {
		//获得容器中的bean对象
		BiapAppServiceImpl biapAppService = (BiapAppServiceImpl) this.applicationContext
				.getBean("biapAppService");
		System.out.println(biapAppService);
		// 创建代理工厂
		ProxyFactory proxyFactory = new ProxyFactory();
//		proxyFactory.setInterfaces(new Class[]{BiapAppService.class});
		proxyFactory.setTarget(biapAppService);
//		proxyFactory.addAdvice(new InvokeRecordBeforeAdvice());
		BiapAppServiceImpl proxyBiapAppService = (BiapAppServiceImpl) proxyFactory
				.getProxy();
		BiapApp biapApp = proxyBiapAppService.getBiapAppById("biap");
		System.out.println(biapApp);
	}
}

 5 运行结果

 

java.lang.ClassCastException: $Proxy12 cannot be cast to com.hhb.biap.service.impl.BiapAppServiceImpl
	at com.hhb.biap.interceptor.test.InvokeRecordBeforeAdviceTest.testBeforeInterceptor(InvokeRecordBeforeAdviceTest.java:16)
	at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
	at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
	at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
	at java.lang.reflect.Method.invoke(Method.java:597)
	at org.springframework.test.context.junit4.SpringTestMethod.invoke(SpringTestMethod.java:160)
	at org.springframework.test.context.junit4.SpringMethodRoadie.runTestMethod(SpringMethodRoadie.java:233)
	at org.springframework.test.context.junit4.SpringMethodRoadie$RunBeforesThenTestThenAfters.run(SpringMethodRoadie.java:333)
	at org.springframework.test.context.junit4.SpringMethodRoadie.runWithRepetitions(SpringMethodRoadie.java:217)
	at org.springframework.test.context.junit4.SpringMethodRoadie.runTest(SpringMethodRoadie.java:197)
	at org.springframework.test.context.junit4.SpringMethodRoadie.run(SpringMethodRoadie.java:143)
	at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.invokeTestMethod(SpringJUnit4ClassRunner.java:160)
	at org.junit.internal.runners.JUnit4ClassRunner.runMethods(JUnit4ClassRunner.java:51)
	at org.junit.internal.runners.JUnit4ClassRunner$1.run(JUnit4ClassRunner.java:44)
	at org.junit.internal.runners.ClassRoadie.runUnprotected(ClassRoadie.java:27)
	at org.junit.internal.runners.ClassRoadie.runProtected(ClassRoadie.java:37)
	at org.junit.internal.runners.JUnit4ClassRunner.run(JUnit4ClassRunner.java:42)
	at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.run(SpringJUnit4ClassRunner.java:97)
	at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:46)
	at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38)
	at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:467)
	at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:683)
	at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:390)
	at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:197)

 以上我们运行可以看出ClassCastException异常,因为我之前对biapAppService做了aop事务的拦截,所以从BeanFactory中获得的对象已经是代理对象[JDK动代理的对象]

 

<aop:config>
			<aop:advisor pointcut="execution(public * com.hhb.biap.service.impl.*.*(..))"
				advice-ref="biap_txAdvice" />
		</aop:config>

		<tx:advice id="biap_txAdvice" transaction-manager="biap_transactionManager" >
			<tx:attributes>
				<tx:method name="insert*" isolation="DEFAULT" propagation="REQUIRED" rollback-for="java.lang.Exception"/>
				<tx:method name="update*" isolation="DEFAULT" propagation="REQUIRED" rollback-for="java.lang.Exception"/>
				<tx:method name="delete*" isolation="DEFAULT" propagation="REQUIRED" rollback-for="java.lang.Exception"/>
				<tx:method name="get*" read-only="true" propagation="SUPPORTS" />
				<tx:method name="query*" read-only="true" propagation="SUPPORTS" />
				<tx:method name="*" isolation="DEFAULT" propagation="REQUIRED" rollback-for="java.lang.Exception"/>
			</tx:attributes>
		</tx:advice>

 所以会报错,现在我们将aop的配置下使之不会被spring拦截到。再运行,输出结果

 

 com.hhb.biap.service.impl.BiapAppServiceImpl@6d3b92
 com.hhb.biap.model.BiapApp@cd83d8

 从上面我们看出,并没有我想要输出的接口,因为我们没有调用增加一个增强

proxyFactory.addAdvice(new InvokeRecordBeforeAdvice())

 结果再运行,打印出了我们想要的结果。

 

com.hhb.biap.service.impl.BiapAppServiceImpl@17b1d64
invock method:com.hhb.biap.service.impl.BiapAppServiceImpl.toString
invock method:com.hhb.biap.service.impl.BiapAppServiceImpl.getBiapAppById
 com.hhb.biap.model.BiapApp@13a0212

 toString()怎么会调用呢,因为我们调用代理对象的toString()方法,这是我后面加上去的,

 

System.out.println("代理对象:"+ proxyBiapAppService);

 

 

这里Spring默认帮我们使用的是CGLIB代理机制,因为CGLIB代理拦截每个方法的调用

 而这一切是怎么做到的呢,接下来我们看ProxyFactory这个类里面做了什么事情......

public class ProxyFactory extends ProxyCreatorSupport implements AopProxy {
/**
  * Create a new ProxyFactory.
  */
 public ProxyFactory() {
 }
//.......
}

 我们看它父类的构造函数

 

public class ProxyCreatorSupport extends AdvisedSupport {

	/** The AopProxyFactory to use */
	private AopProxyFactory aopProxyFactory;

	/** List of AdvisedSupportListener */
	private List listeners = new LinkedList();

	/** Set to true when the first AOP proxy has been created */
	private boolean active = false;


	/**
	 * Create a new ProxyCreatorSupport instance.
	 */
	public ProxyCreatorSupport() {
		this.aopProxyFactory = new DefaultAopProxyFactory();
	}
//......

}

ProxyCreatorSupport 这个类我想我们很熟悉......

 它为我们默认初始化了一个DefaultAopProxyFactory对象,而这个对象负责代理对象创建。接下来我们看

 proxyFactory.getProxy()所干的事情

public Object getProxy() {
		return createAopProxy().getProxy();
	}

  接下来我们看createAopProxy()所干的事情,调用时父类ProxyCreatorSupport的方法

/**
	 * Subclasses should call this to get a new AOP proxy. They should <b>not</b>
	 * create an AOP proxy with <code>this</code> as an argument.
	 */
	protected final synchronized AopProxy createAopProxy() {
		if (!this.active) {
			activate();
		}
		return getAopProxyFactory().createAopProxy(this);
	}

 getAopProxyFactory()返回的就是spring默认给我们创建的一个对象,接下我们看看DefaultAopProxyFactory

这个类又是怎么回事

public class DefaultAopProxyFactory implements AopProxyFactory, Serializable {
          public AopProxy createAopProxy(AdvisedSupport config) throws AopConfigException {
		if (config.isOptimize() || config.isProxyTargetClass() || hasNoUserSuppliedProxyInterfaces(config)) {
			Class targetClass = config.getTargetClass();
			if (targetClass == null) {
				throw new AopConfigException("TargetSource cannot determine target class: " +
						"Either an interface or a target is required for proxy creation.");
			}
			if (targetClass.isInterface()) {
				return new JdkDynamicAopProxy(config);
			}
			if (!cglibAvailable) {
				throw new AopConfigException(
						"Cannot proxy target class because CGLIB2 is not available. " +
						"Add CGLIB to the class path or specify proxy interfaces.");
			}
			return CglibProxyFactory.createCglibProxy(config);
		}
		else {
			return new JdkDynamicAopProxy(config);
		}
	} 
}

 从上面我们可以看出,这个是真正创建代理对象的地方,而参数AdvisedSupport config就是我们ProxyFactory,因为ProxyFactory间接继承了AdvisedSupport

public class ProxyCreatorSupport extends AdvisedSupport {
//...
}

 分析下这个方法,

 

config.isOptimize()  这个表示是否启用优化,是否可以用cglib创建对象,从方法可以看出,如我们指定了创建代理对象使用的接口,

proxyFactory.setInterfaces(new Class[]{BiapAppService.class});

 

将将会覆盖cglib的代理,这个config.isProxyTargetClass()也是同理,表示是否能代理类对象。这个hasNoUserSuppliedProxyInterfaces(config) 。从我们的测试代码中可以看出,这个我们都没有指定,所以这个条件hasNoUserSuppliedProxyInterfaces(config) 满足,所以我们看到是打印对象的时候也被拦截了。

接下来我们分析前面2个条件为什么没有满足,我们看AdvisedSupport 的继承和实现

public class AdvisedSupport extends ProxyConfig implements Advised {
      //...
}

 我们再看

public class ProxyConfig implements Serializable {

	/** use serialVersionUID from Spring 1.2 for interoperability */
	private static final long serialVersionUID = -8409359707199703185L;


	private boolean proxyTargetClass = false;

	private boolean optimize = false;
                 //......
} 

 我想我们知道原因了。这就是spring美妙的设计给我们所带来的,我们只有去看看他们的源代码我们窥探到它有血有肉的地方。

最终的测试代码

public void testBeforeInterceptor() {
		//获得容器中的bean对象
		BiapAppServiceImpl biapAppService = (BiapAppServiceImpl) this.applicationContext
				.getBean("biapAppService");
		System.out.println(biapAppService);
		// 创建代理工厂
		ProxyFactory proxyFactory = new ProxyFactory();
//		proxyFactory.setOptimize(Boolean.TRUE);
//		proxyFactory.setProxyTargetClass(Boolean.TRUE);
		proxyFactory.setInterfaces(new Class[]{BiapAppService.class});
		proxyFactory.setTarget(biapAppService);
		proxyFactory.addAdvice(new InvokeRecordBeforeAdvice());
		//注意这里不能再用BiapAppServiceImpl转换,否则会报错,
		//因为我们返回的是代理对象
		BiapAppService proxyBiapAppService = (BiapAppService) proxyFactory
				.getProxy();
		System.out.println("代理对象:"+ proxyBiapAppService);
		BiapApp biapApp = proxyBiapAppService.getBiapAppById("biap");
		System.out.println(biapApp);
	}

 

接下来我们看看类的设计图纸:

其他类的图片

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值