有個接口TestServiceInter,有兩個實現方法TestService和Test2Service。他們都有sayHello();咱們的需求是在調用這兩個方法以前,要先完成寫日誌的功能;html
我在各個實現類的sayHello()方法裏面寫上寫日誌的功能就是了。spring
package com.jdc.aop;/** * @author DEllComputer * @Title: TestService * @ProjectName SpringAop * @Description: * @date 2018/12/251:38 PM */public interface TestService { /** * @Description: * @param ${tags} * @return ${return_type} * @throws * @author jdc * @date 2018/12/25 1:39 PM */ void sayHi(String name); }
public class TestServiceImpl implements TestService { private String name; /** * @Description: say hi * @param ${tags} * @return ${return_type} * @throws * @author jdc * @date 2018/12/25 1:40 PM */ @Override public void sayHi(String name ) { System.out.println("hi:" + name); } public String getName() { return name; } public void setName(String name) { this.name = name; } }
/** * @author DEllComputer * @Title: MyBeforeAdvice * @ProjectName SpringAop * @Description: 寫日誌前置通知 * @date 2018/12/251:42 PM */public class MyBeforeAdvice implements MethodBeforeAdvice { /** * @Description: 寫日誌的功能 * @param ${tags} * @return ${return_type} * @throws * @author jdc * @date 2018/12/25 1:44 PM */ @Override public void before(Method method, Object[] objects, Object o) throws Throwable { System.out.println("我是寫日誌的功能。"); } }
前置通知須要實現MethodBeforeAdvice接口,前置通知是在目標方法調用以前調用;ide
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> <!-- 配置前置通知 --> <bean id="myBeforeAdvice" class="com.jdc.aop.advice.MyBeforeAdvice"/> <!-- 配置被代理對象 --> <bean id="logTestServiceImpl" class="com.jdc.aop.TestServiceImpl"> <property name="name" value="Test"></property> </bean> <!-- 配置代理對象 --> <bean id="proxyFactoryBean" class="org.springframework.aop.framework.ProxyFactoryBean"> <!-- 代理接口集 配置哪些接口要被代理 --> <property name="proxyInterfaces"> <list> <value>com.jdc.aop.TestService</value> </list> </property> <!-- 把通知織入代理對象 --> <property name="interceptorNames"> <value>myBeforeAdvice</value> </property> <!-- 配置被代理的對象 --> <property name="target" ref="logTestServiceImpl"/> </bean> </beans>
ProxyFactoryBean是一個代理對象,若是咱們被代理的對象實現了接口,Spring使用的是jdk動態代理技術實現的動態代理;因此咱們要告訴代理對象,咱們的哪 些接口須要被代理,而後哪一個對象須要被代理,個人加強實現應該怎麼被織入到代理對象(前置,後置,環繞....)
public class TestMain { public static void main(String[] args) { ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml"); //獲取代理對象 TestService log = (TestService) ac.getBean("proxyFactoryBean"); log.sayHi("哈哈"); } }
注意這裏,咱們須要獲取代理對象,而不是目標對象,否則不能調用前置通知的代碼。測試