国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 學院 > 開發設計 > 正文

Spring面向切面 --- AspectJ --- 簡單使用

2019-11-14 21:03:09
字體:
來源:轉載
供稿:網友
SPRing面向切面 --- aspectJ --- 簡單使用Spring面向切面 --- AspectJ --- 簡單使用

昨天回復說說的時候突然寫下了下面的一段話:分享一下:

<!--*******************************************

其實經過的記憶是可以進行道德化的篡改的,就像夏目漱石的《我是貓》;但是不管怎么改,真正的事實是由每一個人的碎片拼起來的;經濟學里計算成本的是在計算將來的成本而不是過去的成本,就像動漫《未來日記》一樣;過去發生的事情總是在影響著將來,但是過去發生的事情卻不能充當將來下一個操作符的影響要素。********************************************-->

面向切面編程用到了動態代理,感興趣的讀者可以參考我的日志:

http://m.survivalescaperooms.com/kodoyang/p/DesignPattern_DynamicProxy.html

1.AspectJ - AOP面向切面編程是面向對象的一個補充

在保存用戶前添加一個日志,再加上時刻的記錄??傊?,就是方法前后加一點業務邏輯

新建資源庫:Preference>java>BuildPath>UserLibraries,AddLibrary>UserLibrary

/Spring_AOP/src/yuki/spring/aop/imitate/UserService.java

package yuki.spring.aop.imitate;public interface UserService {    void save();}

/Spring_AOP/src/yuki/spring/aop/imitate/UserServiceImpl.java

package yuki.spring.aop.imitate;public class UserServiceImpl implements UserService{    public void save(){        System.out.println("user saved...");    }}

/Spring_AOP/src/yuki/spring/aop/imitate/Intercepter.java

package yuki.spring.aop.imitate;import java.lang.reflect.InvocationHandler;import java.lang.reflect.Method;public abstract class Intercepter implements InvocationHandler {    private Object target;    public void setTarget(Object target) {        this.target = target;    }        protected abstract void beforeMethod();    protected abstract void afterMethod();        @Override    public Object invoke(Object proxy, Method m, Object[] args)            throws Throwable {        beforeMethod();        m.invoke(target, args);        afterMethod();        return null;    }}

/Spring_AOP/src/yuki/spring/aop/imitate/LogIntercepter.java

package yuki.spring.aop.imitate;public class LogIntercepter extends Intercepter {    @Override    public void beforeMethod() {        System.out.println("log start...");    }    @Override    protected void afterMethod() {        System.out.println("log end...");    }}

/Spring_AOP/src/yuki/spring/aop/imitate/TimeIntercepter.java

package yuki.spring.aop.imitate;public class TimeIntercepter extends Intercepter {    @Override    protected void beforeMethod() {        System.out.println("start:" + System.currentTimeMillis());    }    @Override    protected void afterMethod() {        System.out.println("end:" + System.currentTimeMillis());    }}

/Spring_AOP/src/yuki/spring/aop/imitate/ProxyTest.java

package yuki.spring.aop.imitate;import java.lang.reflect.Proxy;public class ProxyTest {    public static void main(String[] args) {        UserService service = new UserServiceImpl();                Intercepter[] intercepters = {new LogIntercepter(), new TimeIntercepter()};        for(Intercepter intercepter : intercepters){            intercepter.setTarget(service);            service = (UserService) Proxy.newProxyInstance(                    service.getClass().getClassLoader(),                     service.getClass().getInterfaces(), intercepter);        }                service.save();            }}

上面的程序中,兩個類繼承了實現InvocationHandler接口的抽象類,

也可以說是間接地實現了InvocationHandler接口;

聲明為UserService接口類型的service,

在循環中每一次接收Proxy.newProxyInstance方法的計算結果后就會改變它指向的對象。

運行結果如下:

start:1409496143248log start...user saved...log end...end:1409496143250

<aop:aspectj-autoproxy></aop:aspectj-autoproxy>

注解的方式實現AOP使用的是aspectj,aspectj是專門用來產生動態代理的一個框架可以使用aspectj注解的方式實現spring的AOP,把這個邏輯織入到原來的邏輯里面去

引入jar:aspectjrt.jar、aspectj-weaver.jar:http://www.java2s.com/Code/Jar/a/aspectj.htm

在類上添加注解 @Aspect和 @Component,在方法上添加注解 @Before切入點語法 @Before("execution(public void yuki.spring.aop.annotation"+ ".UserServiceImpl.save(yuki.spring.aop.annotation.User))")引入jar:aopalliance-.jar:http://www.java2s.com/Code/Jar/a/Downloadaopalliancejar.htm@Component要加在對應的實現類上,因為getBean后要獲取這個對象

/Spring_AOP/src/yuki/spring/aop/annotation/User.java

package yuki.spring.aop.annotation;public class User {}

/Spring_AOP/src/yuki/spring/aop/annotation/UserService.java

package yuki.spring.aop.annotation;public interface UserService {    void save(User user);}

/Spring_AOP/src/yuki/spring/aop/annotation/UserServiceImpl.java

package yuki.spring.aop.annotation;import org.springframework.stereotype.Component;@Component("userService")public class UserServiceImpl implements UserService{    public void save(User user){        System.out.println("user saved...");    }}

/Spring_AOP/src/annotation.xml

<?xml version="1.0" encoding="UTF-8"?><beans xmlns:aop="http://www.springframework.org/schema/aop"    xmlns:context="http://www.springframework.org/schema/context"     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-4.0.xsd                         http://www.springframework.org/schema/aop                         http://www.springframework.org/schema/aop/spring-aop-4.0.xsd                         http://www.springframework.org/schema/context                         http://www.springframework.org/schema/context/spring-context-4.0.xsd ">        <context:annotation-config></context:annotation-config>    <context:component-scan base-package="yuki.spring.aop.annotation"></context:component-scan>        <aop:aspectj-autoproxy></aop:aspectj-autoproxy>    </beans>

/Spring_AOP/src/yuki/spring/aop/annotation/LogIntercepter.java

package yuki.spring.aop.annotation;import org.aspectj.lang.annotation.Aspect;import org.aspectj.lang.annotation.Before;import org.springframework.stereotype.Component;@Aspect@Componentpublic class LogIntercepter {    @Before("execution(public void yuki.spring.aop.annotation"            + ".UserServiceImpl.save(yuki.spring.aop.annotation.User))")    public void beforeMethod() {        System.out.println("method start...");    }}

/Spring_AOP/src/yuki/spring/aop/annotation/TimeIntercepter.java

package yuki.spring.aop.annotation;import org.aspectj.lang.annotation.AfterReturning;import org.aspectj.lang.annotation.Aspect;import org.aspectj.lang.annotation.Before;import org.springframework.stereotype.Component;@Aspect@Componentpublic class TimeIntercepter {    @Before("execution("            + "public * yuki.spring.aop.annotation..*.*(..)"            + ")")    public void beforeMethod() {        System.out.println("before:" + System.currentTimeMillis());    }    @AfterReturning("execution( public * yuki.spring.aop.annotation..*.*(..) )")    public void afterReturningMethod() {        System.out.println("afterReturning:" + System.currentTimeMillis());    }}

/Spring_AOP/src/yuki/spring/aop/annotation/PointcutIntercepter.java

package yuki.spring.aop.annotation;import org.aspectj.lang.annotation.AfterReturning;import org.aspectj.lang.annotation.Aspect;import org.aspectj.lang.annotation.Before;import org.aspectj.lang.annotation.Pointcut;import org.springframework.stereotype.Component;@Aspect@Componentpublic class PointcutIntercepter {    @Pointcut("execution( public * yuki.spring.aop.annotation..*.*(..) )")    public void myMethod() {}        @Before("myMethod()")    public void beforeMethod() {        System.out.println("PointcutIntercepter::before");    }    @AfterReturning("myMethod()")    public void afterReturningMethod() {        System.out.println("PointcutIntercepter::afterReturning");    }}

/Spring_AOP/src/yuki/spring/aop/annotation/UserServiceTest.java

package yuki.spring.aop.annotation;import org.junit.Test;import org.springframework.context.support.ClassPathXmlapplicationContext;public class UserServiceTest {    @Test    public void testSave() {                @SuppressWarnings("resource")        ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("annotation.xml");        UserService service = (UserService) context.getBean("userService");        service.save(new User());        context.destroy();    }}

運行結果如下:

八月 31, 2014 10:59:10 下午 org.springframework.context.support.ClassPathXmlApplicationContext prepareRefresh信息: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@1637f22: startup date [Sun Aug 31 22:59:10 CST 2014]; root of context hierarchy八月 31, 2014 10:59:10 下午 org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions信息: Loading XML bean definitions from class path resource [annotation.xml]method start...PointcutIntercepter::beforebefore:1409497152268user saved...afterReturning:1409497152268PointcutIntercepter::afterReturning八月 31, 2014 10:59:12 下午 org.springframework.context.support.ClassPathXmlApplicationContext doClose信息: Closing org.springframework.context.support.ClassPathXmlApplicationContext@1637f22: startup date [Sun Aug 31 22:59:10 CST 2014]; root of context hierarchy

術語解釋:JoinPoint:連接點、切入點,在哪里把邏輯加進區PointCut:JoinPoint的集合,可以一次性定義好多切入點Aspect:切面,切面類加進去的邏輯可以認為是一個切面Advice:在這個點上建議怎么辦,比如 @BeforeTarget:被織入邏輯的被代理對象Weave:織入

2.切面和通知切入點:// the execution of any public method:execution(public * *(..))// the execution of any method with a name beginning with "set":execution(* set*(..))// the execution of any method defined by the AccountService interface:execution(* com.xyz.service.AccountService.*(..))// the execution of any method defined in the service package:execution(* com.xyz.service..(..))// the execution of any method defined in the service package or a sub-package:execution(* com.xyz.service..*.*(..))spring也定義了自己的織入點語法,但是我們只要用AspectJ的語法就可以了

通知:Before、AfterReturn、AfterThrowing、After(意思是finally)Around:在ProceedingJoinPoint.proceed()前后加邏輯

如果織入點表達式相同,可以定義Pointcut定義方法名接收織入點表達式的值,使用時 @AfterReturning("pointcut()")

/Spring_AOP/src/yuki/spring/aop/pointcut/User.java

package yuki.spring.aop.pointcut;public class User {}

/Spring_AOP/src/yuki/spring/aop/pointcut/UserService.java

package yuki.spring.aop.pointcut;public interface UserService {    void save(User user);}

/Spring_AOP/src/yuki/spring/aop/pointcut/UserServiceImpl.java

package yuki.spring.aop.pointcut;import org.springframework.stereotype.Component;@Component("userService")public class UserServiceImpl implements UserService{    public void save(User user){        System.out.println("user saved...");        //throw new RuntimeException("exception...");    }}

/Spring_AOP/src/yuki/spring/aop/pointcut/TransactionAspect.java

package yuki.spring.aop.pointcut;import org.aspectj.lang.ProceedingJoinPoint;import org.aspectj.lang.annotation.AfterReturning;import org.aspectj.lang.annotation.AfterThrowing;import org.aspectj.lang.annotation.Around;import org.aspectj.lang.annotation.Aspect;import org.aspectj.lang.annotation.Before;import org.aspectj.lang.annotation.Pointcut;import org.springframework.stereotype.Component;@Aspect@Componentpublic class TransactionAspect {    @Pointcut("execution( public * yuki.spring.aop.pointcut..*.*(..) )")    public void pointcut(){}        @Before("pointcut()")    public void before(){        System.out.println("before");    }    @AfterReturning("pointcut()")    public void afterReturning(){        System.out.println("afterReturning");    }    @AfterThrowing("pointcut()")    public void afterThrowing(){        System.out.println("afterThrowing");    }        @Around("pointcut()")    public void aroundMethod(ProceedingJoinPoint pjp) throws Throwable{        System.out.println("method around start...");        pjp.proceed();        System.out.println("method around end...");    }    }

/Spring_AOP/src/pointcut.xml

<?xml version="1.0" encoding="UTF-8"?><beans xmlns:aop="http://www.springframework.org/schema/aop"    xmlns:context="http://www.springframework.org/schema/context"     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-4.0.xsd                         http://www.springframework.org/schema/aop                         http://www.springframework.org/schema/aop/spring-aop-4.0.xsd                         http://www.springframework.org/schema/context                         http://www.springframework.org/schema/context/spring-context-4.0.xsd ">        <context:annotation-config></context:annotation-config>    <context:component-scan base-package="yuki.spring.aop.pointcut"></context:component-scan>        <aop:aspectj-autoproxy></aop:aspectj-autoproxy>    </beans>

/Spring_AOP/src/yuki/spring/aop/pointcut/UserServiceTest.java

package yuki.spring.aop.pointcut;import org.junit.Test;import org.springframework.context.support.ClassPathXmlApplicationContext;public class UserServiceTest {    @Test    public void testSave() {                @SuppressWarnings("resource")        ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("pointcut.xml");        UserService service = (UserService) context.getBean("userService");        service.save(new User());        context.destroy();    }}

運行結果如下:

八月 31, 2014 11:09:22 下午 org.springframework.context.support.ClassPathXmlApplicationContext prepareRefresh信息: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@1637f22: startup date [Sun Aug 31 23:09:22 CST 2014]; root of context hierarchy八月 31, 2014 11:09:22 下午 org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions信息: Loading XML bean definitions from class path resource [pointcut.xml]method around start...beforeuser saved...method around end...afterReturning八月 31, 2014 11:09:24 下午 org.springframework.context.support.ClassPathXmlApplicationContext doClose信息: Closing org.springframework.context.support.ClassPathXmlApplicationContext@1637f22: startup date [Sun Aug 31 23:09:22 CST 2014]; root of context hierarchy

可以實現聲明式的異常管理,struts2已經實現了聲明式的異常管理,這里略過通知執行的先后順序

3.cglib如果被代理的類實現了接口,就會使用JDK自帶的Proxy和InvocationHandler來實現代理當被代理的類沒有實現接口,它會用cglib直接操作二進制碼的形式來產生代理的代碼引入jar:cglib-2.2.jar:http://www.java2s.com/Code/Jar/c/Downloadcglib22jar.htm

/Spring_AOP/src/yuki/spring/aop/cglib/UserService.java

package yuki.spring.aop.cglib;import org.springframework.stereotype.Component;@Componentpublic class UserService {    public void save(){        System.out.println("user saved...");    }}

/Spring_AOP/src/yuki/spring/aop/cglib/CglibAspect.java

package yuki.spring.aop.cglib;import org.aspectj.lang.ProceedingJoinPoint;import org.aspectj.lang.annotation.Around;import org.aspectj.lang.annotation.Aspect;import org.aspectj.lang.annotation.Pointcut;import org.springframework.stereotype.Component;@Aspect@Componentpublic class CglibAspect {    @Pointcut("execution( public * yuki.spring.aop.cglib..*.*(..) )")    public void pointcut(){}        @Around("pointcut()")    public void aroundMethod(ProceedingJoinPoint pjp) throws Throwable{        System.out.println("method around start...");        pjp.proceed();        System.out.println("method around end...");    }    }

/Spring_AOP/src/cglib.xml

<?xml version="1.0" encoding="UTF-8"?><beans xmlns:aop="http://www.springframework.org/schema/aop"    xmlns:context="http://www.springframework.org/schema/context"     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-4.0.xsd                         http://www.springframework.org/schema/aop                         http://www.springframework.org/schema/aop/spring-aop-4.0.xsd                         http://www.springframework.org/schema/context                         http://www.springframework.org/schema/context/spring-context-4.0.xsd ">        <context:annotation-config></context:annotation-config>    <context:component-scan base-package="yuki.spring.aop.cglib"></context:component-scan>        <aop:aspectj-autoproxy></aop:aspectj-autoproxy>    </beans>

/Spring_AOP/src/yuki/spring/aop/cglib/UserServiceTest.java

package yuki.spring.aop.cglib;import org.junit.Test;import org.springframework.context.support.ClassPathXmlApplicationContext;public class UserServiceTest {    @Test    public void testSave() {                @SuppressWarnings("resource")        ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("cglib.xml");        UserService service = (UserService) context.getBean("userService");        System.out.println(service.getClass());        service.save();        context.destroy();    }}

運行結果如下:

八月 31, 2014 11:14:17 下午 org.springframework.context.support.ClassPathXmlApplicationContext prepareRefresh信息: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@1637f22: startup date [Sun Aug 31 23:14:17 CST 2014]; root of context hierarchy八月 31, 2014 11:14:17 下午 org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions信息: Loading XML bean definitions from class path resource [cglib.xml]class yuki.spring.aop.cglib.UserService$$EnhancerBySpringCGLIB$$b8ea6837method around start...user saved...method around end...八月 31, 2014 11:14:19 下午 org.springframework.context.support.ClassPathXmlApplicationContext doClose信息: Closing org.springframework.context.support.ClassPathXmlApplicationContext@1637f22: startup date [Sun Aug 31 23:14:17 CST 2014]; root of context hierarchy

4.xml:AOPpointcut可以聲明在aspect的里面或外面

eclipse:從源碼視圖切換到設計視圖

在執行UserService.save()時,發現符合配置的切入點表達式對應的是LogAspect.before(),于是先執行before,然后save()可以直接指定pointcut,也可以在外部指定然后再引用它如果使用第三方類的切面類邏輯,那么就必須要使用xml配置的方式

/Spring_AOP/src/yuki/spring/aop/xml/UserService.java

package yuki.spring.aop.xml;import org.springframework.stereotype.Component;@Component("userService")public class UserService {    public void save(){        System.out.println("user saved...");    }}

/Spring_AOP/src/yuki/spring/aop/xml/LogAspect.java

package yuki.spring.aop.xml;public class LogAspect {    public void before() {        System.out.println("method start...");    }}

/Spring_AOP/src/xml.xml

<?xml version="1.0" encoding="UTF-8"?><beans xmlns:aop="http://www.springframework.org/schema/aop"    xmlns:context="http://www.springframework.org/schema/context"     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-4.0.xsd                         http://www.springframework.org/schema/aop                         http://www.springframework.org/schema/aop/spring-aop-4.0.xsd                         http://www.springframework.org/schema/context                         http://www.springframework.org/schema/context/spring-context-4.0.xsd ">        <context:annotation-config></context:annotation-config>    <context:component-scan base-package="yuki.spring.aop.xml"></context:component-scan>        <bean id="logAspect" class="yuki.spring.aop.xml.LogAspect"></bean>    <!--     <aop:config>        <aop:pointcut id="servicePointcut"                expression="execution( public * yuki.spring.aop.xml..*.*(..) )" />        <aop:aspect id="logAspect" ref="logAspect">            <aop:before method="before" pointcut-ref="servicePointcut"/>        </aop:aspect>    </aop:config>     -->    <aop:config>        <aop:aspect id="logAspect" ref="logAspect">            <aop:before method="before"                     pointcut="execution( public * yuki.spring.aop.xml..*.*(..) )"/>        </aop:aspect>    </aop:config></beans>

/Spring_AOP/src/yuki/spring/aop/xml/UserServiceTest.java

package yuki.spring.aop.xml;import org.junit.Test;import org.springframework.context.support.ClassPathXmlApplicationContext;public class UserServiceTest {    @Test    public void testSave() {                @SuppressWarnings("resource")        ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("xml.xml");        UserService service = (UserService) context.getBean("userService");        service.save();        context.destroy();    }}

運行結果如下:

八月 31, 2014 11:19:01 下午 org.springframework.context.support.ClassPathXmlApplicationContext prepareRefresh信息: Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@1637f22: startup date [Sun Aug 31 23:19:01 CST 2014]; root of context hierarchy八月 31, 2014 11:19:01 下午 org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions信息: Loading XML bean definitions from class path resource [xml.xml]method start...user saved...八月 31, 2014 11:19:02 下午 org.springframework.context.support.ClassPathXmlApplicationContext doClose信息: Closing org.springframework.context.support.ClassPathXmlApplicationContext@1637f22: startup date [Sun Aug 31 23:19:01 CST 2014]; root of context hierarchy

5.tomcat debug的熱部署在tomcat的jdk虛擬機參數中添加-Dcom.sun.management.jmxremote=true

如果修改配置文件,使用了自定義標簽的jsp頁面,修改了注解,等等情況:還是要重啟服務器的

在方法內部修改代碼,不用重啟服務器,這已經是很大的便捷了,

有興趣的小伙伴們去研究功能更強大的熱部署吧。。。。

目錄結構:

本文參考了[尚學堂馬士兵_Spring_AOP]的公開課程

更多好文請關注:http://m.survivalescaperooms.com/kodoyang/

>*_*<

kongdongyang

2014/8/31


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 阿尔山市| 克山县| 桓台县| 当阳市| 左云县| 和硕县| 三台县| 丹凤县| 龙门县| 潞西市| 中宁县| 门头沟区| 长治县| 金昌市| 揭西县| 舞阳县| 扎兰屯市| 平凉市| 弥勒县| 济宁市| 辛集市| 内乡县| 遂溪县| 独山县| 台北县| 镇平县| 义马市| 肥城市| 高邮市| 南汇区| 锡林郭勒盟| 治县。| 锡林浩特市| 鸡东县| 崇礼县| 瑞昌市| 合江县| 嘉黎县| 顺平县| 夏河县| 茌平县|