重识Java动态代理(一)被忽略的场景

1,722 阅读4分钟

banner窄.png

铿然架构  |  作者  /  铿然一叶 这是铿然架构的第 15 篇原创文章

相关阅读:

萌新快速成长之路
如何编写软件设计文档
JAVA编程思想(一)通过依赖注入增加扩展性
JAVA编程思想(二)如何面向接口编程
JAVA编程思想(三)去掉别扭的if,自注册策略模式优雅满足开闭原则
JAVA编程思想(四)Builder模式经典范式以及和工厂模式如何选?
Java编程思想(七)使用组合和继承的场景
JAVA基础(一)简单、透彻理解内部类和静态内部类
JAVA基础(二)内存优化-使用Java引用做缓存
JAVA基础(三)ClassLoader实现热加载
JAVA基础(四)枚举(enum)和常量定义,工厂类使用对比
JAVA基础(五)函数式接口-复用,解耦之利刃
Seata源码(一)初始化
Seata源码(二)事务基础对象
Seata源码(三)事务处理类结构和流程
Seata源码(四)全局锁GlobalLock
Seata源码(五)Seata数据库操作
Seata源码(六)Seata的undo日志操作
Seata源码(七)Seata事务故障处理
Seata源码(八)Seata事务生命周期hook
Seata源码(九)TCC核心类和处理逻辑
Seata源码(十)RM接收到请求后的调用过程
Seata源码(十一)TC接收到请求后的处理过程\


一、关于Java动态代理

谈到Java的动态代理,大多数人会想到:

  1. 一种代理模式,可以在实际调用方法前、后增加公共方法调用,例如记录日志,用户鉴权。
  2. cglib代理和JDK动态代理的优缺点
  3. 从一些应用场景来看,代理可做的事情也可以通过AOP实现,所以暂时我们还用不到动态代理。

二、被忽略的场景

Java动态代理经常被忽略的一个应用场景是:动态实现接口方法而不需要实现类,这和作为代理类的场景是有区别的,如图:

  1. 左图的代理类实现了接口类,并持有实现类的实例,在调用实现类的方法前、后可以调用代理类的方法,实现类的方法最终也会被调用。
  2. 右图没有实现类,动态生成的代理类是唯一的实现类。

那么动态实现接口方法的具体应用场景又是什么?
这个应用场景归纳起来就是:可以用于声明式接口编程,即根据接口声明的注解,参数以及返回值来动态生成接口类的实现,例如Spring中的Fegin以及Mybatis中的DAO,它们都没有具体的实现类,但是可以动态体现不同的行为。

三、动态实现接口方法Demo

1.Person.java

public interface Person {
	@Say(content = "你好!")
	void doSomeThing();
}

一个接口类,通过注解来声明要说什么内容。

2.Dog.java

public interface Dog {
	@Say(content = "汪,汪!")
	void doSomeThing();
}

同样的接口类,不过和人要说的内容不同。

3.Say.java

@Documented
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Say {
	String content() default "";
}

注解类,定义“说”这个行为,说的内容可在使用时声明。

4.AnimalProxy.java

public class AnimalProxy implements InvocationHandler {
	@Override
	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
		// 获取接口的注解,根据注解来操作
		Say say = method.getAnnotation(Say.class);
		if (null != say) {
			// 实例可以通过注解决定如何实现需要的业务操作,这里只是简单的获取注解内容
			System.out.println(say.content());
		}
		// Object result = method.invoke(target, args); 这里不需要调用目标方法,因为是接口类
		return null;
	}
}

代理类,根据接口类的注解实现动态处理。

5.ProxyFactory.java

public class ProxyFactory {
	public static Object getProxy(Class<?> clazz) {
		AnimalProxy proxy = new AnimalProxy();
		Object newInstanceObject = Proxy.newProxyInstance(clazz.getClassLoader(), new Class[] { clazz }, proxy);
		return (Object) newInstanceObject;
	}
}

代理工厂类。

6.TestMain.java

public class TestMain {
	public static void main(String[] args) {
		Person person = (Person) ProxyFactory.getProxy(Person.class);
		person.doSomeThing();
		Dog dog = (Dog) ProxyFactory.getProxy(Dog.class);
		dog.doSomeThing();
	}
}

测试类,可以看到Person和Dog的doSomeThing()方法并没有任何实现类来实现,但是却可以被调用,并表现出不同的行为,输出结果为:

你好!
汪,汪!

本篇我们已经初步了解如何动态实现接口方法,在后续的文章中将逐步讲解:如何在Spring中完成一个类似Fegin的声明式编程例子。

完整实例代码扫码加入微信公众号并回复:webfullstack,获取仓库地址。

<---左边点赞!

end.