网站公告栏怎么做国外界面设计网站

bicheng/2025/10/13 22:26:44/文章来源:
网站公告栏怎么做,国外界面设计网站,amon wordpress,文件链接 win wordpress本文主要内容#xff1a;掌握Pointcut的12种用法。 Aop相关阅读 阅读本文之前#xff0c;需要先掌握下面3篇文章内容#xff0c;不然会比较吃力。 Spring系列第15篇#xff1a;代理详解#xff08;java动态代理CGLIB代理)Spring系列第30篇#xff1a;jdk动态代理…本文主要内容掌握Pointcut的12种用法。 Aop相关阅读 阅读本文之前需要先掌握下面3篇文章内容不然会比较吃力。 Spring系列第15篇代理详解java动态代理CGLIB代理)Spring系列第30篇jdk动态代理和cglib代理Spring系列第31篇Aop概念详解Spring系列第32篇AOP核心源码、原理详解Spring系列第33篇ProxyFactoryBean创建AOP代理 本文继续AOP目前手动Aop中三种方式已经介绍2种了本文将介绍另外一种AspectJProxyFactory可能大家对这个比较陌生但是Aspect这个注解大家应该很熟悉吧通过这个注解在spring环境中实现aop特别的方便。 而AspectJProxyFactory这个类可以通过解析Aspect标注的类来生成代理aop代理对象对开发者来说使创建代理变的更简洁了。 先了解几个概念 文中会涉及几个概念先了解一下。 target 用来表示目标对象即需要通过aop来增强的对象。 proxy 代理对象target通过aop增强之后生成的代理对象。 AspectJ AspectJ是什么? AspectJ是一个面向切面的框架是目前最好用最方便的AOP框架和spring中的aop可以集成在一起使用通过Aspectj提供的一些功能实现aop代理变得非常方便。 AspectJ使用步骤 1.创建一个类使用Aspect标注 2.Aspect标注的类中通过Pointcut定义切入点 3.Aspect标注的类中通过AspectJ提供的一些通知相关的注解定义通知 4.使用AspectJProxyFactory结合Ascpect标注的类来生成代理对象先来个案例感受一下AspectJ是多么的方便。 来个类 package com.javacode2018.aop.demo9.test1;public class Service1 {public void m1() {System.out.println(我是 m1 方法);}public void m2() {System.out.println(10 / 0);System.out.println(我是 m2 方法);} }通过AspectJ来对Service1进行增强来2个通知一个前置通知一个异常通知这2个通知需要对Service1中的所有方法生效实现如下 package com.javacode2018.aop.demo9.test1;import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.AfterThrowing; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.annotation.Pointcut;//1这个类需要使用Aspect进行标注 Aspect public class Aspect1 {//2定义了一个切入点可以匹配Service1中所有方法Pointcut(execution(* com.javacode2018.aop.demo9.test1.Service1.*(..)))public void pointcut1() {}//3定义了一个前置通知这个通知对刚刚上面我们定义的切入点中的所有方法有效Before(value pointcut1())public void before(JoinPoint joinPoint) {//输出连接点的信息System.out.println(前置通知 joinPoint);}//4定义了一个异常通知这个通知对刚刚上面我们定义的切入点中的所有方法有效AfterThrowing(value pointcut1(), throwing e)public void afterThrowing(JoinPoint joinPoint, Exception e) {//发生异常之后输出异常信息System.out.println(joinPoint ,发生异常 e.getMessage());}}1类上使用Aspect标注 2通过Pointcut注解标注在方法上面用来定义切入点 3使用Before标注在方法上面定义了一个前置通知通过value引用了上面已经定义的切入点表示这个通知会对Service1中的所有方法生效在通知中可以通过这个类名.方法名()引用Pointcut定义的切入点表示这个通知对这些切入点有效若Before和Pointcut在一个类的时候直接通过方法名()引用当前类中定义的切入点 4这个使用AfterThrowing定义了一个异常通知也是对通过value引用了上面已经定义的切入点表示这个通知会对Service1中的所有方法生效若Service1中的方法抛出了Exception类型的异常都会回调afterThrowing方法。 来个测试类 package com.javacode2018.aop.demo9;import com.javacode2018.aop.demo9.test1.Aspect1; import com.javacode2018.aop.demo9.test1.Service1; import org.junit.Test; import org.springframework.aop.aspectj.annotation.AspectJProxyFactory;public class AopTest9 {Testpublic void test1() {try {//对应目标对象Service1 target new Service1();//创建AspectJProxyFactory对象AspectJProxyFactory proxyFactory new AspectJProxyFactory();//设置被代理的目标对象proxyFactory.setTarget(target);//设置标注了Aspect注解的类proxyFactory.addAspect(Aspect1.class);//生成代理对象Service1 proxy proxyFactory.getProxy();//使用代理对象proxy.m1();proxy.m2();} catch (Exception e) {}} }运行输出 前置通知execution(void com.javacode2018.aop.demo9.test1.Service1.m1()) 我是 m1 方法 前置通知execution(void com.javacode2018.aop.demo9.test1.Service1.m2()) execution(void com.javacode2018.aop.demo9.test1.Service1.m2()),发生异常/ by zero使用是不是特方便。 AspectJProxyFactory原理 Aspect标注的类上这个类中可以通过通过Pointcut来定义切入点可以通过Before、Around、After、AfterRunning、AfterThrowing标注在方法上来定义通知定义好了之后将Aspect标注的这个类交给AspectJProxyFactory来解析生成Advisor链进而结合目标对象一起来生成代理对象大家可以去看一下源码比较简单这里就不多解释了。 本文的重点在Aspect标注的类上Aspect中有2个关键点比较重要 Pointcut标注在方法上用来定义切入点有11种用法本文主要讲解这11种用法。Aspect类中定义通知可以通过Before、Around、After、AfterRunning、AfterThrowing标注在方法上来定义通知这个下一篇介绍。 Pointcut的12种用法 作用 用来标注在方法上来定义切入点。 定义 格式 注解(value“表达标签 (表达式格式)”) 如 Pointcut(execution(* com.javacode2018.aop.demo9.test1.Service1.*(..)))表达式标签10种 execution用于匹配方法执行的连接点within用于匹配指定类型内的方法执行this用于匹配当前AOP代理对象类型的执行方法注意是AOP代理对象的类型匹配这样就可能包括引入接口也类型匹配target用于匹配当前目标对象类型的执行方法注意是目标对象的类型匹配这样就不包括引入接口也类型匹配args用于匹配当前执行的方法传入的参数为指定类型的执行方法within用于匹配所以持有指定注解类型内的方法target用于匹配当前目标对象类型的执行方法其中目标对象持有指定的注解args用于匹配当前执行的方法传入的参数持有指定注解的执行annotation用于匹配当前执行方法持有指定注解的方法beanSpring AOP扩展的AspectJ没有对于指示符用于匹配特定名称的Bean对象的执行方法 10种标签组成了12种用法 1、execution 使用execution(方法表达式)匹配方法执行。 execution格式 execution(modifiers-pattern? ret-type-pattern declaring-type-pattern? name-pattern(param-pattern) throws-pattern?)其中带 ?号的 modifiers-pattern?declaring-type-pattern?hrows-pattern?是可选项ret-type-pattern,name-pattern, parameters-pattern是必选项modifier-pattern? 修饰符匹配如public 表示匹配公有方法ret-type-pattern 返回值匹配* 表示任何返回值全路径的类名等declaring-type-pattern? 类路径匹配name-pattern 方法名匹配* 代表所有set*代表以set开头的所有方法(param-pattern) 参数匹配指定方法参数(声明的类型)(…)代表所有参数(*,String)代表第一个参数为任何值,第二个为String类型(…,String)代表最后一个参数是String类型throws-pattern? 异常类型匹配 举例说明 类型匹配语法 很多地方会按照类型的匹配先来说一下类型匹配的语法。 首先让我们来了解下AspectJ类型匹配的通配符 *匹配任何数量字符..匹配任何数量字符的重复如在类型模式中匹配任何数量子包而在方法参数模式中匹配任何数量参数0个或者多个参数匹配指定类型及其子类型仅能作为后缀放在类型模式后边 2、within 用法 within(类型表达式)目标对象target的类型是否和within中指定的类型匹配 匹配原则 target.getClass().equals(within表达式中指定的类型)案例 有2个类父子关系 父类C1 package com.javacode2018.aop.demo9.test2;public class C1 {public void m1() {System.out.println(我是m1);}public void m2() {System.out.println(我是m2);} }子类C2 package com.javacode2018.aop.demo9.test2;public class C2 extends C1 {Overridepublic void m2() {super.m2();}public void m3() {System.out.println(我是m3);} }来个Aspect类 package com.javacode2018.aop.demo9.test2;import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.annotation.Pointcut;Aspect public class AspectTest2 {Pointcut(within(C1)) //1public void pc() {}Before(pc()) //2public void beforeAdvice(JoinPoint joinpoint) {System.out.println(joinpoint);}}注意1匹配的类型是C1也就是说被代理的对象的类型必须是C1类型的才行需要和C1完全匹配 下面我们对C2创建代理 Test public void test2(){C2 target new C2();AspectJProxyFactory proxyFactory new AspectJProxyFactory();proxyFactory.setTarget(target);proxyFactory.addAspect(AspectTest2.class);C2 proxy proxyFactory.getProxy();proxy.m1();proxy.m2();proxy.m3(); }运行输出 我是m1 我是m2 我是m3原因是目标对象是C2类型的C2虽然是C1的子类但是within中表达式指定的是要求类型必须是C1类型的才匹配。 如果将within表达式修改为下面任意一种就可以匹配了 Pointcut(within(C1)) Pointcut(within(C2)) 再次运行输出 execution(void com.javacode2018.aop.demo9.test2.C1.m1()) 我是m1 execution(void com.javacode2018.aop.demo9.test2.C2.m2()) 我是m2 execution(void com.javacode2018.aop.demo9.test2.C2.m3()) 我是m33、this 用法 this(类型全限定名)通过aop创建的代理对象的类型是否和this中指定的类型匹配注意判断的目标是代理对象this中使用的表达式必须是类型全限定名不支持通配符。 匹配原则 如:this(x)则代理对象proxy满足下面条件时会匹配 x.getClass().isAssignableFrom(proxy.getClass());案例 来个接口 package com.javacode2018.aop.demo9.test3;public interface I1 {void m1(); }来个实现类 package com.javacode2018.aop.demo9.test3;public class Service3 implements I1 {Overridepublic void m1() {System.out.println(我是m1);}}来个Aspect类 package com.javacode2018.aop.demo9.test3;import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.annotation.Pointcut;Aspect public class AspectTest3 {//1匹配proxy是Service3类型的所有方法Pointcut(this(Service3))public void pc() {}Before(pc())public void beforeAdvice(JoinPoint joinpoint) {System.out.println(joinpoint);}}测试代码 Test public void test3() {Service3 target new Service3();AspectJProxyFactory proxyFactory new AspectJProxyFactory();proxyFactory.setTarget(target);//获取目标对象上的接口列表Class?[] allInterfaces ClassUtils.getAllInterfaces(target);//设置需要代理的接口proxyFactory.setInterfaces(allInterfaces);proxyFactory.addAspect(AspectTest3.class);//获取代理对象Object proxy proxyFactory.getProxy();//调用代理对象的方法((I1) proxy).m1();System.out.println(proxy是否是jdk动态代理对象 AopUtils.isJdkDynamicProxy(proxy));System.out.println(proxy是否是cglib代理对象 AopUtils.isCglibProxy(proxy));//判断代理对象是否是Service3类型的System.out.println(Service3.class.isAssignableFrom(proxy.getClass())); }运行输出 我是m1 proxy是否是jdk动态代理对象true proxy是否是cglib代理对象false false从输出中可以看出m1方法没有被增强原因this表达式要求代理对象必须是Service3类型的输出中可以看出代理对象并不是Service3类型的此处代理对象proxy是使用jdk动态代理生成的。 我们可以将代码调整一下使用cglib来创建代理 proxyFactory.setProxyTargetClass(true);再次运行会发现m2被拦截了结果如下 execution(void com.javacode2018.aop.demo9.test3.Service3.m1()) 我是m1 proxy是否是jdk动态代理对象false proxy是否是cglib代理对象true true4、target 用法 target(类型全限定名)判断目标对象的类型是否和指定的类型匹配注意判断的是目标对象的类型表达式必须是类型全限定名不支持通配符。 匹配原则 如:target(x)则目标对象target满足下面条件时会匹配 x.getClass().isAssignableFrom(target.getClass());案例 package com.javacode2018.aop.demo9.test4;import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.annotation.Pointcut;Aspect public class AspectTest4 {//1目标类型必须是Service3类型的Pointcut(target(com.javacode2018.aop.demo9.test3.Service3))public void pc() {}Before(pc())public void beforeAdvice(JoinPoint joinpoint) {System.out.println(joinpoint);}}测试代码 Test public void test4() {Service3 target new Service3();AspectJProxyFactory proxyFactory new AspectJProxyFactory();proxyFactory.setProxyTargetClass(true);proxyFactory.setTarget(target);proxyFactory.addAspect(AspectTest4.class);//获取代理对象Object proxy proxyFactory.getProxy();//调用代理对象的方法((I1) proxy).m1();//判断target对象是否是Service3类型的System.out.println(Service3.class.isAssignableFrom(target.getClass())); }运行输出 execution(void com.javacode2018.aop.demo9.test3.Service3.m1()) 我是m1 truewithin、this、target对比 5、args 用法 args(参数类型列表)匹配当前执行的方法传入的参数是否为args中指定的类型注意是匹配传入的参数类型不是匹配方法签名的参数类型参数类型列表中的参数必须是类型全限定名不支持通配符args属于动态切入点也就是执行方法的时候进行判断的这种切入点开销非常大非特殊情况最好不要使用。 举例说明 案例 下面的m1方法参数是Object类型的。 package com.javacode2018.aop.demo9.test5;public class Service5 {public void m1(Object object) {System.out.println(我是m1方法,参数 object);} }Aspect类 package com.javacode2018.aop.demo9.test5;import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.annotation.Pointcut;import java.util.Arrays; import java.util.stream.Collectors;Aspect public class AspectTest5 {//1匹配只有1个参数其类型是String类型的Pointcut(args(String))public void pc() {}Before(pc())public void beforeAdvice(JoinPoint joinpoint) {System.out.println(请求参数 Arrays.stream(joinpoint.getArgs()).collect(Collectors.toList()));} }测试代码调用2次m1方法第一次传入一个String类型的第二次传入一个int类型的看看效果 Test public void test5() {Service5 target new Service5();AspectJProxyFactory proxyFactory new AspectJProxyFactory();proxyFactory.setTarget(target);proxyFactory.addAspect(AspectTest5.class);Service5 proxy proxyFactory.getProxy();proxy.m1(路人);proxy.m1(100); }运行输出 请求参数[路人] 我是m1方法,参数路人 我是m1方法,参数100输出中可以看出m1第一次调用被增强了第二次没有被增强。 args会在调用的过程中对参数实际的类型进行匹配比较耗时慎用。 6、within 用法 within(注解类型)匹配指定的注解内定义的方法。 匹配规则 调用目标方法的时候通过java中Method.getDeclaringClass()获取当前的方法是哪个类中定义的然后会看这个类上是否有指定的注解。 被调用的目标方法Method对象.getDeclaringClass().getAnnotation(within中指定的注解类型) ! null来看3个案例。 案例1 目标对象上有within中指定的注解这种情况时目标对象的所有方法都会被拦截。 来个注解 package com.javacode2018.aop.demo9.test9;import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target;Retention(RetentionPolicy.RUNTIME) Target(ElementType.TYPE) public interface Ann9 { }来个目标类用Ann9标注 package com.javacode2018.aop.demo9.test9;Ann9 public class S9 {public void m1() {System.out.println(我是m1方法);} }来个Aspect类 package com.javacode2018.aop.demo9.test9;import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.annotation.Pointcut;Aspect public class AspectTest9 {/*** 定义目标方法的类上有Ann9注解*/Pointcut(within(Ann9))public void pc() {}Before(pc())public void beforeAdvice(JoinPoint joinPoint) {System.out.println(joinPoint);} }测试代码 Test public void test9() {S9 target new S9();AspectJProxyFactory proxyFactory new AspectJProxyFactory();proxyFactory.setTarget(target);proxyFactory.addAspect(AspectTest9.class);S9 proxy proxyFactory.getProxy();proxy.m1(); }m1方法在类S9中定义的S9上面有Ann9注解所以匹配成功 运行输出 execution(void com.javacode2018.aop.demo9.test9.S9.m1()) 我是m1方法案例2 定义注解时未使用Inherited说明子类无法继承父类上的注解这个案例中我们将定义一个这样的注解将注解放在目标类的父类上来看一下效果。 定义注解Ann10 package com.javacode2018.aop.demo9.test10;import java.lang.annotation.*;Retention(RetentionPolicy.RUNTIME) Target(ElementType.TYPE) Inherited public interface Ann10 { }来2个父子类 注意 S10Parent为父类并且使用了Anno10注解内部定义了2个方法大家注意一下 而S10位代理的目标类继承了S10Parent内部重写了父类的m2方法并且又新增了一个m3方法 package com.javacode2018.aop.demo9.test10;Ann10 class S10Parent {public void m1() {System.out.println(我是S10Parent.m1()方法);}public void m2() {System.out.println(我是S10Parent.m2()方法);} }public class S10 extends S10Parent {Overridepublic void m2() {System.out.println(我是S10.m2()方法);}public void m3() {System.out.println(我是S10.m3()方法);} }来个Aspect类 package com.javacode2018.aop.demo9.test10;import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.annotation.Pointcut;Aspect public class AspectTest10 {//匹配目标方法声明的类上有Anno10注解Pointcut(within(com.javacode2018.aop.demo9.test10.Ann10))public void pc() {}Before(pc())public void beforeAdvice(JoinPoint joinPoint) {System.out.println(joinPoint);} }测试用例 S10为目标类依次执行代理对象的m1、m2、m3方法最终会调用目标类target中对应的方法。 Test public void test10() {S10 target new S10();AspectJProxyFactory proxyFactory new AspectJProxyFactory();proxyFactory.setTarget(target);proxyFactory.addAspect(AspectTest10.class);S10 proxy proxyFactory.getProxy();proxy.m1();proxy.m2();proxy.m3(); }运行输出 execution(void com.javacode2018.aop.demo9.test10.S10Parent.m1()) 我是S10Parent.m1()方法 我是S10.m2()方法 我是S10.m3()方法分析结果 从输出中可以看出只有m1方法被拦截了其他2个方法没有被拦截。 确实是这样的m1方法的是由S10Parent定义的这个类上面有Ann10注解。 而m2方法虽然也在S10Parent中定义了但是这个方法被子类S10重写了所以调用目标对象中的m2方法的时候此时发现m2方法是由S10定义的而S10.class.getAnnotation(Ann10.class)为空所以这个方法不会被拦截。 同样m3方法也是S10中定义的也不会被拦截。 案例3 对案例2进行改造在注解的定义上面加上Inherited此时子类可以继承父类的注解此时3个方法都会被拦截了。 下面上代码下面代码为案例2代码的一个拷贝不同地方只是注解的定义上多了Inherited 定义注解Ann11 import java.lang.annotation.*;Retention(RetentionPolicy.RUNTIME) Target(ElementType.TYPE) Inherited public interface Ann11 { }2个父子类 package com.javacode2018.aop.demo9.test11;Ann11 class S11Parent {public void m1() {System.out.println(我是S11Parent.m1()方法);}public void m2() {System.out.println(我是S11Parent.m2()方法);} }public class S11 extends S11Parent {Overridepublic void m2() {System.out.println(我是S11.m2()方法);}public void m3() {System.out.println(我是S11.m3()方法);} }Aspect类 package com.javacode2018.aop.demo9.test11;import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.annotation.Pointcut;Aspect public class AspectTest11 {Pointcut(within(com.javacode2018.aop.demo9.test11.Ann11))public void pc() {}Before(pc())public void beforeAdvice(JoinPoint joinPoint) {System.out.println(joinPoint);} }测试用例 Test public void test11() {S11 target new S11();AspectJProxyFactory proxyFactory new AspectJProxyFactory();proxyFactory.setTarget(target);proxyFactory.addAspect(AspectTest11.class);S11 proxy proxyFactory.getProxy();proxy.m1();proxy.m2();proxy.m3(); }运行输出 execution(void com.javacode2018.aop.demo9.test11.S11Parent.m1()) 我是S11Parent.m1()方法 execution(void com.javacode2018.aop.demo9.test11.S11.m2()) 我是S11.m2()方法 execution(void com.javacode2018.aop.demo9.test11.S11.m3()) 我是S11.m3()方法这次3个方法都被拦截了。 7、target 用法 target(注解类型)判断目标对象target类型上是否有指定的注解target中注解类型也必须是全限定类型名。 匹配规则 target.class.getAnnotation(指定的注解类型) ! null2种情况可以匹配 注解直接标注在目标类上注解标注在父类上但是注解必须是可以继承的即定义注解的时候需要使用Inherited标注 案例1 注解直接标注在目标类上这种情况目标类会被匹配到。 自定义一个注解Ann6 package com.javacode2018.aop.demo9.test6;import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target;Retention(RetentionPolicy.RUNTIME) Target(ElementType.TYPE) public interface Ann6 { }目标类S6上直接使用Ann1 package com.javacode2018.aop.demo9.test6;Ann6 public class S6 {public void m1() {System.out.println(我是m1);} }来个Aspect类 package com.javacode2018.aop.demo9.test6;import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.annotation.Pointcut;Aspect public class AspectTest6 {//1目标类上有Ann1注解Pointcut(target(Ann1))public void pc() {}Before(pc())public void beforeAdvice(JoinPoint joinPoint) {System.out.println(joinPoint);} }测试代码 Test public void test6() {S6 target new S6();AspectJProxyFactory proxyFactory new AspectJProxyFactory();proxyFactory.setTarget(target);proxyFactory.addAspect(AspectTest6.class);S6 proxy proxyFactory.getProxy();proxy.m1();System.out.println(目标类上是否有 Ann6 注解 (target.getClass().getAnnotation(Ann6.class) ! null)); }运行输出 execution(void com.javacode2018.aop.demo9.test6.S6.m1()) 我是m1 目标类上是否有 Ann6 注解true案例2 注解标注在父类上注解上没有Inherited这种情况下目标类无法匹配到下面看代码 注解Ann7 package com.javacode2018.aop.demo9.test7;import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target;Retention(RetentionPolicy.RUNTIME) Target(ElementType.TYPE) public interface Ann7 { }来2个父子类父类上有Ann7之类S7为目标类 package com.javacode2018.aop.demo9.test7;import java.lang.annotation.Target;Ann7 class S7Parent { }public class S7 extends S7Parent {public void m1() {System.out.println(我是m1);}public static void main(String[] args) {System.out.println(S7.class.getAnnotation(Target.class));} }来个Aspect类 package com.javacode2018.aop.demo9.test7;import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.annotation.Pointcut;Aspect public class AspectTest7 {/*** 匹配目标类上有Ann7注解*/Pointcut(target(com.javacode2018.aop.demo9.test7.Ann7))public void pc() {}Before(pc())public void beforeAdvice(JoinPoint joinPoint) {System.out.println(joinPoint);} }测试代码 Test public void test7() {S7 target new S7();AspectJProxyFactory proxyFactory new AspectJProxyFactory();proxyFactory.setTarget(target);proxyFactory.addAspect(AspectTest7.class);S7 proxy proxyFactory.getProxy();proxy.m1();System.out.println(目标类上是否有 Ann7 注解 (target.getClass().getAnnotation(Ann7.class) ! null)); }运行输出 我是m1 目标类上是否有 Ann7 注解false分析结果 Ann7标注在了父类上但是Ann7定义的时候没有使用Inherited说明之类无法继承父类上面的注解所以上面的目标类没有被拦截下面我们将Ann7的定义改一下加上Inherited package com.javacode2018.aop.demo9.test7;import java.lang.annotation.*;Retention(RetentionPolicy.RUNTIME) Target(ElementType.TYPE) Inherited public interface Ann7 { }再次运行输出 execution(void com.javacode2018.aop.demo9.test7.S7.m1()) 我是m1 目标类上是否有 Ann7 注解true此时目标对象被拦截了。 8、args 用法 args(注解类型)方法参数所属的类上有指定的注解注意不是参数上有指定的注解而是参数类型的类上有指定的注解。 案例1 Pointcut(args(Ann8))匹配方法只有一个参数并且参数所属的类上有Ann8注解可以匹配下面的代码m1方法的第一个参数类型是Car类型Car类型上有注解Ann8 Ann8 class Car { }public void m1(Car car) {System.out.println(我是m1); }案例2 Pointcut(args(*,Ann8))匹配方法只有2个参数且第2个参数所属的类型上有Ann8注解可以匹配下面代码 Ann8 class Car { }public void m1(String name,Car car) {System.out.println(我是m1); }案例3 Pointcut(args(..,com.javacode2018.aop.demo9.test8.Ann8))匹配参数数量大于等于1且最后一个参数所属的类型上有Ann8注解 Pointcut(args(*,com.javacode2018.aop.demo9.test8.Ann8,..))匹配参数数量大于等于2且第2个参数所属的类型上有Ann8注解 Pointcut(args(..,com.javacode2018.aop.demo9.test8.Ann8,*))匹配参数数量大于等于2且倒数第2个参数所属的类型上有Ann8注解这个案例代码大家自己写一下体验一下。 9、annotation 用法 annotation(注解类型)匹配被调用的方法上有指定的注解。 案例 定义一个注解可以用在方法上 package com.javacode2018.aop.demo9.test12;import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target;Retention(RetentionPolicy.RUNTIME) Target(ElementType.METHOD) public interface Ann12 { }定义2个类 S12Parent为父类内部定义了2个方法2个方法上都有Ann12注解 S12是代理的目标类也是S12Parent的子类内部重写了m2方法重写之后m2方法上并没有Ann12注解S12内部还定义2个方法m3和m4而m3上面有注解Ann12 package com.javacode2018.aop.demo9.test12;class S12Parent {Ann12public void m1() {System.out.println(我是S12Parent.m1()方法);}Ann12public void m2() {System.out.println(我是S12Parent.m2()方法);} }public class S12 extends S12Parent {Overridepublic void m2() {System.out.println(我是S12.m2()方法);}Ann12public void m3() {System.out.println(我是S12.m3()方法);}public void m4() {System.out.println(我是S12.m4()方法);} }来个Aspect类 当被调用的目标方法上有Ann12注解的时会被beforeAdvice处理。 package com.javacode2018.aop.demo9.test12;import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.annotation.Pointcut;Aspect public class AspectTest12 {Pointcut(annotation(com.javacode2018.aop.demo9.test12.Ann12))public void pc() {}Before(pc())public void beforeAdvice(JoinPoint joinPoint) {System.out.println(joinPoint);} }测试用例 S12作为目标对象创建代理然后分别调用4个方法 Test public void test12() {S12 target new S12();AspectJProxyFactory proxyFactory new AspectJProxyFactory();proxyFactory.setTarget(target);proxyFactory.addAspect(AspectTest12.class);S12 proxy proxyFactory.getProxy();proxy.m1();proxy.m2();proxy.m3();proxy.m4(); }运行输出 execution(void com.javacode2018.aop.demo9.test12.S12Parent.m1()) 我是S12Parent.m1()方法 我是S12.m2()方法 execution(void com.javacode2018.aop.demo9.test12.S12.m3()) 我是S12.m3()方法 我是S12.m4()方法分析结果 m1方法位于S12Parent中上面有Ann12注解被连接了m3方法上有Ann12注解被拦截了而m4上没有Ann12注解没有被拦截这3个方法的执行结果都很容易理解。 重点在于m2方法的执行结果没有被拦截m2方法虽然在S12Parent中定义的时候也有Ann12注解标注但是这个方法被S1给重写了在S1中定义的时候并没有Ann12注解代码中实际上调用的是S1中的m2方法发现这个方法上并没有Ann12注解所以没有被拦截。 10、bean 用法 bean(bean名称)这个用在spring环境中匹配容器中指定名称的bean。 案例 来个类BeanService package com.javacode2018.aop.demo9.test13;public class BeanService {private String beanName;public BeanService(String beanName) {this.beanName beanName;}public void m1() {System.out.println(this.beanName);} }来个Aspect类 package com.javacode2018.aop.demo9.test13;import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.annotation.Pointcut; import org.springframework.stereotype.Component;Aspect public class Aspect13 {//拦截spring容器中名称为beanService2的beanPointcut(bean(beanService2))public void pc() {}Before(pc())public void beforeAdvice(JoinPoint joinPoint) {System.out.println(joinPoint);} }来个spring配置类 package com.javacode2018.aop.demo9.test13;import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.EnableAspectJAutoProxy;Configuration EnableAspectJAutoProxy // 这个可以启用通过AspectJ方式自动为符合条件的bean创建代理 public class MainConfig13 {//将Aspect13注册到spring容器Beanpublic Aspect13 aspect13() {return new Aspect13();}Beanpublic BeanService beanService1() {return new BeanService(beanService1);}Beanpublic BeanService beanService2() {return new BeanService(beanService2);} }这个配置类中有个EnableAspectJAutoProxy这个注解大家可能比较陌生这个属于aop中自动代理的范围后面会有文章详细介绍这块这里大家暂时先不用关注。 测试用例 下面启动spring容器加载配置类MainConfig13然后分别获取beanService1和beanService2调用他们的m1方法看看效果 Test public void test13() {AnnotationConfigApplicationContext context new AnnotationConfigApplicationContext(MainConfig13.class);//从容器中获取beanService1BeanService beanService1 context.getBean(beanService1, BeanService.class);beanService1.m1();//从容器中获取beanService2BeanService beanService2 context.getBean(beanService2, BeanService.class);beanService2.m1(); }运行输出 beanService1 execution(void com.javacode2018.aop.demo9.test13.BeanService.m1()) beanService2beanService2的m1方法被拦截了。 11、reference pointcut 表示引用其他命名切入点。 有时我们可以将切入专门放在一个类中集中定义。 其他地方可以通过引用的方式引入其他类中定义的切入点。 语法如下 Pointcut(完整包名类名.方法名称())若引用同一个类中定义切入点包名和类名可以省略直接通过方法就可以引用。 比如下面我们可以将所有切入点定义在一个类中 package com.javacode2018.aop.demo9.test14;import org.aspectj.lang.annotation.Pointcut;public class AspectPcDefine {Pointcut(bean(bean1))public void pc1() {}Pointcut(bean(bean2))public void pc2() {} }下面顶一个一个Aspect类来引用上面的切入点 package com.javacode2018.aop.demo9.test14;import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Pointcut;Aspect public class Aspect14 {Pointcut(com.javacode2018.aop.demo9.test14.AspectPcDefine.pc1())public void pointcut1() {}Pointcut(com.javacode2018.aop.demo9.test14.AspectPcDefine.pc1() || com.javacode2018.aop.demo9.test14.AspectPcDefine.pc2())public void pointcut2() {}}12、组合型的pointcut Pointcut定义时还可以使用、||、!运算符。 多个匹配都需要满足||多个匹配中只需满足一个!匹配不满足的情况下 Pointcut(bean(bean1) || bean(bean2)) //匹配bean1或者bean2 Pointcut(target(Ann1) Annotation(Ann2)) //匹配目标类上有Ann1注解并且目标方法上有Ann2注解 Pointcut(target(Ann1) !target(Ann2)) // 匹配目标类上有Ann1注解但是没有Ann2注解总结 本文详解了Pointcut的12种用法案例大家一定要敲一遍敲的过程中会遇到问题然后解决问题才能够加深理解。 有问题的也欢迎大家留言交流谢谢 Spring系列 Spring系列第1篇为何要学springSpring系列第2篇控制反转IoC与依赖注入DISpring系列第3篇Spring容器基本使用及原理Spring系列第4篇xml中bean定义详解(-)Spring系列第5篇创建bean实例这些方式你们都知道Spring系列第6篇玩转bean scope避免跳坑里Spring系列第7篇依赖注入之手动注入Spring系列第8篇自动注入autowire详解高手在于坚持Spring系列第9篇depend-on到底是干什么的Spring系列第10篇primary可以解决什么问题Spring系列第11篇bean中的autowire-candidate又是干什么的Spring系列第12篇lazy-initbean延迟初始化Spring系列第13篇使用继承简化bean配置(abstract parent)Spring系列第14篇lookup-method和replaced-method比较陌生怎么玩的Spring系列第15篇代理详解Java动态代理cglib代理Spring系列第16篇深入理解java注解及spring对注解的增强预备知识Spring系列第17篇Configration和Bean注解详解(bean批量注册)Spring系列第18篇ComponentScan、ComponentScans详解(bean批量注册)Spring系列第18篇import详解(bean批量注册)Spring系列第20篇Conditional通过条件来控制bean的注册Spring系列第21篇注解实现依赖注入Autowired、Resource、Primary、QulifierSpring系列第22篇Scope、DependsOn、ImportResource、Lazy 详解Spring系列第23篇Bean生命周期详解Spring系列第24篇父子容器详解Spring系列第25篇Value【用法、数据来源、动态刷新】Spring系列第26篇国际化详解Spring系列第27篇spring事件机制详解Spring系列第28篇Bean循环依赖详解Spring系列第29篇BeanFactory扩展BeanFactoryPostProcessor、BeanDefinitionRegistryPostProcessorSpring系列第30篇jdk动态代理和cglib代理Spring系列第31篇aop概念详解Spring系列第32篇AOP核心源码、原理详解Spring系列第33篇ProxyFactoryBean创建AOP代理

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/90483.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

做网站快速排名软件做网站的费用会计分录

1 相关概念以及理解 1 python的元组与列表类似,不同之处在于元组的元素不可修改 2 有序且可重复 3 语法: (值1,值2,值3...值n) 最简单的元组为(值,) 而这个(值)并不是元组 4 关键字 tuple 5 因为元组不可修改,我们重新赋值元组的时候,只是改变了指向关系, 原来的元组还是依然…

网站购买域名吗九牛科技网站开发微信营销

File类概述和构造方法 File:它是文件和目录路径名的抽象表示 文件和目录是可以通过File封装成对象的 对于File而言,其封装的并不是一个真正存在的文件,仅仅是一个路径名而已,它可以存在,也可以不存在 我们对Fie的操…

网站底部有很多图标广州网站建设及推广

当我们听到“算法”这个词时,很自然地会想到数学。然而实际上,许多算法并不涉及复杂数学,而是更多地依赖基本逻辑,这些逻辑在我们的日常生活中处处可见。 在正式探讨算法之前,有一个有趣的事实值得分享:你…

设计好的集团网站建设多少钱建筑模板工

广度优先算法(Breadth-First Search)是在图和树领域的搜索方法,其核心思想是从一个起始点开始,访问其所有的临近节点,然后再按照相同的方式访问这些临近节点的节点,这种访问方式类似涟漪泛起,一…

南宁网站推广方案如何做12免费建站网站

一、ps全称:Adobe Photoshop ,Adobe Photoshop是由Adobe Systems 开发和发行的专门用于图形图像处理的软件。 二、PS的应用领域: 1.在平面设计中的应用 2.在插画设计中的应用 3.在网页设计中的应用 4.在界面设计中的应用 5.在数码艺术中的应用…

北京微网站魅族官方网站挂失手机找到怎么做

本专栏栏目提供文章与程序复现思路,具体已有的论文与论文源程序可翻阅本博主免费的专栏栏目《论文与完整程序》 论文与完整源程序_电网论文源程序的博客-CSDN博客https://blog.csdn.net/liang674027206/category_12531414.html 电网论文源程序-CSDN博客电网论文源…

主机屋wordpress建站长沙有哪些设计公司

简介 LVS(Linux Virtual Server)即Linux虚拟服务器,是由章文嵩博士主导的开源负载均衡项目,目前LVS已经被集成到Linux内核模块中。该项目在Linux内核中实现了基于IP的数据请求负载均衡调度方案,终端互联网用户从外部访…

怎么做区块链网站wordpress怎么分段

一、 毛泽东思想是马中化的第一果,是关于搞革命,搞改造,搞建设的理论。 二、新民主主义革命 新民主主义革命的三大法宝:1.统一战线 2. 武装斗争 3.党的建设 政治纲领 经济纲领 文化纲领 乱世造英雄 三、社会主义改造理论&#xff…

厦门好的网站设计信誉好的企业网站建设

一、概念 1.支持向量机(support vector machine,SVM): (1)基于统计学理论的监督学习方法,但不属于生成式模型,而是判别式模型。 (2)支持向量机在各个领域内的…

网站建设大概要多少钱网站模板套用

数据分析之Excel的使用 SUM()求和SUMIF()单条件求和SUMIFS()多条件求和日期函数YEAR()提取年份MONTH()提取月份DAY()提取日 SUBTOTAL()聚合函数IF()函数IF嵌套 VLOOKUP()搜索取值MATCH()返回行值或列值INDEX()定位取值 SUM()求和 SUMIF()单条件求和 SUMIFS()多条件求和 日期…

什么专业可以做网站编辑优化搜索点击次数的方法

文章目录 一、安装和使用Anaconda 1、anaconda是什么? 2、为什么需要安装anaconda? 3、如何安装anaconda? 通过官网下载页面 开源软件下载 安装步骤: 4、jupyter汉化 5、如何管理包? 1.列出已安装的包 2.安装…

备案期间能否做网站解析黄冈网站推广软件哪里买

SpringMvc和SpringBoot对比 首先我们回顾下原来搭建一个springmvc的hello-word的web项目(xml配置的)我们是不是要在pom中导入各种依赖,然后各个依赖有可能还会存在版本冲突需要各种排除。当你历尽千辛万苦的把依赖解决了,然后还需…

创意网站建设策划方案做得大气的网站

一、开通微信支付的首要条件是:认证服务号或政府媒体类认证订阅号(一般认证订阅号无法申请微信支付)二、微信支付分为老版支付和新版支付,除了较早期申请的用户为老版支付,现均为新版微信支付。三、公众平台微信支付开…

网站建设运营合作合同做网站的是什么软件

文章目录T1:数字游戏题目CODET2:公交换乘题目CODET3:纪念品题目题解CODET4:加工领奖题目题解CODE关于普及组的想法&游记T1:数字游戏 题目 小 K 同学向小 P 同学发送了一个长度为 8 的 01 字符串来玩数字游戏&…

中国建设银行山东省分行网站企业网站教程

在React中,可以使用ref来获取DOM节点的引用。以下是一些示例代码,说明如何使用ref来获取DOM节点: 使用React.createRef()(React 16.3版本以及更高版本的推荐方法): import React, { Component } from react…

电子商务网站设计模板商城网站怎么建设

1、哥德巴赫猜想 任意大于2的偶数,都可以表示成为2个素数之和 2、代码实现 #include <iostream> #include <math.h>using namespace std;//判断是否是质数 bool isPrime(int n) {if (n < 2)return false;if (n == 2)return true;for (int i = …

php制作投票网站为什么做的网站打开自动缩放

TCP&#xff1a;传输控制协议 特点&#xff1a;面向连接、可靠通信 TCP的最终目的&#xff1a;要保证在不可靠的信道上实现可靠的传输 TCP主要有三个步骤实现可靠传输&#xff1a;三次握手建立连接&#xff0c;传输数据进行确认&#xff0c;四次挥手断开连接 三次握手建立可靠…

图片类网站建设定西市小企业网站建设

当前情况: 显示: 注意看右侧的树是没有文字的,数据已经渲染,个数是对的,但就是没有文字, 解决: 对比以后发现是template中的#default{data}没有写大括号导致的 所以写上大括号后: 正常显示

官网网站搭建需要多少钱好看的网页设计作品欣赏

Materialization是函数式编程里的一个专业术语, 用于特指函数式编程中查询被实际执行并生成结果的这一过程.首先, 搜了一下中文资料, 暂时没有对该词的中文翻译, CSDN\博客园\阿里云上所有关于它的文档都没有做中文翻译, 直接沿用这个英文单词.难道轮到我来创造这个词的中文翻…

个人能建设网站吗百度指数查询排行榜

外观模式&#xff08;Facade Pattern&#xff09;是一种结构型设计模式&#xff0c;它提供了一个统一的接口&#xff0c;用于访问子系统中的一组接口。外观模式通过隐藏子系统的复杂性&#xff0c;简化了客户端与子系统之间的交互&#xff0c;提供了一个更简单、更直观的接口。…