实现AOP的几种方式

源码地址:https://github.com/yuboYUBO/aop-iml

AOP是一种编程思想,是OOP编程思想的补充。AOP的实现有很多种,下面简单介绍几种。

测试AOP需要的一些基础类

Font.java

package com.java.proxy;

import lombok.Data;

@Data
public class Font {

	private String name;
}

FontProvider.java

package com.java.proxy;


public interface FontProvider {
	
	Font getFont(String name);
	
	void printName(String name);
}

FontProviderFromDisk.java

package com.java.proxy;

public class FontProviderFromDisk implements FontProvider {

	
	@Override
	public Font getFont(String name) {
		Font font = new Font();
		font.setName(name);
		return font;
	}

	@Override
	public void printName(String name) {
		System.out.println(name);
		
	}

}

ProviderFactory.java

package com.java.proxy;

public abstract class ProviderFactory {

	
	public static FontProvider getFontProvider() {
        return new FontProviderFromDisk();
    }
}

Bussiness.java

package com.java.proxy;

public class Business {

	public static void main(String[] args) {
		 FontProvider fontProvider = ProviderFactory.getFontProvider();
	        Font font = fontProvider.getFont("微软雅黑");
	        System.out.println(font);
	}

}

执行Bussiness代码的结果:

上面实现的功能就是,通过工厂获取FontProvider,通过Provider获取Font,最后调用功能(这里只是简单的输出)。

下面我们要在getFont()前后加些功能,达到aop的效果。直接修改以前的FontProviderFromDisk类,违反了开闭原则,是不好的,如果有很多FontProvider的实现,需要一一修改,容易出错。

1、代理模式静态的实现AOP

为了编码的规范,遵守一些编程原则,我们使用代理模式,增加一个具有缓存功能的代理类。

代理类CachedFontProvider.java

package com.java.proxy.staticporxy;

import java.util.HashMap;
import java.util.Map;

import com.java.proxy.Font;
import com.java.proxy.FontProvider;
import com.java.proxy.FontProviderFromDisk;

/**
 * 给FontProvider的getFont添加缓存功能,用静态代理来实现
 * @author 
 *
 */
public class CachedFontProvider implements FontProvider {

	private FontProvider fontProvider;
	
	private Map<String, Font> cached = new HashMap<String, Font>();
	
	
	public CachedFontProvider() {
		fontProvider = new FontProviderFromDisk();
	}



	@Override
	public Font getFont(String name) {
		System.out.println("静态代理getFont()");
		Font font = cached.get(name);
		if(font == null) {
			font = fontProvider.getFont(name);
			cached.put(name, font);
		}
		return font;
	}



	@Override
	public void printName(String name) {
		System.out.println("静态代理printName()");
		fontProvider.printName(name);;
		
	}

	
}

工厂类也需要做一些改变ProviderFactory.java

/**
 * 每个字体都增加了缓存功能,其实工厂就是用的缓存字体提供器,跟io一样
 * 使用代理(静态),已经避免了再去修改每个字体提供器(这违反了开闭原则,而且工作量很大,容易出错;而且如果要增加别的功能
 * 比如日志打印,权限检查,异常处理,每个都要去修改,代码重复,而且很麻烦)
 * 
 * ② 然而为什么要用动态代理?
 *考虑以下各种情况,有多个提供类,每个类都有getXxx(String name)方法,
 *每个类都要加入缓存功能,使用静态代理虽然也能实现,但是也是略显繁琐,需要手动一一创建代理类。
 * @author wb-yb197819
 *
 */
public class ProviderFactory {

	public static FontProvider getFontProvider() {
		return new CachedFontProvider();
	}
	
	
	
}

测试类Bussiness.java

package com.java.proxy.staticporxy;

import com.java.proxy.Font;
import com.java.proxy.FontProvider;

public class Business {

	public static void main(String[] args) {
		 FontProvider fontProvider = ProviderFactory.getFontProvider();
	        Font font = fontProvider.getFont("微软雅黑");
	        System.out.println(font);
	        fontProvider.printName("代理模式实现AOP");
	}

}

执行结果:

说明:我们通过工厂获取provider的某个功能代理对象,在调用getFont(),在代理的对象方法中添加功能,调用被代理的功能,实现AOP效果。

2、aspectj静态代理实现AOP

利用aspectj实现AOP功能,需要安装eclipse的aspectj插件(编译*.aj文件,jdk编译不了*.aj文件),依赖aspectj的jar包。

具体操作参考:在maven项目中使用aspectj

Hello.java

package com.java.proxy.aspectj;

public class Hello {
	public void hello(String name) {
        System.out.println(name+",hello!");
    }
}

HelloWorld.aj

package com.java.proxy.aspectj;

public aspect HelloWorld {
	/**
     * 第一个*号是指返回值不限,第二个*号是指方法名不限
     * 括号只是任意个数类型不限的形参
     */
	before() : call(* com.java.proxy.aspectj.*.*(..)) {
        System.out.println("hello前的检查,哈哈");
    }
    after() : call(* com.java.proxy.aspectj.*.*(..)) {
        System.out.println("hello后的检查,哈哈");
    }
}

测试类

package com.java.proxy.aspectj;

public class Test {

	public static void main(String[] args) {
		Hello hello = new Hello();
        hello.hello("张三");
	}

}

执行结果:

说明:aspectj帮我们将功能,在编译的阶段将织入到被代理的class文件里面。因为是静态织入,所以性能比较不错。平时很少用到,如果工作中需要使用aspectj,那就要全面掌握aspectj,aspectj官网:https://www.eclipse.org/aspectj/

 

 

3、jdk动态代理实现AOP

CachedProviderHandler.java

package com.java.proxy.dynamicporxy;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.Map;

public class CachedProviderHandler implements InvocationHandler{
	
	private Map<String, Object> cached = new HashMap<String, Object>();
	private Object target;
	
	public CachedProviderHandler(Object target) {
		this.target = target;
	}

	/**
	 * invoke方法可以处理target的所有方法,这里用if判断只处理了getXXX()方法,增加了缓存功能。
	 */
	@Override
	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
		System.out.println("使用动态代理了!");
		Class<?>[] types = method.getParameterTypes();
		if(method.getName().matches("get.+") && types.length == 1 && types[0] == String.class) {
			System.out.println("getXXX()方法,使用缓存");
			String key = (String)args[0];
			Object value = cached.get(key);
			if(value == null) {
				value = method.invoke(target, args);
				cached.put(key, value);
			}
			return value;
		}
		return method.invoke(target, args);
	}

}

工厂类ProviderFactory.java

package com.java.proxy.dynamicporxy;

import java.lang.reflect.Proxy;

import com.java.proxy.FontProvider;
import com.java.proxy.FontProviderFromDisk;

public class ProviderFactory {

	public static FontProvider getFontProvider() {
		Class<FontProvider> targetClass = FontProvider.class;
		return (FontProvider)Proxy.newProxyInstance(targetClass.getClassLoader(), new Class[]{targetClass},
				new CachedProviderHandler(new FontProviderFromDisk()));
		
	}
}

测试类

package com.java.proxy.dynamicporxy;

import com.java.proxy.Font;
import com.java.proxy.FontProvider;

public class Business {

	public static void main(String[] args) {
		FontProvider fontProvider = ProviderFactory.getFontProvider();
		Font font = fontProvider.getFont("微软雅黑");
		System.out.println(font);
		fontProvider.printName("sdfdf");
	}

}

执行结果:

说明:jdk动态代理需要用Proxy.newProxyInstance来生成代理对象,这里有依赖被代理对象的接口,如果没有接口的就不行。

一句话:jdk动态代理比较好用,就是获取代理对象稍显麻烦。

4、cglib动态代理实现AOP

CGLibProxy.java

package com.java.proxy.cglib;

import java.lang.reflect.Method;

import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;

public class CGLibProxy implements MethodInterceptor{
	
	@SuppressWarnings("unchecked")
	public <T> T getProxy(Class<T> cls) {
		return (T) Enhancer.create(cls, this);
		
	}

	@Override
	public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
		System.out.println("使用cglib1");
		Object result = proxy.invokeSuper(obj, args);
		System.out.println("使用cglib2");
		return result;
	}

}

测试类

package com.java.proxy.cglib;

import com.java.proxy.FontProviderFromDisk;

public class Business {

	public static void main(String[] args) {
		CGLibProxy cgLibProxy = new CGLibProxy();
		FontProviderFromDisk proxy = cgLibProxy.getProxy(FontProviderFromDisk.class);
		proxy.printName("微软雅黑");
	}

}

执行结果:

说明:cglib较前面提到的这几种实现AOP功能,是最好用的。用Enhancer.create(Class, this)获取代理对象更方便,不依赖于接口,代理功能实现起来更简单。

先写到这里,后面在完善。

 

<think>好的,我现在需要回答用户关于Spring AOP实现方式的问题。首先,我得回忆一下Spring AOP的基本概念和实现方式。根据用户提供的引用内容,Spring AOP主要是通过动态代理来实现的,具体有两种方式:基于JDK的动态代理和基于CGLIB的动态代理。此外,可能还涉及到AspectJ的集成,但需要区分Spring AOP和AspectJ的不同。 接下来,我需要确认用户的问题核心是“Spring AOP实现方式”,特别是切面编程的具体方法。用户可能想知道不同的实现手段以及它们的区别。根据引用[1]和[2],Spring AOP主要使用动态代理,而引用[2]提到Spring Boot中的AOP实现可能与AspectJ结合使用,但需要明确Spring AOP本身是否支持其他方式。 首先,要明确两种代理方式的适用场景:JDK动态代理适用于接口实现的类,而CGLIB可以代理没有接口的类。这点很重要,需要详细说明。然后,可能还需要提到如何通过配置选择使用哪种代理方式,比如在Spring配置中设置proxy-target-class属性。 另外,用户可能对AspectJ的注解支持感兴趣,比如@Aspect、@Pointcut、@Around等,这些是Spring AOP中定义切面的常用方式。需要解释如何通过注解声明切面,并将其纳入Spring管理。 还需要考虑XML配置的方式,虽然现在可能更常用注解,但传统项目可能仍在使用XML配置AOP,这也是实现方式之一。因此,应该涵盖注解和XML两种配置方法。 然后,需要比较Spring AOP和AspectJ的区别,引用[2]提到了两者的对比,可能需要说明Spring AOP是基于代理的,而AspectJ是在编译时或加载时织入,更强大但更复杂。这部分有助于用户理解不同实现方式的优缺点。 最后,根据用户提供的示例,回答需要结构清晰,分点列出,可能包括实现方式、配置方法、注解使用、代理类型比较等。同时,确保所有LaTeX格式正确,如类名或方法名使用行内格式,独立公式如果有的话使用$$,但可能这里用不上太多数学公式,更多的是代码示例。 检查是否有遗漏的部分,比如是否需要提到AOP的应用场景或优势,但用户问题集中在实现方式,所以应集中在技术细节上。确保引用的正确性,引用[1]、[2]、[3]中的相关内容需要适当标注。 总结步骤: 1. 介绍Spring AOP的两种动态代理实现方式:JDK和CGLIB,解释它们的区别和适用场景。 2. 说明基于注解的配置方式,如@Aspect、@Pointcut等,以及如何启用注解支持。 3. 提及XML配置方式,作为传统方法。 4. 比较Spring AOP与AspectJ的不同,帮助用户理解选择依据。 5. 提供代码示例,展示如何定义切面和使用注解。 6. 确保回答符合格式要求,正确使用引用标注,并生成相关问题。 现在需要组织这些点,确保逻辑连贯,语言准确,符合中文表达习惯,并正确应用引用标注。</think>### Spring AOP实现方式 Spring AOP主要通过**动态代理技术**实现面向切面编程,其核心实现方式可分为以下两类: --- #### 1. **基于JDK动态代理** - **适用场景**:当目标类实现了接口时,Spring AOP默认使用JDK动态代理。代理对象会实现相同的接口,并在调用目标方法前后插入切面逻辑。 - **特点**: - 仅支持接口代理,无法代理未实现接口的类。 - 运行时生成代理类,性能开销较小。 - **示例**: ```java public interface UserService { void saveUser(); } public class UserServiceImpl implements UserService { public void saveUser() { System.out.println("保存用户"); } } ``` JDK动态代理会为`UserService`接口生成代理对象,拦截方法调用[^1]。 --- #### 2. **基于CGLIB动态代理** - **适用场景**:当目标类未实现接口时,Spring AOP会使用CGLIB库生成目标类的子类作为代理。 - **特点**: - 可以代理普通类(无需接口)。 - 通过继承实现代理,因此`final`类或方法无法被代理。 - 性能略低于JDK动态代理,但功能更灵活。 - **配置方式**: 在Spring配置中设置`proxy-target-class=true`,强制使用CGLIB代理: ```xml <aop:aspectj-autoproxy proxy-target-class="true"/> ``` 或通过注解启用: ```java @EnableAspectJAutoProxy(proxyTargetClass = true) ``` --- #### 3. **基于注解的切面定义** Spring AOP支持通过`@Aspect`注解声明切面,结合以下注解定义通知(Advice): - **`@Pointcut`**:定义切入点表达式,指定哪些方法需要被拦截。 - **`@Before`/`@After`/`@Around`**:定义前置通知、后置通知和环绕通知。 - **示例**: ```java @Aspect @Component public class LoggingAspect { @Pointcut("execution(* com.example.service.*.*(..))") public void serviceMethods() {} @Around("serviceMethods()") public Object logExecutionTime(ProceedingJoinPoint joinPoint) throws Throwable { long start = System.currentTimeMillis(); Object result = joinPoint.proceed(); long end = System.currentTimeMillis(); System.out.println("方法执行时间:" + (end - start) + "ms"); return result; } } ``` 需在配置类中添加`@EnableAspectJAutoProxy`以启用注解支持[^2][^3]。 --- #### 4. **基于XML的切面配置(传统方式)** 通过XML文件定义切面和通知,适用于早期Spring版本或需要解耦代码的场景: ```xml <aop:config> <aop:aspect ref="loggingAspect"> <aop:pointcut id="serviceMethods" expression="execution(* com.example.service.*.*(..))"/> <aop:around pointcut-ref="serviceMethods" method="logExecutionTime"/> </aop:aspect> </aop:config> ``` 需在Bean定义中注入切面类`loggingAspect`。 --- #### 对比:Spring AOP与AspectJ | 特性 | Spring AOP | AspectJ | |---------------------|--------------------------------|----------------------------------| | **实现方式** | 动态代理(运行时织入) | 编译时/加载时织入 | | **性能** | 适用于轻量级场景 | 更高性能,适合复杂切面 | | **功能范围** | 仅支持方法级别拦截 | 支持字段、构造方法等更细粒度控制 | | **依赖** | 无需额外编译步骤 | 需编译器或类加载器支持 | ---
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值