基于接口的动态代理
要求:被代理类最少实现一个接口
提供者:JDK官方
基于接口的动态代理:
涉及的类:Proxy
创建代理对象的方法:newProxyInstance()
方法的三个参数:
ClassLoader:类加载器。用于加载代理对象字节码的。和被代理对象使用相同的类加载器
user.getClasss().getClassLoader()(有具体的实现类)
UserDao.class.getClassLoader();(有接口,没有实现类)
Class[]:字节码数组。用于让代理对象和被代理对象具有相同的行为(相同的方法)
被代理对象是一个类:写的是被代理对象实现的接口。
userServiceImpl.getClass().getInterfaces();
被代理对象是一个接口:写的是被代理对象本身。
new Class[]{UserDao.class}
InvocationHandler:一个接口。用于提供增强的方法的。写的是该接口的实现类。
该接口的实现类,谁用谁写。通常情况下提供一个匿名内部类即可。
public class ProxyTest {
public static void main(String[] args) {
//需求1:创建一个UserServiceImpl类的代理对象,让保存方法输出一句话“执行了保存用户操作”
UserService userService = new UserServiceImpl();
UserService proxyUserService = (UserService)Proxy.newProxyInstance(userService.getClass().getClassLoader(),
userService.getClass().getInterfaces(),
new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
if(method.getName().equals("save")){
System.out.println("执行了保存用户操作");
}
return null;
}
});
proxyUserService.save(null);
//需求2:创建一个UserDao的代理对象,让根据id查询用户输出一句话“执行了查询用户”
UserDao proxyUserDao = (UserDao)Proxy.newProxyInstance(UserDao.class.getClassLoader(),
new Class[]{UserDao.class},
new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
if(method.getName().equals("findById")){
System.out.println("执行了查询用户");
}
return null;
}
});
proxyUserDao.findById("1");
}