深圳 网站设计师 招聘,自己的网站如何做分销,国内最大的c2c网站,wordpress物流主题代理模式#xff1a;控制访问的设计模式
什么是代理模式#xff1f;
代理模式是一种常见的设计模式#xff0c;它允许通过代理对象来控制对真实对象的访问。代理模式的主要目的是在不改变原始对象的情况下#xff0c;提供额外的功能或控制访问。
为什么要使用代理模式控制访问的设计模式
什么是代理模式
代理模式是一种常见的设计模式它允许通过代理对象来控制对真实对象的访问。代理模式的主要目的是在不改变原始对象的情况下提供额外的功能或控制访问。
为什么要使用代理模式
代理模式有以下几个主要的应用场景
访问控制代理模式可以限制对真实对象的直接访问只有通过代理对象才能访问真实对象。这样可以实现对真实对象的访问控制例如权限验证、身份验证等。增加额外功能代理模式可以在不修改真实对象的情况下为其增加额外的功能。代理对象可以在调用真实对象的方法前后执行一些额外的操作例如日志记录、性能监控、缓存等。远程访问代理模式可以实现远程访问即通过代理对象访问位于不同地址空间的真实对象。这对于分布式系统或跨网络的应用程序非常有用。
代理模式的两种分类
静态代理
静态代理是在编译时就已经确定代理对象和真实对象的关系。代理对象和真实对象实现相同的接口或继承相同的父类代理对象持有真实对象的引用并在调用真实对象的方法前后执行一些额外的操作。
优点 简单易懂
缺点 需要为每个真实对象编写一个代理类当真实对象较多时会导致代码冗余
案例 一个简单的日志记录功能
假设我们有一个 UserService 接口和一个实现类 UserServiceImpl它提供了用户管理的一些基本操作方法如添加用户、删除用户等。现在我们需要在每个方法执行前后记录日志例如在方法执行前打印 “Before” 的日志在方法执行后打印 “After” 的日志。
// UserService 接口
public interface UserService {void addUser();void deleteUser();
}// 实现类 UserServiceImpl
public class UserServiceImpl implements UserService {Overridepublic void addUser() {System.out.println(添加用户...);}Overridepublic void deleteUser() {System.out.println(删除用户...);}
}// 静态代理类
public class UserServiceProxy implements UserService {private UserService userService;public UserServiceProxy(UserService userService) {this.userService userService;}Overridepublic void addUser() {System.out.println(Before...);userService.addUser();System.out.println(After...);}Overridepublic void deleteUser() {System.out.println(Before...);userService.deleteUser();System.out.println(After...);}// 其他方法同样的方式实现
}接下来我们可以使用代理类来代替真实对象进行操作。
public class Client {public static void main(String[] args) {// 静态代理UserServiceProxy proxy new UserServiceProxy(new UserServiceImpl());proxy.addUser();System.out.println(------------------);proxy.deleteUser();}
}结果
动态代理
动态代理是一种在运行时动态生成代理类的代理模式。它可以在不修改原始类的情况下为原始类提供额外的功能或控制访问。在Java中有两种常见的动态代理方式JDK动态代理和CGLIB动态代理
JDK动态代理
JDK动态代理是通过Java的反射机制实现的。它要求被代理的类必须实现一个接口。JDK动态代理提供了一个Proxy类和一个InvocationHandler接口通过这两个类可以动态生成代理类。
案例 简单的日志记录功能
定义一个接口 UserService它提供了用户管理的一些基本操作方法。
public interface UserService {void addUser();void deleteUser();
}真实的用户服务类UserServiceImpl它实现了 UserService 接口。
public class UserServiceImpl implements UserService {Overridepublic void addUser() {System.out.println(添加用户...);}Overridepublic void deleteUser() {System.out.println(删除用户...);}
}通过Proxy类创建代理对象UserServiceProxy对目标对象的方法进行拦截和增强
public class UserServiceProxy {private Object target;public UserServiceProxy(Object target) {this.target target;}/*** target目标对象即要被代理的对象。* getProxyInstance()该方法返回一个代理对象该代理对象实现了目标对象所实现的接口并在方法调用前后执行额外的逻辑。* Proxy.newProxyInstance()这是Java提供的创建代理对象的方法。它接受三个参数类加载器、目标对象实现的接口数组和一个InvocationHandler对象。* InvocationHandler这是一个接口用于定义代理对象的方法调用处理逻辑。在invoke()方法中我们可以在目标方法调用前后执行额外的逻辑。*/public Object getProxyInstance(){return Proxy.newProxyInstance(target.getClass().getClassLoader(),target.getClass().getInterfaces(),new InvocationHandler() {Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println(gdk代理模式---开始);Object invoke method.invoke(target, args);System.out.println(gdk代理模式---结束);return invoke;}});}
}最后我们可以使用 Proxy 类的 newProxyInstance 方法来创建代理对象。
public class Client {public static void main(String[] args) {// JDK动态代理UserService proxy (UserService) new UserServiceProxy(new UserServiceImpl()).getProxyInstance();proxy.addUser();System.out.println(-----------------);proxy.deleteUser();}
}CGLIB动态代理
CGLIB动态代理是通过继承被代理类来实现的它不要求被代理的类实现接口。CGLIB动态代理使用了字节码生成库来生成代理类。
案例简单的日志记录功能
定义一个类 UserService它提供了用户管理的一些基本操作方法。
public interface UserService {void addUser();void deleteUser();// 带返回结果int getUserCount();
}真实的用户服务类UserServiceImpl它实现了 UserService 接口。
public class UserServiceImpl implements UserService {Overridepublic void addUser() {System.out.println(添加用户...);}Overridepublic void deleteUser() {System.out.println(删除用户...);}// 带返回结果Overridepublic int getUserCount() {System.out.println(查询用户数量...);return 66;}
}一个代理类 UserServiceProxy 它实现了MethodInterceptor 接口。
public class UserServiceProxy implements MethodInterceptor {// 维护一个目标对象private Object target;// 构造器传入一个被代理的对象public UserServiceProxy(Object target) {this.target target;}// 返回一个代理对象target的代理对象public Object getProxyInstance() {// 1. 创建一个工具类Enhancer enhancer new Enhancer();// 2. 设置父类enhancer.setSuperclass(target.getClass());// 3. 设置回调函数enhancer.setCallback(this);// 4. 创建子类对象即代理对象return enhancer.create();}// 重写 intercept 方法会调用目标对象的方法Overridepublic Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {System.out.println(cglib代理模式---开始);Object returnVal method.invoke(target, objects);System.out.println(cglib代理模式---结束);return returnVal;}
}最后我们可以使用UserServiceProxy的getProxyInstance方法创建代理对象实例调用方法。
public static void main(String[] args) {// 创建目标对象UserService userService new UserServiceImpl();// 获取代理对象并将目标对象传递到代理对象UserService proxyInstance (UserService) new UserServiceProxy(userService).getProxyInstance();// 执行代理对象的方法触发intecept方法从而实现目标对象的调用proxyInstance.addUser();System.out.println(-------------);int userCount proxyInstance.getUserCount();System.out.println(userCount);}jdk动态代理和cglib动态代理的区别
实现方式jdk动态代理是通过反射实现的而cglib动态代理是通过继承目标类来实现的。目标类限制jdk动态代理要求目标类必须要实现接口而cglib动态代理则没有这个限制。性能jdk动态代理相对于cglib动态代理来说因为实现方式不同生成的代理类的效率会低一些。对象类型jdk动态代理只能代理实现了接口的类cglib通过继承实现不能代理 final 类。依赖库jdk动态代理是Java自带的库不需要额外的依赖而cglib动态代理需要依赖cglib库。
总结
代理模式是一种非常有用的设计模式它可以实现访问控制、增加额外功能和远程访问。静态代理在编译时确定代理对象和真实对象的关系而动态代理在运行时动态生成代理对象。动态代理又分为jdk动态代理和cglib动态代理分别基于接口和类来实现代理功能。根据具体的需求和场景选择适合的代理模式可以提高代码的可维护性和灵活性。 区别
与适配器模式的区别适配器模式主要改变所考虑对象的接口而代理模式不能改变所代理类的接口。与装饰器模式的区别装饰器模式为了增强功能而代理模式是为了加以控制。