代理可以理解为请一个中间人帮忙处理一些事情。
代理支持任意接口类型的实现类对象做代理,也可以直接为接本身做代理。
可以为被代理对象的所有方法做代理。
可以在不改变方法源码的情况下,实现对方法功能的增强。
简化了编程工作、提高了软件系统的可扩展性,也提高了开发效率。
关键:必须有接口,实现类要实现接口(代理通常是基于接口实现的)。创建一个实现类的对象,该对象为业务对象。为业务对象做一个代理对象。
代理的一个案例快速理解
需求:
业务功能的性能统计:某企业用户管理业务,需包含用户登录,用户删除,用户查询功能,并要统计每个功能的耗时。
分析:
- 定义一个UserService表示用户业务接口,规定必须完成用户登录,用户删除,用户查询功能。
- 定义一个实现类UserServiceE实现UserService,并完成相关功能,且统计每个功能的耗时。
- 定义测试类,创建实现类对象,调用方法。
实现:
1.业务接口:
public interface UserService { public String login(String name,String psd); public void deleteUser(); public String selectUser(); }
2.实现类
这里为了模拟方便,每个业务方法只是用sleep延时。
public class UserServiceE implements UserService { @Override public String login(String name,String psd) { String rs="密码错误!"; if("admin".equals(name)&&"123456".equals(psd)) rs="密码正确"; try { Thread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } return rs; } @Override public void deleteUser() { System.out.println("删除用户~"); try { Thread.sleep(1500); } catch (InterruptedException e) { e.printStackTrace(); } } @Override public String selectUser() { System.out.println("查询用户~"); try { Thread.sleep(2500); } catch (InterruptedException e) { e.printStackTrace(); } return "查询了用户~"; } }
3.代理类
代理类里面需要传入实现类的对象,在里面就用到了反射的一些知识。
- 写好一个接口类型的静态方法,直接写return返回代理 Proxy
- 方法newProxyInstance
- 对于newProxyInstance有三个参数:类、接口和InvocationHandler
- 实现InvocationHandler
- 在invoke里面运行目标方法,invoke反射。
public class proxyUtil { public static UserService getProxy(UserService obj){ return (UserService) Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj.getClass().getInterfaces(), new InvocationHandler() { @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { long startTime=System.currentTimeMillis(); Object rs = method.invoke(obj,args); long endTime=System.currentTimeMillis(); System.out.println(method.getName()+" 耗时:"+(endTime-startTime)/1000.0+"秒。"); return rs; } }); } }
4.main方法
使用动态代理后,就不用再在各个方法中进行修改代码来统计了,而且可扩展性更好。如果对代理类实现泛型,就更加通用了。
public static void main(String[] args) { UserService u= proxyUtil.getProxy( new UserServiceE()); System.out.println(u.login("admin", "123456")); u.deleteUser(); System.out.println(u.selectUser()); }