Java动态代理与字节码实现AOP

Java动态代理

jdk动态代理机制,在运行期间,为相应的接口动态生成对应的代理对象。将横切关注点封装到动态代理的InvocationHandler中,在系统运行期间,根据横切关注点需要织入的模块位置,将横切逻辑织入到相应的代理类中,以动态类为载体的横切逻辑。SpringAOP默认的机制为动态代理。
动态代理是基于代理模式设计的一种方式,代理模式中一般涉及到的对象有:
接口抽象:是被访问者或者访问资源的抽象。
目标类(被代理对象):被访问者或者被访问资源的具体实现。即代理对象只需要关注自己所做的事情。
代理类:被访问者或者被访问资源的代理实现类。
客户访问端:代表访问者的抽象角色
jdk动态代理:一类一接口

一类:Proxy类
一接口:InvocationHandler接口,重写invoke方法,在invoke方法中完成非业务逻辑的织入。
Proxy类使用Proxy.newProxyInstance(参数一,参数二,参数三)方法创建代理对象。
参数一:实现接口的类的类加载器
参数二:代理类的接口数组
参数三:实现接口的类以及传递被代理对象的对象


import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.time.Instant;

//客户端
public class DynamaticProxyDemo {

	public static void main(String[] args) {
		ISubject2 obj = (ISubject2) Proxy.newProxyInstance(ProxyRunner.class.getClassLoader(),
				new Class[] { ISubject2.class }, new ProxyRunner(new Target2()));
		obj.proxyMethod();
	}

}

//动态代理
class ProxyRunner implements InvocationHandler {
	Object target;

	ProxyRunner(Object obj) {
		this.target = obj;
	}

	@Override
	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
		if (method.getName().equals("proxyMethod")) {
			System.out.println(Instant.now());
			method.invoke(target, args);
			System.out.println(Instant.now());
		}
		return null;
	}
}

//目标类继承接口
class Target2 implements ISubject2 {
	@Override
	public void proxyMethod() {
		System.out.println("代理对象2的代理方法");
	}
}

interface ISubject2 {
	void proxyMethod();
}

//目标类2继承接口
class Target implements ISubject {
	@Override
	public void proxyMethod() {
		System.out.println("代理对象的代理方法");
	}
}

interface ISubject {
	void proxyMethod();
}

字节码技术

SpringAOP发现目标对象实现了响应的接口,则采用的是动态代理方式,如果目标对象没有实现任何接口Interface,SpringAOP会尝试使用一个称为CGLIB的开源动态的字节码生成类库,为目标生成动态的代理对象实例。
CGLIB可以对实现了某种接口的类,或者没有实现任何接口的类进行扩展。
字节码结束使用MethodInterceptor接口,重写intercept方法。
测试需要引入的包:

引入的jar包


import java.lang.reflect.Method;
import java.time.Instant;

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

public class ClazzSkillDemo {
	public static void main(String[] args) {
	//使用Enhancer 对象设置目标类以及实现MethodInterceptor 接口的类
		Enhancer enhance = new Enhancer();
		enhance.setSuperclass(TargetClass.class);
		enhance.setCallback(new ClazzRunner());
		TargetClass proxy = (TargetClass) enhance.create();
		proxy.targetMethod();
	}
}

//实现MethodInterceptor接口
class ClazzRunner implements MethodInterceptor {
	@Override
	public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
		if (method.getName().equals("targetMethod")) {
			System.out.println(Instant.now());
			// method.invoke(obj, args);
			proxy.invokeSuper(obj, args);
			System.out.println(Instant.now());
		}
		return null;
	}

}

//目标类,目标类没有实现接口,没有继承任何类
class TargetClass {
	public void targetMethod() {
		System.out.println("这是一个需要增强的方法");
	}
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值