搭建公司网站费用,服务器搭建网站跑不满宽带,网站和数字界面设计师,wordpress 导出评论学习代理的设计模式的时候#xff0c;经常碰到的一个经典场景就是想统计某个方法的执行时间。 1 静态代理模式的产生 需求1. 统计方法执行时间
统计方法执行时间#xff0c;在很多API/性能监控中都有这个需求。 下面以简单的计算器为例子#xff0c;计算加法耗时。代码如下…学习代理的设计模式的时候经常碰到的一个经典场景就是想统计某个方法的执行时间。 1 静态代理模式的产生 需求1. 统计方法执行时间
统计方法执行时间在很多API/性能监控中都有这个需求。 下面以简单的计算器为例子计算加法耗时。代码如下 public Long add(Integer a, Integer b) {long result 0;for(int i0; i100000000; i) {result i a b;}return result;
}
方法很简单就是计算两数之和。这里为了方便统计耗时所以循环了很多次。 统计耗时那么实现起来也很简单
public Long add(Integer a, Integer b) {long start System.currentTimeMillis();long result 0;for(int i0; i100000000; i) {result i a b;}long end System.currentTimeMillis();System.out.println(cost time: (end - start) ms);return result;}
统计代码和业务代码杂糅在一起是否是感觉有点混乱有没有一种方法在不影响原有业务逻辑情况下实现统计耗时的功能 需求2.不影响原有业务逻辑实现方法的耗时统计 很快我们想到一种方法那就是定义一个父类专门做耗时统计业务代码通过抽象方法定义子类扩展具体的业务方法即可。
代码如下
public abstract class AbstractTime {public Long tickTock(Integer a, Integer b) {long start System.currentTimeMillis();long result calculate(a,b);long end System.currentTimeMillis();System.out.println(cost time: (end - start) ms);return result;}protected abstract Long add(Integer a, Integer b);
}
计算加法的耗时统计如下
public class AddTime extends AbstractTime {Overridepublic Long add(Integer a, Integer b) {long result 0;for(int i0; i100000000; i) {result i a b;}return result;}public static void main(String[] args){AddTime addTime new AddTime();addTime.tickTock(1, 2);}
}
很好实现无侵入式统计方法耗时完成既定目标。
那么问题又来了假如我们这个方法还需要继承另外一个类这个时候怎么办呢 需求3使用接口方式实现无侵入式统计方法耗时
我们先把需要实现的业务逻辑通过接口的方式封装起来定义一个接口。
public interface Calculator {Long add(Integer a, Integer b);
}实现业务接口的方法类
public class AddCalculator implements Calculator {Overridepublic Long calculate(Integer a, Integer b) {long result 0;for(int i0; i100000000; i) {result i a b;}return result;}
}
使用接口无侵入式实现方法耗时统计的方案就是设计模式里的经典方案代理模式。
这里使用代理模式实现的代理类如下
public class AddCalculatorProxy implements Calculator {private Calculator calculator;public AddCalculatorProxy(Calculator calculator) {this.calculator calculator;}Overridepublic Long calculate(Integer a, Integer b) {long start System.currentTimeMillis();// 具体的业务逻辑类Long result calculator.add(a, b);long end System.currentTimeMillis();System.out.println(cost time: (end - start) ms);return result;}public static void main(String[] args) {Calculator calculator new AddCalculator();Calculator proxy new AddCalculatorProxy(calculator);proxy.add(1, 2);}
}
这就是静态代理模式的推演过程。 2. 静态代理模式是什么
静态代理模式是一种设计模式用于在不修改目标对象的前提下通过代理对象来控制对目标对象的访问。以下是关于静态代理模式的详细说明 2.1. 定义
静态代理模式中代理类和目标类实现相同的接口代理类持有目标类的实例并通过代理类间接调用目标类的方法。代理类可以在方法执行前后添加额外的逻辑。 2.2. 特点
接口目标类和代理类都实现了同一个接口。 代理类代理类持有一个目标类的引用并在其方法中调用目标类的方法。 扩展性可以在不修改目标类的情况下通过代理类添加额外的功能如日志记录、性能监控等。 2.3. 代码分析
根据上面的例子以下是对静态代理模式的实现分析
接口定义
public interface Calculator {Long add(Integer a, Integer b);
}定义了一个 Calculator 接口包含一个 add 方法。 目标类
public class AddCalculator implements Calculator {Overridepublic Long calculate(Integer a, Integer b) {long result 0;for(int i0; i100000000; i) {result i a b;}return result;}
}AddCalculator 是目标类实现了 Calculator 接口。 提供了具体的业务逻辑例如计算两个数的和并进行循环累加。 代理类
public class AddCalculatorProxy implements Calculator {private Calculator calculator;public AddCalculatorProxy(Calculator calculator) {this.calculator calculator;}Overridepublic Long calculate(Integer a, Integer b) {long start System.currentTimeMillis();// 调用目标类的业务逻辑Long result calculator.add(a, b);long end System.currentTimeMillis();System.out.println(cost time: (end - start) ms);return result;}
}AddCalculatorProxy 是代理类也实现了 Calculator 接口。 持有一个 Calculator 类型的目标类实例。 在 calculate 方法中代理类在调用目标类的 add 方法前后添加了时间统计的逻辑。 测试代码
public static void main(String[] args) {Calculator calculator new AddCalculator();Calculator proxy new AddCalculatorProxy(calculator);proxy.add(1, 2);
}创建目标类实例 AddCalculator。 使用代理类 AddCalculatorProxy 包装目标类实例。 调用代理类的 add 方法时会自动执行代理类中的额外逻辑如性能统计。 2.4. 优点
职责分离将核心业务逻辑与附加功能分离符合单一职责原则。 增强功能可以在不修改目标类的情况下通过代理类添加新的功能。 2.5. 缺点
代码膨胀每新增一个目标类就需要创建一个对应的代理类可能导致代码量增加。 灵活性不足代理类和目标类必须实现相同的接口缺乏动态性。 2.6. 总结
静态代理模式适用于需要在目标类的基础上扩展功能的场景。它通过代理类封装目标类的行为同时保持接口的一致性。例子代码很好地展示了静态代理模式的应用通过代理类实现了性能监控的功能。