静态代理和动态代理
设计模式中的代理模式理解。代理模式又分为动态代理和静态代理。今天就来入门一下 这两种代理的区别。
首先 来说说 为什么要用代理模式呢。在我们日常程序开发中 经常会遇到这样的需求,要求改变某一个执行方法的逻辑,实际情况中 经常见到的 比如 在执行某个方法的时候 先来 查看执行此方法的权限。
通常的方法 一开始 我们也许会直接改变方法体 在里面进行硬代码植入,这样就改变了原来的实现代码,如果此方法在程序种用到很多 势必造成大量的修改,也会改变 原有的测试准确性。在加上设计模式的OCP(开闭原则) 这也是不可行的。
public interface UserManager { /** * 添加 */ public void add(); /** * 查询 * @return */ public int selectById(); /** * 删除 */ public void del(); }
public class UserManagerIml implements UserManager { @Override public void add() { // TODO Auto-generated method stub System.out.println("添加"); } @Override public int selectById() { // TODO Auto-generated method stub System.out.println("查询"); return 2; } @Override public void del() { // TODO Auto-generated method stub System.out.println("删除"); }
以上是接口和实现类的代码。
突然老板 哪天说 我要在这些方法执行前 加一些特殊的处理,以checkMethod() 方法为例
public void checkMethod(){ System.out.println("--------security----------"); }
难道我们要在每个方法调用前都要加上此方法吗,那是一个什么样的工作量呢?就算加上了,后期老板又说 不需要呢? 我们继续注释掉? 这个时候 就出现了我们常说的代理模式,首先来看静态代理的处理方法。
依然创建一个类 来实现 UserManager 接口。
public class StaticProxyTest implements UserManager { private UserManager um; public StaticProxyTest(UserManager um){ this.um = um; } @Override public void add() { checkMethod(); um.add(); } @Override public int selectById() { checkMethod(); return um.selectById(); } @Override public void del() { checkMethod(); um.del(); } //切点 public void checkMethod(){ System.out.println("--------security----------"); } }
实现代理功能的类,实现的方法也只是代理功能,并非真正的调用 最终实现方法。虽然此方法避免了 在程序各个地方 进行粘贴checkMethod方法,但是在 这个代理类中 还是重复了粘贴,并且此代理类 只代理了UserManager 一个类。拓展性 依然有局限。
动态代理:
import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; public class SecurityHandler implements InvocationHandler { private Object targetObject; //创建代理 public Object createProxyObject(Object object){ this.targetObject = object; return Proxy.newProxyInstance(object.getClass().getClassLoader(), object.getClass().getInterfaces(), this); } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { // TODO Auto-generated method stub //切点 checkMethod(); return method.invoke(targetObject, args); } public void checkMethod(){ System.out.println("--------security----------"); } }
动态代理 运用了几个 java jdk中常用的一些基础知识。在实现createProxyObject创建代理方法的时候 我们要实现一个代理Proxy.newProxyInstance创建代理,传入要代理类对象的装载器,接口方法,以及实现InvocationHandler接口的方法类,并返回代理类。代理类创建成功时 在执行中运用反射 Method调用 invoke()方法 传递参数。
public class Client { public static void main(String[] args) { SecurityHandler sh = new SecurityHandler(); UserManager um = (UserManager)sh.createProxyObject(new UserManagerIml()); um.add(); } }
执行打印:
--------security----------
添加
在以前常用的 一些 系统中 一些权限的处理和实现 就是动态代理实现。还有spring 最重要的特性aop也是动态代理原理。