本文最后更新于:2 个月前
代理模式
基本介绍:
- 代理模式:为一个对象提供一个替身,以控制对这个对象的访问(被代理的对象)。即通过代理对象访问目标对象,这样的好处是可以再目标对象实现的基础上,增强额外的功能操作,即扩展目标对象的功能。
- 被代理的对象可以是远程对象、创建开销大的对象或者需要安全控制的对象
- 代理模式有三种不同的形式,主要有三种
- 静态代理
- 动态代理(
JDK
代理,接口代理)
Cglib
代理(可以再内存动态的创建对象,而不需要实现接口,属于动态代理的范畴)
1.静态代理
静态代理在使用时,需要定义接口或者父类(即目标对象)与代理对象一起实现相同的接口或者是继承相同父类
用例图:
代码实现:
1 2 3 4 5
| public interface ITeacherDao { void teach(); }
|
1 2 3 4 5 6 7
| public class TeacherDao implements ITeacherDao { @Override public void teach() { System.out.println(" 老师授课中 。。。。。"); } }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
| public class TeacherDaoProxy implements ITeacherDao{ private ITeacherDao target; public TeacherDaoProxy(ITeacherDao target) { this.target = target; }
@Override public void teach() { System.out.println("开始代理 完成某些操作。。。。。 "); target.teach(); System.out.println("提交。。。。。"); } }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| public class Client {
public static void main(String[] args) { TeacherDao teacherDao = new TeacherDao(); TeacherDaoProxy teacherDaoProxy = new TeacherDaoProxy(teacherDao); teacherDaoProxy.teach(); } }
|
静态代理优缺点:
- 优点‘:在不修改目标对象的功能前提下,能通过代理对象对目标功能进行扩展
- 缺点:因为代理对象需要与目标对象实现一样的接口,所以会有很多代理类
- 一旦接口增加方法,目标对象与代理对象都要维护
2.动态代理
基本介绍
- 代理对象,不需要实现接口,但是目标对象要实现接口,否则不能用动态代理
- 代理对象的生成,是利用
JDK
的API
,动态的在内存中构建代理对象
- 动态代理也叫:
JDK
代理,接口代理
JDK
中生成代理对象的API
代理类所在包:java.lang.reflect.Proxy
JDK
实现代理类只需要使用newProxyInstance
方法,但是该方法需要接收三个参数,完整方法为
1 2 3 4
| >public static Object newProxyInstance( >ClassLoader loader, >Class<?>[] interfaces, >InvocationHandler h)
|
代码实现
1 2 3 4 5
| public interface ITeacherDao { void teach(); void sayHello(String name); }
|
1 2 3 4 5 6 7 8 9 10 11 12
| public class TeacherDao implements ITeacherDao { @Override public void teach() { System.out.println(" 老师授课中.... "); }
@Override public void sayHello(String name) { System.out.println("hello " + name); } }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55
| public class ProxyFactory<T> {
private T target;
public ProxyFactory(T target) { this.target = target; }
public T getProxyInstance() {
return (T) Proxy.newProxyInstance( target.getClass().getClassLoader(), target.getClass().getInterfaces(), new InvocationHandler() { @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { String name = method.getName(); if (!name.equals("")){ System.out.println("执行了对应方法"); System.out.println("方法执行前 代理开始了 "); Object returnVal = method.invoke(target, args); System.out.println("方法执行后 代理又开始了"); return returnVal; } return method.invoke(target,args); } }); } }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
| public class Client { public static void main(String[] args) { ITeacherDao target = new TeacherDao();
ITeacherDao proxyInstance = new ProxyFactory<ITeacherDao>(target).getProxyInstance();
System.out.println("proxyInstance=" + proxyInstance.getClass()); proxyInstance.sayHello(" 龙龙龙 "); } }
|
3.Cglib代理
基本介绍:
静态代理和JDK
代理模式都要求目标对象是实现一个接口,但是有时候目标对象只是一个单独的对象,并没有实
现任何的接口,这个时候可使用目标对象子类来实现代理-这就是Cglib
代理
Cglib代理也叫作子类代理,它是在内存中构建一个子类对象从而实现对目标对象功能扩展,有些书也将Cglib代
理归属到动态代理。
Cglib
是一个强大的高性能的代码生成包,它可以在运行期扩展java
类与实现java
接口.它广泛的被许多AOP
的
框架使用,例如Spring AOP
,实现方法拦截
在AOP
编程中如何选择代理模式:
- 目标对象需要实现接口,用
JDK
代理
- 目标对象不需要实现接口,用
Cglib
代理
Cglib
包的底层是通过使用字节码处理框架ASM
来转换字节码并生成新的类
设计类图:
代码实现
1 2 3 4 5 6 7
| public class TeacherDao { public String teach() { System.out.println(" 老师授课中 , 我是cglib代理,不需要实现接口 "); return "hello"; } }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35
| public class ProxyFactory implements MethodInterceptor {
private Object target; public ProxyFactory(Object target) { this.target = target; }
public Object getProxyInstance() { Enhancer enhancer = new Enhancer(); enhancer.setSuperclass(target.getClass()); enhancer.setCallback(this); return enhancer.create(); } @Override public Object intercept(Object arg0, Method method, Object[] args, MethodProxy arg3) throws Throwable { System.out.println("Cglib代理模式 ~~ 开始"); Object returnVal = method.invoke(target, args); System.out.println("Cglib代理模式 ~~ 提交"); return returnVal; } }
|
1 2 3 4 5 6 7 8 9 10 11 12 13
| public class Client { public static void main(String[] args) { TeacherDao target = new TeacherDao(); TeacherDao proxyInstance = (TeacherDao)new ProxyFactory(target).getProxyInstance();
String res = proxyInstance.teach(); System.out.println("res=" + res); } }
|
4.其他变体
几种常见的代理模式介绍一几种变体
- 防火墙代理:内网通过代理穿透防火墙,实现对公网的访问。
- 缓存代理:比如 当请求图片文件等资源时,先到缓存代理取,如果取到资源则ok,如果取不到资源,再到公网或者数据库取,然后缓存。
- 远程代理:远程对象的本地代表,通过它可以把远程对象当本地对象来调用。远程代理通过网络和真正的远程对象沟通信息。
- 同步代理:主要使用在多线程编程中,完成多线程间同步工作