必要なのはランタイム メタプログラミングではなく、ファースト クラスの関数です。
以下は、アリティがそれぞれ 1 と 2 のファースト クラス関数を表しています。
abstract class UnaryFunction<A, B> {
public abstract B apply(A a);
}
abstract class BinaryFunction<A, B, C> {
public abstract C apply(A a, B b);
}
簡単にするために、上記のクラスの特殊なバージョンを使用しましょう。
abstract class UnaryOperation {
public abstract int apply(int a);
}
abstract class BinaryOperation {
public abstract int apply(int a, int b);
}
次に、必要な算術演算の辞書を作成します。
Map<String, BinaryOperation> ops = new HashMap<String, BinaryOperation>();
ops.put("ADD", new BinaryOperation() {
public int apply(int a, int b) {
return a + b;
}
});
ops.put("MUL", new BinaryOperation() {
public int apply(int a, int b) {
return a * b;
}
});
// etc.
1 つのパラメーターに部分的に適用されるメソッドを追加しBinaryOperation
ます。
abstract class BinaryOperation {
public abstract int apply(int a, int b);
public UnaryOperation partial(final int a) {
return new UnaryOperation() {
public int apply(int b) {
return BinaryOperation.this.apply(a, b);
}
};
}
}
これで、メソッドを記述できますcalculate
。
public UnaryOperation calculate(int x, String opString) {
BinaryOperation op = ops.get(opString);
if(op == null)
throw new RuntimeException("Operation not found.");
else
return op.partial(x);
}
使用する:
UnaryOperation f = calculate(3, "ADD");
f.apply(5); // returns 8
UnaryOperation g = calculate(9, "MUL");
f.apply(11); // returns 99
上記のソリューションで使用されている抽象化、つまりファースト クラス関数インターフェイスと部分適用は、両方ともこのライブラリで利用できます。