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)));
我想创建一个接受类似这样的东西的方法
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)));