大白话谈Java代理
引言
代理是什么?为什么要使用代理
代理模式(Proxy)是通过代理对象访问目标对象,这样可以在目标对象基础上增强额外的功能,如添加权限,访问控制和审计等功能。
相当于现在有一个目标类和一些目标方法,但是规定了你不能修改该类的任何代码,或者你无法访问该类的代码,有一个需求,让你对目标方法进行增强处理(在方法执行前,执行后做一些处理等等),这个时候就需要使用到代理;而且代理使用起来很方便,Java的代理包括:静态代理和动态代理。
一、静态代理
静态代理相对简单并且更容易理解,我们先从静态代理开始说起
首先,使用静态代理需要满足三个条件:目标接口、目标类、自定义代理类
目标类:一切的一切,不管如何代理,最终一定会落到目标类上;
目标接口:目标类实现的接口;
代理类:对目标类的方法进行增强;
1、目标接口
public interface TargetInteface {
void method1();
void method2();
int method3(Integer i);
}
2、目标类
public class Target implements TargetInteface {
@Override
public void method1() {
System.out.println("method1 running ...");
}
@Override
public void method2() {
System.out.println("method2 running ...");
}
@Override
public int method3(Integer i) {
System.out.println("method3 running ...");
return i;
}
}
3、代理类
在这里就可以看到代理的原理,即在代理类的内部方法内,在增强原方法的同时也调用原方法,实现了增强的需求
public class TargetProxy implements TargetInteface {
private Target target = new Target();
@Override
public void method1() {
System.out.println("执行方法前...");
target.method1();
System.out.println("执行方法后...");
}
@Override
public void method2() {
System.out.println("执行方法前...");
target.method2();
System.out.println("执行方法后...");
}
@Override
public int method3(Integer i) {
System.out.println("执行方法前...");
int method3 = target.method3(i);
System.out.println("执行方法后...");
return method3;
}
}
4、执行结果
public static void main(String[] args) {
TargetInteface target = new TargetProxy();
target.method1();
System.out.println("-----------------------------");
target.method2();
System.out.println("-----------------------------");
System.out.println(target.method3(3));
}
=============执行结果===============
执行方法前...
method1 running ...
执行方法后...
-----------------------------
执行方法前...
method2 running ...
执行方法后...
-----------------------------
执行方法前...
method3 running ...
执行方法后...
3
Process finished with exit code 0
二、动态代理
从上述代码可以看到代理的原理,也很容易可以理解;这里有一个很大的弊端,上述目标类只有三个方法,那么在代理类中需要写三个同样的方法;但是如果有几十上百个,上千个方法,那么代理类就同样需要些同样数量的方法,这在代码量和需求上都有没有必要,那么这种情况下怎么实现代理呢;这个时候 Java提供了两种动态代理模式:JDK动态代理和CGLIB动态代理