不通过网站可以做360全景吗,重庆做企业年报在哪个网站做,新赣州网,网创项目资源网站目录 AOP概念AOP底层原理AOP(JDK动态代理)使用 JDK 动态代理#xff0c;使用 Proxy 类里面的方法创建代理对象**编写** **JDK** 动态代理代码 AOP(术语)AOP操作#xff08;准备工作#xff09;**AOP** **操作#xff08;**AspectJ注解)**AOP** **操作#xff08;**AspectJ… 目录 AOP概念AOP底层原理AOP(JDK动态代理)使用 JDK 动态代理使用 Proxy 类里面的方法创建代理对象**编写** **JDK** 动态代理代码 AOP(术语)AOP操作准备工作**AOP** **操作**AspectJ注解)**AOP** **操作**AspectJ **配置文件** 开篇: 欢迎再次来到 Spring 5 学习系列在这个博客中我们将深入研究 Spring 框架的AOP概念原理动态代理术语Aspect操作案例注解与配置方式。 AOP
概念
什么是AOP
1面向切面编程方面利用 AOP 可以对业务逻辑的各个部分进行隔离从而使得
业务逻辑各部分之间的耦合度降低提高程序的可重用性同时提高了开发的效率。
2通俗描述不通过修改源代码方式在主干功能里面添加新功能
3使用登录例子说明 AOP AOP底层原理
AOP 底层使用动态代理 两种情况 第一种 有接口情况使用 JDK 动态代理 创建接口实现类代理对象增强类的方法 **第二种 **没有接口情况使用 CGLIB 动态代理 创建子类的代理对象增强类的方法
AOP(JDK动态代理)
使用 JDK 动态代理使用 Proxy 类里面的方法创建代理对象
下面是jdk8官方文档
compact1, compact2, compact3
java.lang.reflect
Class Proxy
java.lang.Object
java.lang.reflect.Proxystatic Object newProxyInstance(ClassLoader loader, 类?[] interfaces, InvocationHandler h)
返回指定接口的代理类的实例该接口将方法调用分派给指定的调用处理程序。public static Object newProxyInstance(ClassLoader loader,类?[] interfaces,InvocationHandler h)throws IllegalArgumentException
返回指定接口的代理类的实例该接口将方法调用分派给指定的调用处理程序。
Proxy.newProxyInstance因为与IllegalArgumentException相同的原因而Proxy.getProxyClass 。参数loader - 类加载器来定义代理类interfaces - 代理类实现的接口列表h - 调度方法调用的调用处理函数
结果具有由指定的类加载器定义并实现指定接口的代理类的指定调用处理程序的代理实例
异常IllegalArgumentException - 如果对可能传递给 getProxyClass有任何 getProxyClass被违反
SecurityException -如果安全管理器S存在任何下列条件得到满足给定的loader是null 并且调用者的类加载器不是null 并且调用s.checkPermission与RuntimePermission(getClassLoader)权限拒绝访问;
对于每个代理接口 intf 呼叫者的类加载器是不一样的或类加载器的祖先intf和调用s.checkPackageAccess()拒绝访问intf ;
任何给定的代理接口的是非公和呼叫者类是不在同一runtime package作为非公共接口和调用s.checkPermission与ReflectPermission(newProxyInPackage.{package name})权限拒绝访问。
NullPointerException - 如果 interfaces数组参数或其任何元素是 null 或者如果调用处理程序 h是 null调用 newProxyInstance 方法 方法有三个参数 第一参数类加载器 第二参数增强方法所在的类这个类实现的接口支持多个接口 第三参数实现这个接口 InvocationHandler创建代理对象写增强的部分
编写 JDK 动态代理代码
1创建接口定义方法
public interface UserDao {public int add(int a,int b);public String update(String id);
}2创建接口实现类实现方法
public class UserDaoImpl implements UserDao {Overridepublic int add(int a, int b) {return ab;}Overridepublic String update(String id) {return id;}
}3使用 Proxy 类创建接口代理对象
//创建代理对象代码
class UserDaoProxy implements InvocationHandler {//1 把创建的是谁的代理对象把谁传递过来//有参数构造传递private Object obj;public UserDaoProxy(Object obj) {this.obj obj;}//增强的逻辑Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {//方法之前System.out.println(方法之前行....method.getName() :传递的参数... Arrays.toString(args));//被增强的方法执行Object res method.invoke(obj, args);//方法之后System.out.println(方法之后执行....obj);return res;}
}
// 测试类这里Proxy.newProxyInstance
public class JDKProxy {public static void main(String[] args) {Class[] interfaces {UserDao.class};UserDaoImpl userDaoImpl new UserDaoImpl();UserDao dao (UserDao)Proxy.newProxyInstance(JDKProxy.class.getClassLoader(), interfaces, new UserDaoProxy(userDaoImpl));int add dao.add(2, 3);System.out.println(result:add);String update dao.update(6);System.out.println(result:update);}
}结果
方法之前执行....add :传递的参数...[2, 3]
方法之后执行....com.zhuyh.spring.day0123.aop.dao.impl.UserDaoImpl2c7b84de
result:5
方法之前执行....update :传递的参数...[6]
方法之后执行....com.zhuyh.spring.day0123.aop.dao.impl.UserDaoImpl2c7b84de
result:6AOP(术语)
连接点
类里面哪些方法可以增强这些点被称为连接点
切入点
实际被真正增强的方法
通知增强
实际增强的逻辑部分称为通知增强通知增强有多种类型 前置通知后置通知环绕通知异常通知最终通知
切面是动作
把通知增强应用到切入点过程
AOP操作准备工作
Spring 框架一般都是基于 AspectJ 实现 AOP 操作
1AspectJ 不是 Spring 组成部分独立 AOP 框架一般把 AspectJ 和 Spirng 框架一起使
用进行 AOP 操作
基于 AspectJ 实现 AOP 操作
1基于 xml 配置文件实现
2基于注解方式实现使用
在项目工程里面引入 AOP 相关依赖 切入点表达式
1切入点表达式作用知道对哪个类里面的哪个方法进行增强
2语法结构 execution([权限修饰符] [返回类型] [类全路径] [方法名称] [参数列表])
举例 1对 com.atguigu.dao.BookDao 类里面的 add 进行增强
execution(* com.atguigu.dao.BookDao.add(…))
举例 2对 com.atguigu.dao.BookDao 类里面的所有的方法进行增强
execution(* com.atguigu.dao.BookDao.* (…))举例 3对 com.atguigu.dao 包里面所有类类里面所有方法进行增强
execution(* com.atguigu.dao.. (…))
AOP **操作**AspectJ注解)
创建类在类里面定义方法
public class User {public void add() {System.out.println(add.......);}
}创建增强类编写增强逻辑
在增强类里面创建方法让不同方法代表不同通知类型
//增强的类
public class UserProxy {public void before() {//前置通知System.out.println(before......);}
}进行通知的配置
在 spring 配置文件中开启注解扫描
?xml version1.0 encodingUTF-8?
beans xmlnshttp://www.springframework.org/schema/beans xmlns:xsihttp://www.w3.org/2001/XMLSchema-instance xmlns:contexthttp://www.springframework.org/schema/context xmlns:aophttp://www.springframework.org/schema/aop xsi:schemaLocationhttp://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd!-- 开启注解扫描 --context:component-scan basepackagecom.atguigu.spring5.aopanno/context:component-scan使用注解创建 User 和 UserProxy 对象
// 被增强的类
public class User{}
// 增强的类
public class UserProxy{}在增强类上面添加注解 Aspect
//增强的类
Component
Aspect //生成代理对象
public class UserProxy {}在 spring 配置文件中开启生成代理对象
!-- 开启 Aspect 生成代理对象--
aop:aspectj-autoproxy/aop:aspectj-autoproxy配置不同类型的通知
在增强类的里面在作为通知方法上面添加通知类型注解使用切入点表达式配置
//增强的类
Component
Aspect //生成代理对象
public class UserProxy {//前置通知//Before 注解表示作为前置通知Before(value execution(* com.atguigu.spring5.aopanno.User.add(..)))public void before() {System.out.println(before.........);}//后置通知返回通知AfterReturning(value execution(*
com.atguigu.spring5.aopanno.User.add(..)))public void afterReturning() {System.out.println(afterReturning.........);}//最终通知After(value execution(* com.atguigu.spring5.aopanno.User.add(..)))public void after() {System.out.println(after.........);}//异常通知AfterThrowing(value execution(*
com.atguigu.spring5.aopanno.User.add(..)))public void afterThrowing() {System.out.println(afterThrowing.........);}//环绕通知Around(value execution(* com.atguigu.spring5.aopanno.User.add(..)))public void around(ProceedingJoinPoint proceedingJoinPoint) throws
Throwable {System.out.println(环绕之前.........);//被增强的方法执行proceedingJoinPoint.proceed();System.out.println(环绕之后.........);}
}相同的切入点抽取
//相同切入点抽取
Pointcut(value execution(* com.atguigu.spring5.aopanno.User.add(..)))
public void pointdemo() {
}
//前置通知
//Before 注解表示作为前置通知
Before(value pointdemo())
public void before() {System.out.println(before.........);
}有多个增强类多同一个方法进行增强设置增强类优先级
在增强类上面添加注解 Order(数字类型值)数字类型值越小优先级越高
Component
Aspect
Order(1)
public class PersonProxy{}完全使用注解开发
创建配置类不需要创建 xml 配置文件
Configuration
ComponentScan(basePackages {com.atguigu})
EnableAspectJAutoProxy(proxyTargetClass true)
public class ConfigAop {
}AOP **操作**AspectJ 配置文件 创建两个类增强类和被增强类创建方法 在 spring 配置文件中创建两个类对象
!--创建对象--
bean idbook classcom.atguigu.spring5.aopxml.Book/bean
bean idbookProxy classcom.atguigu.spring5.aopxml.BookProxy/bean在 spring 配置文件中配置切入点
!--配置 aop 增强--
aop:config!--切入点--aop:pointcut idp expressionexecution(*
com.atguigu.spring5.aopxml.Book.buy(..))/!--配置切面--aop:aspect refbookProxy!--增强作用在具体的方法上--aop:before methodbefore pointcut-refp//aop:aspect
/aop:config感谢您阅读 Spring 5 学习系列的第三篇在这篇文章中我们探索了Spring5的AOP概念原理动态代理术语Aspect操作案例注解与配置方式
下一篇文章即将发布 在第四篇中我们将深入研究Spring 5的JdbcTemplate为您分享我学习的Spring5的收获请继续关注我的系列。
谢谢您的陪伴 如果您有任何问题、建议或想要了解的特定主题请随时在评论中告诉我们。我们期待与您共同探索Spring 5共同提升我们的Java开发技能
敬请期待第四篇的发布我们将很快与您再次见面
学习视频来源尚硅谷Spring5