静态代理:由程序员创建或特定工具自动生成源代码,再对其编译。在程序运行前,代理类的.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:
人吃饭之前要洗手.....
人要吃饭 !!
人吃饭之后要洗碗...