Java 8 函数式编程 - 需要在 class 上编写泛型函数

Java 8 Functional Programming - Need to write a generic function on class

我想创建一个接受类似这样的东西的方法

set(nodeStatus, status, NodeStatus::setStatus, Status::valueOf);
set(nodeStatus, errorCode, NodeStatus::setErrorCode, ErrorCode::valueOf);

Status 和 ErrorCode 是 java 中的枚举。

签名和伪代码

set(NodeStatus nodeStatus, byte[] status, ?nodeStatusOperator , ?ValueTransformer) {
   1. convert byte[] status to appropriate value as per ValueTransformer
   2. nodeStatusOperator sets this transformed value according to the lambda passed.
}

我想知道在 java 中应该使用什么方法签名来完成这个以及为什么。我尝试了各种消费者、BiConsumers 等,但无法做到这一点。有人可以帮忙吗?

据我所知,你需要的是:

public <T> void set (NodeStatus nodeStatus, 
                     byte [] value, 
                     BiConsumer<NodeStatus,T> setter, 
                     Function<byte[],T> transformer) {
    T transformedValue = transformer.apply(value);
    setter.accept(nodeStatus, transformedValue);
}

(如果value可以是byte[]以外的其他类型参数,您可以将其替换为其他类型参数。)

P.s.: setter是一个BiConsumer,因为你在实例方法上使用静态方法引用(例如NodeStatus::setErrorCode),所以第一个参数BiConsumer 必须是 NodeStatus 实例 setErrorCode() 将被调用。

P.p.s:正如 glglgl 所指出的,您可以将代码简化为:

public <T> void set (byte [] value, 
                     Consumer<T> setter, 
                     Function<byte[],T> transformer) {
    T transformedValue = transformer.apply(value);
    setter.accept(transformedValue);
}

并这样称呼它:

set(status, nodeStatus::setStatus, Status::valueOf);

...其中 nodeStatus 是您要操作的 NodeStatus 的实例。

有点不清楚您要实现的目标。当您可以只传递一个适用于该特定 NodeStatus 实例的函数时,为什么要传递 NodeStatus 和函数,例如:

static <T> void set(byte[] status, Consumer<T> nodeStatusOperator, Function<String, T> transformer) {
    nodeStatusOperator.accept(transformer.apply(new String(status)));
}

public static void main(String[] args) {
    NodeStatus nodeStatus = new NodeStatus();
    byte[] status = new byte[0];
    set(status, nodeStatus::setStatus, Status::valueOf);
    set(status, nodeStatus::setErrorCode, ErrorCode::valueOf);
}

与更直接的方法相比,这种通用性为您带来了什么?

nodeStatus.setStatus(Status.valueOf(new String(status)));
nodeStatus.setErrorCode(ErrorCode.valueOf(new String(status)));