java动态代理

静态代理:由程序员创建或特定工具自动生成源代码,再对其编译。在程序运行前,代理类的.class 文件就已经存在了;
动态代理:在程序运行时,运用反射机制动态创建而成。

JDK 的动态代理创建机制—- 通过接口

使用 JDK 实现动态代理,JDK 的动态代理只能对实现了接口的类进行代理, 被代理的对象必须要实现接口。看下面的案例。

public interface IPerson {
 void eat();
}
public class Person implements IPerson {
  @Override 
  public void eat() { 
    System.out.println("人要吃饭!!"); 
  }
}
public class PersonInvocationHandler implements InvocationHandler {                
  private Person person;

  public PersonInvocationHandler(Person person) {
    this.person = person;
  } 
  @Override 
  public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { 
    System.out.println("人吃饭之前要洗手....."); 
    Object invoke = method.invoke(person, args); 
    System.out.println("人吃饭之后要洗碗..."); // 这里可以没有返回值, 具体看方法是否需要返回值. 
    return invoke;
  }
}
public class JDKPorxy { 
  public static void main(String[] args) { 
    Person p = new Person(); // 这里创建一个动态代理类. 人要吃饭之前要 洗手, 吃饭之后还有洗碗. 
    Object o = Proxy.newProxyInstance(p.getClass().getClassLoader(), 
      p.getClass().getInterfaces(), 
      new PersonInvocationHandler(p)); // 必须使用接口 
    IPerson person = (IPerson) o; 
    person.eat(); //end main 
  }
}

output:

人吃饭之前要洗手.....
人要吃饭 !! 
人吃饭之后要洗碗...

cglib 生成动态代理类的机制—- 通过类继承

JDK 中提供的生成动态代理类的机制有个鲜明的特点是: 某个类必须有实现的接口,而生成的代理类也只能代理某个类接口定义的方法,比如:如果上面例子的 ElectricCar 实现了继承自两个接口的方法外,另外实现了方法 bee() , 则在产生的动态代理类中不会有这个方法了!更极端的情况是:如果某个类没有实现接口,那么这个类就不能同 JDK 产生动态代理了!幸好我们有 cglib。“CGLIB(Code Generation Library),是一个强大的,高性能,高质量的 Code 生成类库,它可以在运行期扩展 Java 类与实现 Java 接口。”

public class User { 
  public void eat() { 
    System.out.println("人要吃饭!!"); 
  }
}

实现方法的拦截器

public class CGlibInterceptor implements MethodInterceptor {
  @Override
  public Object intercept(Object obj, Method method, Object[] args,MethodProxy proxy)
   throws Throwable {
    System.out.println("人吃饭之前要洗手.....");
    Object invoke = proxy.invokeSuper(obj, args); 
    System.out.println("人吃饭之后要洗碗..."); // 这里可以没有返回值, 具体看方法是否需要返回值. 
    return invoke;
  }
}
public class CGlibProxy { 

  public static void main(String[] args) {
    User user = new User(); 
    //cglib 中加强器,用来创建动态代理  
    Enhancer enhancer = new Enhancer();
    // 设置要创建动态代理的类 
    enhancer.setSuperclass(user.getClass()); 
    // 设置回调(方法拦截器),这里相当于是对于代理类上所有方法的调用,都会调用 CallBack,而 Callback 则需要实行 intercept() 方法进行拦截 
    enhancer.setCallback(new CGlibInterceptor()); 
    User u = (User) enhancer.create();
    u.eat(); 
    //end main 
  }
}

output:

人吃饭之前要洗手.....
人要吃饭 !! 
人吃饭之后要洗碗...

你可能感兴趣的:(java动态代理)