阐述AOP的含义(代码)

什么是AOP

        AOP(Aspect-OrientedProgramming,面向切面编程),可以说是OOP(Object-Oriented Programing,面向对象编程)的补充和完善。OOP引入封装、继承和多态性等概念来建立一种对象层次结构,用以模拟公共行为的一个集合。当我们需要为分散的对象引入公共行为的时候,OOP则显得无能为力。也就是说,OOP允许你定义从上到下的关系,但并不适合定义从左到右的关系。例如日志功能。日志代码往往水平地散布在所有对象层次中,而与它所散布到的对象的核心功能毫无关系。对于其他类型的代码,如安全性、异常处理和透明的持续性也是如此。这种散布在各处的无关的代码被称为横切(cross-cutting)代码,在OOP设计中,它导致了大量代码的重复,而不利于各个模块的重用。

       而AOP技术则恰恰相反,它利用一种称为“横切”的技术,剖解开封装的对象内部,并将那些影响了多个类的公共行为封装到一个可重用模块,并将其名为“Aspect”,即方面。所谓“方面”,简单地说,就是将那些与业务无关,却为业务模块所共同调用的逻辑或责任封装起来,便于减少系统的重复代码,降低模块间的耦合度,并有利于未来的可操作性和可维护性。AOP代表的是一个横向的关系,如果说“对象”是一个空心的圆柱体,其中封装的是对象的属性和行为;那么面向方面编程的方法,就仿佛一把利刃,将这些空心圆柱体剖开,以获得其内部的消息。而剖开的切面,也就是所谓的“方面”了。然后它又以巧夺天功的妙手将这些剖开的切面复原,不留痕迹。

1> AOP底层使用动态代理 第一种情况有接口,使用JDK动态代理 创建接口实现类代理对象,增强类的方法。 第二种情况无接口,使用CGLIB动态代码,创建子类的代理对象,增强类的方法。 AOP(JDK动态代理,代码实现如下)

i.创建接口与接口的实现类

package com.openlab.dao.impl;

import org.springframework.stereotype.Component;
import org.springframework.stereotype.Repository;

@Component
public class Deptdaoimpl {
    public int unpate() {
        System.out.println("deptimpl执行的方法");
        return 1;
    }
}

ii.创建InvocationHandler

package com.openlab.proxy;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

public class Myhandler implements InvocationHandler {
    private Object ratget;
    public Myhandler(Object ratget){
        this.ratget=ratget;
    }
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("方法运行之前执行");
        Object res =method.invoke(ratget,args);
        System.out.println("方法运行之后执行");
        return res;
    }
}

iii.调用

public class Testproxy {
@Test
    public void test01(){
        ApplicationContext context =new ClassPathXmlApplicationContext("beans3.xml");
      Deptdaoimpl deptdaoimpl =  context.getBean(Deptdaoimpl.class);
      deptdaoimpl.unpate();
    }

通知方法: 

  1. 前置通知:在我们执行目标方法之前运行(@Before)
  2. 后置通知:在我们目标方法运行结束之后 ,不管有没有异常(@After)
  3. 返回通知:在我们的目标方法正常返回值后运行(@AfterReturning)
  4. 异常通知:在我们的目标方法出现异常后运行(@AfterThrowing)
  5. 环绕通知:动态代理, 需要手动执行joinPoint.procced()(其实就是执行我们的目标方法执行之前相当于前置通知, 执行之后就相当于我们后置通知(@Around)

    iv.切入点表达式 切入点表达式作用:知道对哪个类里面的哪个方法进行增强 语法结构:execution (* com.sample.service.impl.*. *(..))
    1)execution(): 表达式主体。
    2)第一个*号:表示返回类型, *号表示所有的类型。
    3)包名:表示需要拦截的包名,后面的两个句点表示当前包和当前包的所有子包, com.sample.service.impl包、子孙包下所有类的方法。
    4)第二个号:表示类名,号表示所有的类。
    5)(..):最后这个星号表示方法名,*号表示所有的方法,后面括弧里面表示方法的参数,两 个句点表示任何参数 v.具体操作步骤 注意,不要写接口,返回值类型必须是包装类型,不能是基本数据类型。 首先,创建类,在类里面定义方法。
package com.openlab.dao.impl;

import org.springframework.stereotype.Component;
import org.springframework.stereotype.Repository;

@Component
public class Deptdaoimpl {
    public int unpate() {
        System.out.println("deptimpl执行的方法");
        return 1;
    }
}

其次,创建增强类(增强逻辑)在增强类里面,创建方法,让不同方法代表不同通知类型

package com.openlab.proxy;

import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.springframework.stereotype.Component;

@Component
@Aspect
public class deptdaoproxy {
    @Before(value = "execution( * com.openlab.dao.impl.Deptdaoimpl.unpate(..))")
    public void before(){
        System.out.println("创造方法之前");
    }
}

配置文件

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop https://www.springframework.org/schema/aop/spring-aop.xsd">
        <context:component-scan base-package="com.openlab"></context:component-scan>
            <aop:aspectj-autoproxy></aop:aspectj-autoproxy>
</beans>

测试

import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class Testproxy {
@Test
    public void test01(){
        ApplicationContext context =new ClassPathXmlApplicationContext("beans3.xml");
      Deptdaoimpl deptdaoimpl =  context.getBean(Deptdaoimpl.class);
      deptdaoimpl.unpate();
    }

vii.注解开发 首先,创建俩个类,其中一个是增强类

<bean id="deptDao" class="com.openlab.aopxml.DeptDao"></bean>
<bean id="deptDaoProxy" class="com.openlab.aopxml.DeptDaoProxy"></bean>
<aop:config>
<aop:pointcut id="point" expression="execution(*
com.openlab.aopxml.DeptDao.*(..))"/>
<aop:aspect ref="deptDaoProxy">
<aop:before method="before" pointcut-ref="point">
</aop:before>
</aop:aspect>
</aop:config>

Spring AOP是Spring框架中的一个关键特性,它允许开发者在不改变原始代码的情况下为应用程序添加横切关注点。下面是Spring AOP的一些实现代码: 1. 定义切面类 ```java @Aspect @Component public class LoggingAspect { @Before("execution(public * com.example.myapp.service.*.*(..))") public void logBefore(JoinPoint joinPoint) { System.out.println("Before method: " + joinPoint.getSignature().getName()); } @After("execution(public * com.example.myapp.service.*.*(..))") public void logAfter(JoinPoint joinPoint) { System.out.println("After method: " + joinPoint.getSignature().getName()); } @AfterReturning(pointcut = "execution(public * com.example.myapp.service.*.*(..))", returning = "result") public void logAfterReturning(JoinPoint joinPoint, Object result) { System.out.println("After returning from method: " + joinPoint.getSignature().getName() + ", result is: " + result); } @AfterThrowing(pointcut = "execution(public * com.example.myapp.service.*.*(..))", throwing = "error") public void logAfterThrowing(JoinPoint joinPoint, Throwable error) { System.out.println("Exception thrown from method: " + joinPoint.getSignature().getName() + ", exception is: " + error); } } ``` 2. 在Spring配置文件中启用AOP ```xml <aop:aspectj-autoproxy/> ``` 以上代码演示了如何定义一个切面类来实现日志记录功能,并且在Spring配置文件中启用AOP,从而使切面类的方法可以在应用程序的service层中被调用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值