博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Spring 学习教程(一):Spring中AOP基础
阅读量:5012 次
发布时间:2019-06-12

本文共 5554 字,大约阅读时间需要 18 分钟。

Spring中AOP基础

 

一、以一个例子引入

1、例子描述:实现一个加减乘除的计算器,要求有日志,能给在程序执行期间追踪正在发生的活动。

2、代码示例:

ArithmeticCalculator接口:

 

package com.at.aop; public interface ArithmeticCalculator {     int add(int i,int j);    int sub(int i,int j);        int mul(int i,int j);    int div(int i,int j);}

 

ArithmeticCalculator接口的实现类ArithmeticCalculatorLoggingImpl

 

package com.at.aop; public class ArithmeticCalculatorLoggingImpl implements ArithmeticCalculator {     @Override    public int add(int i, int j) {        System.out.println("add 方法输入参数是["+i+","+j+"]");        int result = i + j;        System.out.println("add 方法输出结果是"+result);        return result;    }     @Override    public int sub(int i, int j) {        System.out.println("sub 方法输入参数是["+i+","+j+"]");        int result = i - j;        System.out.println("sub 方法输出结果是"+result);        return result;    }     @Override    public int mul(int i, int j) {        System.out.println("mul 方法输入参数是["+i+","+j+"]");        int result = i * j;        System.out.println("mul 方法输出结果是"+result);        return result;    }     @Override    public int div(int i, int j) {        System.out.println("div 方法输入参数是["+i+","+j+"]");        int result = i / j;        System.out.println("div 方法输出结果是"+result);        return result;    } }

测试函数

 

package com.at.aop; public class TestArithmeticCalculator {     public static void main(String[] args) {        ArithmeticCalculator arithmeticCalculator = null;        arithmeticCalculator = new ArithmeticCalculatorLoggingImpl();                int result = arithmeticCalculator.add(1, 2);        System.out.println(result);                result = arithmeticCalculator.div(4, 2);        System.out.println(result);    }}

 

运行结果:

 

add 方法输入参数是[1,2]add 方法输出结果是33div 方法输入参数是[4,2]div 方法输出结果是22
3、例子总结(1)代码混乱:越来越多的非业务需求(日志和验证等)加入后, 原有的业务方法急剧膨胀。每个方法在处理核心逻辑的同时还必须兼顾其他多个关注点。(2)代码分散: 以日志需求为例, 只是为了满足这个单一需求, 就不得不在多个模块(方法)里多次重复相同的日志代码。如果日志需求发生变化, 必须修改所有模块。

 

二、使用动态代理方式解决冗余代码问题

 

将ArithmeticCalculator接口的实现类改为ArithmeticCalculatorImpl:

 

package com.at.aop; public class ArithmeticCalculatorImpl implements ArithmeticCalculator {     @Override    public int add(int i, int j) {        int result = i + j;        return result;    }     @Override    public int sub(int i, int j) {        int result = i - j;        return result;    }     @Override    public int mul(int i, int j) {        int result = i * j;        return result;    }     @Override    public int div(int i, int j) {        int result = i / j;        return result;    } }

增加一个代理类ArithmeticCalculatorLoggingProxy

 

package com.at.aop; import java.lang.reflect.InvocationHandler;import java.lang.reflect.Method;import java.lang.reflect.Proxy;import java.util.Arrays; public class ArithmeticCalculatorLoggingProxy {     //要代理的对象    private ArithmeticCalculator target;        public ArithmeticCalculatorLoggingProxy(ArithmeticCalculator target) {        super();        this.target = target;    }     public ArithmeticCalculator getLoggingProxy(){        ArithmeticCalculator proxy=null;                //代理对象由哪一个类加载器负责加载        ClassLoader loader = target.getClass().getClassLoader();        //代理对象的类型,也就是其中有哪些方法        Class [] interfaces = new Class[]{ArithmeticCalculator.class};        //当调用代理对象其中的方法时候,执行的代码        InvocationHandler h = new InvocationHandler() {            /**             * proxy:正在返回的那个代理对象。一般情况下,在invoke方法中都不使用该对象。             * method:正在被调用的方法             * args:调用方法时候,传入的参数             */            @Override            public Object invoke(Object proxy, Method method, Object[] args)                    throws Throwable {                String methodName = method.getName();                //日志                System.out.println("使用代理方式的方法"+methodName+"输入参数是"+Arrays.asList(args));                //执行方法                Object result = method.invoke(target, args);                //日志                System.out.println("使用代理方式的方法"+methodName+"输出结果是"+result);                return result;            }        };                proxy = (ArithmeticCalculator) Proxy.newProxyInstance(loader, interfaces, h);                return proxy;    }}

测试函数

 

ArithmeticCalculator target=new ArithmeticCalculatorImpl();        ArithmeticCalculator proxy = new ArithmeticCalculatorLoggingProxy(target).getLoggingProxy();                        int result = proxy.add(1, 2);        System.out.println(result);                result = proxy.div(4, 2);        System.out.println(result);

 

运行结果

使用代理方式的方法add输入参数是[1, 2]使用代理方式的方法add输出结果是33使用代理方式的方法div输入参数是[4, 2]使用代理方式的方法div输出结果是22

三、引入AOP

(一)AOP简介

1、AOP(Aspect-Oriented Programming, 面向切面编程): 是一种新的方法论, 是对传统 OOP(Object-Oriented Programming, 面向对象编程) 的补充。

2、AOP 的主要编程对象是切面(aspect), 而切面模块化横切关注点。
3、在应用 AOP 编程时, 仍然需要定义公共功能, 但可以明确的定义这个功能在哪里, 以什么方式应用, 并且不必修改受影响的类。 这样一来横切关注点就被模块化到特殊的对象(切面)里。
4、AOP 的好处:
(1)每个事物逻辑位于一个位置, 代码不分散, 便于维护和升级.
(2)业务模块更简洁, 只包含核心业务代码。
 

 

(二)AOP图解和相关术语描述

1、图解上面计算器的例子

 

 

2、相关术语

(1)切面(Aspect):  横切关注点(跨越应用程序多个模块的功能)被模块化的特殊对象。

(2)通知(Advice):  切面必须要完成的工作。
(3)目标(Target): 被通知的对象。
(4)代理(Proxy): 向目标对象应用通知之后创建的对象。
(5)连接点(Joinpoint):程序执行的某个特定位置:如类某个方法调用前、调用后、方法抛出异常后等。连接点由两个信息确定:方法表示的程序执行点;相对点表示的方位。例如 ArithmethicCalculator#add() 方法执行前的连接点,执行点为 ArithmethicCalculator#add(); 方位为该方法执行前的位置。
(6)切点(pointcut):每个类都拥有多个连接点:例如 ArithmethicCalculator 的所有方法实际上都是连接点,即连接点是程序类中客观存在的事务。AOP 通过切点定位到特定的连接点。类比:连接点相当于数据库中的记录,切点相当于查询条件。切点和连接点不是一对一的关系,一个切点匹配多个连接点,切点通过 
org.springframework.aop.Pointcut 接口进行描述,它使用类和方法作为连接点的查询条件。

 

 

--------------------- 

 

原文:https://blog.csdn.net/luoyepiaoxue2014/article/details/72855553
微博地址:http://weibo.com/luoyepiaoxue2014 

 
 
 
 

 

 

 

 
 

转载于:https://www.cnblogs.com/xiaohuizhenyoucai/p/11050050.html

你可能感兴趣的文章
UVALive 4128 Steam Roller 蒸汽式压路机(最短路,变形) WA中。。。。。
查看>>
记忆--1.致我们不可缺少的记忆
查看>>
lintcode28- Search a 2D Matrix- easy
查看>>
react项目
查看>>
C# 万年历 农历 节气 节日 星座 星宿 属相 生肖 闰年月 时辰(转)
查看>>
A Simple Tree Problem
查看>>
Modular Inverse [ZOJ 3609]
查看>>
MySQL性能测试工具之mysqlslap使用详解
查看>>
深入理解jsonp跨域请求原理
查看>>
regsvr32注册COM组件失败
查看>>
jmeter,CSV数据加载、数据库连接、正则
查看>>
(独孤九剑)--正则表达式
查看>>
MySQL学习点滴 --分区表
查看>>
4.6.1 测试基础
查看>>
洛谷 P2486 [SDOI2011]染色
查看>>
oo第三单元总结
查看>>
leetcode : Count and Say [基本功]
查看>>
洛谷 P2485 [SDOI2011]计算器 解题报告
查看>>
c#访问存储过程
查看>>
Slickflow.NET 开源工作流引擎基础介绍(三) -- 基于HTML5/Bootstrap的Web流程设计器
查看>>