谈到Java的动态代理,大多数人会想到:
Java动态代理经常被忽略的一个应用场景是:动态实现接口方法而不需要实现类,这和作为代理类的场景是有区别的,如图:
那么动态实现接口方法的具体应用场景又是什么?
这个应用场景归纳起来就是:可以用于声明式接口编程,即根据接口声明的注解,参数以及返回值来动态生成接口类的实现,例如Spring中的Fegin以及Mybatis中的DAO,它们都没有具体的实现类,但是可以动态体现不同的行为。
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,获取仓库地址。