
使用 Builder 模式处理函数或浮点数参数
在编写需要灵活配置参数的函数时,我们经常会遇到这样的需求:某些参数既可以接受一个具体的数值,也可以接受一个函数,该函数根据输入动态地计算出数值。如果直接使用方法重载,会导致大量的重复代码,难以维护。本文将介绍如何使用 Builder 模式来解决这个问题,提供一种更优雅、更灵活的解决方案。
Builder 模式简介
Builder 模式是一种创建型设计模式,它将一个复杂对象的构建过程与其表示分离,使得同样的构建过程可以创建不同的表示。简单来说,Builder 模式允许我们一步一步地构建一个对象,而不是一次性地将所有参数传递给构造函数。这在参数数量较多,且部分参数可选的情况下非常有用。
实现 RangeBuilder
假设我们需要创建一个 RangeBuilder 类,用于生成一个函数,该函数接受一个 PVector 对象作为输入,并返回一个 Double 值。该函数的核心逻辑是根据输入的 PVector 对象,结合 scale、min 和 max 三个参数,计算出一个映射后的值。这三个参数都可以是 Double 类型的数值,也可以是一个 Function
下面是 RangeBuilder 类的代码实现:
import java.util.List;
import java.util.function.Function;
import java.util.stream.Stream;
import java.io.IOException;
class PVector {
public double x;
public double y;
public PVector(double x, double y) {
this.x = x;
this.y = y;
}
}
public class RangeBuilder {
private Function scale;
private Function min;
private Function max;
public RangeBuilder scale(Function scale) {
this.scale = scale;
return this;
}
public RangeBuilder scale(double scale) {
return scale(in -> scale);
}
public RangeBuilder min(Function min) {
this.min = min;
return this;
}
public RangeBuilder min(double min) {
return min(in -> min);
}
public RangeBuilder max(Function max) {
this.max = max;
return this;
}
public RangeBuilder max(double scale) {
return max(in -> scale);
}
public Function build() {
return pv -> map(noise(pv.x * scale.apply(pv), pv.y * scale.apply(pv)), 0, 1, min.apply(pv), max.apply(pv));
}
// Mock noise and map functions for demonstration
private double noise(double x, double y) {
// Replace with your actual noise function implementation
return (Math.sin(x) + Math.cos(y)) / 2.0; // Example noise function
}
private double map(double value, double start1, double stop1, double start2, double stop2) {
return start2 + (stop2 - start2) * ((value - start1) / (stop1 - start1));
}
public static void main(String... args) throws IOException {
List pvs = List.of(new PVector(0, 0));
Stream s = pvs.stream().map(new RangeBuilder().scale(.02)
.min(0)
.max(255)
.build());
Stream s2 = pvs.stream().map(new RangeBuilder().scale(new RangeBuilder().scale(.02)
.min(.1)
.max(.002)
.build())
.min(0)
.max(255)
.build());
s.forEach(System.out::println);
s2.forEach(System.out::println);
}
} 代码解释:
Sail企业网站管理系统(以下称Sail)是一个基于PHP+Mysql架构的企业网站管理系统。Sail 采用模块化方式开发,功能强大灵活易于扩展,并且完全开放源代码,面向大中型站点提供重量级企业网站建设解决方案。2年来,凭借Silence长期积累的丰富的Web开发及数据库经验和勇于创新追求完美的设计理念,使得Sail得到了很多公司和网站的认可,并且越来越多地被应用到大中型商业网站主要功能:单页、文
-
成员变量: scale、min 和 max 都是 Function
类型的成员变量,用于存储对应的函数。 -
Builder 方法: scale(Function
scale)、min(Function min) 和 max(Function max) 方法用于设置对应的函数。这些方法都返回 RangeBuilder 实例本身,以便链式调用。 - 重载方法: scale(double scale)、min(double min) 和 max(double scale) 方法是重载方法,用于直接设置数值。这些方法内部调用了对应的函数设置方法,并将数值包装成一个简单的 Lambda 表达式 in -> scale,从而将数值转换为函数。
-
build() 方法: build() 方法用于构建最终的函数。该方法返回一个 Function
对象,该对象内部使用了 scale、min 和 max 三个函数来计算最终的映射值。 - main() 方法: main() 方法展示了如何使用 RangeBuilder 类来构建用于 Stream 的函数。通过链式调用 scale()、min() 和 max() 方法,可以灵活地配置参数。
使用示例
以下代码展示了如何使用 RangeBuilder 类来构建函数,并将其应用于 Stream:
Listpvs = List.of(new PVector(0, 0)); Stream s = pvs.stream().map(new RangeBuilder().scale(.02) .min(0) .max(255) .build()); Stream s2 = pvs.stream().map(new RangeBuilder().scale(new RangeBuilder().scale(.02) .min(.1) .max(.002) .build()) .min(0) .max(255) .build());
在这个例子中,s 和 s2 都是 Stream
总结
使用 Builder 模式可以有效地解决函数参数既可以是数值,也可以是函数的问题。通过将数值包装成 Lambda 表达式,可以统一参数类型,避免代码重复。Builder 模式还提供了更灵活的 API 设计,使得我们可以一步一步地配置参数,从而提高代码的可读性和可维护性。在实际应用中,可以根据具体的需求,灵活地调整 Builder 类的实现方式,以满足不同的场景。









