代理模式中,代理类与主体类实现同样的接口,代理类持有实体类的引用,并接受客户端对代理类中实体引用的外部注入,并代理实体类的功能。 注:描述中的这种外部注入形式有个专有技术名词:依赖注入
每多一个真实角色就会产生一个代理角色:代码量会翻倍,开发效率低
1.接口
public interface UsersService { public void add(); public void delete(); public void update(); public void query(); }2.真实角色
public class UserServiceImp implements UsersService{ public void add() { System.out.println("添加"); } public void delete() { System.out.println("删除"); } public void update() { System.out.println("修改"); } public void query() { System.out.println("查询"); } }3.代理角色
public class UserServiceProxy implements UsersService{ private UserServiceImp userService; public void setUserService(UserServiceImp userService) { this.userService = userService; } public void add() { log("add"); userService.add(); } public void delete() { log("delete"); userService.delete(); } public void update() { log("update"); userService.update(); } public void query() { log("query"); userService.query(); } public void log(String msg){ System.out.println("使用了"+msg+"方法"); } }4.客户访问代理角色
public class User { public static void main(String[] args) { UserServiceImp userService = new UserServiceImp(); UserServiceProxy proxy = new UserServiceProxy(); proxy.setUserService(userService); proxy.add(); } }运行结果:
使用了add方法 添加 Process finished with exit code 0 JDK中的动态代理是通过反射类Proxy以及InvocationHandler回调接口实现的;但是,JDK中所要进行动态代理的类必须要实现一个接口,也就是说只能对该类所实现接口中定义的方法进行代理,这在实际编程中具有一定的局限性,而且使用反射的效率也并不是很高。
JDK动态代理核心代码:
1.实现这个接口InvocationHandler,创建代理对象
2.Proxy提供创建动态代理类和实例的静态方法:Proxy.newProxyInstance()
//自动生成代理类 public class DynamicProxy implements InvocationHandler { //被代理的接口 private Object target; //set方法 public void setTarget(Object target) { this.target = target; } //生成得到代理类 public Object getProxy(){ return Proxy.newProxyInstance(this.getClass().getClassLoader(), target.getClass().getInterfaces(),this); } //处理代理实例并返回结果 public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { log(method.getName()); Object result = method.invoke(target,args); return result; } //代理角色的附属业务的例子 public void log(String msg){ System.out.println("使用了"+msg+"方法"); } }target可表示任务场景中的真实角色,这是一个模板!
客户实现业务:
public class User { public static void main(String[] args) { //真实角色 UserServiceImp userService = new UserServiceImp(); //代理角色,因为是动态代理所以不存在 DynamicProxy dp = new DynamicProxy(); //设置代理对象 dp.setTarget(userService); //动态生成代理类 UsersService proxy =(UsersService) dp.getProxy(); //调用真实角色中的方法 proxy.delete(); } }