Как получить арифметические операторы во время выполнения?

Как я могу получить арифметические операторы во время выполнения в Java? Предположим, если у меня есть значения

  • ADD следует добавить номер
  • MUL тогда надо умножить число

Например

  public calculate(int x, String str){
   while(str.equals("some value")){
     If( str.equals("ADD"))
        // it should return me something like x+
     if( str.equals("MUL"))
        it return me something like x*
    }
    if( str.equals("FINAL"))
        it should return me x+x*x

  }

3 ответа

Решение

Что вам нужно, это не метапрограммирование во время выполнения, а функции первого класса.

Следующее представляет функции первого класса, с арностью 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.

Добавьте метод, который частично применяется 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

Абстракции, используемые в вышеупомянутом решении, а именно интерфейсы функций первого класса и частичное применение, доступны в этой библиотеке.

public class Calculator {
    public static enum Operation {ADD, MUL, SUB, DIV};
    private int x; // store value from previous operations

    public void calculate(int x, Operation operation) {
        switch(operation) {
        case ADD:
            this.x += x;
            break;
        case MUL:
            this.x *= x;
            break;
        case SUB:
            this.x -= x;
            break;
        case DIV:
            this.x /= x;
            break;
        }
    }

    public int getResult() {
        return this.x;
    }
}

Чтобы использовать его в другом месте вашего кода:

public static void main(String[] args) {
    Calculator c = new Calculator();
    c.calculate(4, Calculator.Operation.ADD);
    // Other operations
    c.getResult(); // get final result
}

Предполагая, что вы пытаетесь просто добавить и умножить xПросто сделайте следующее:

public int calculate(int x, String str) {
    // while(true) is gonna get you into some trouble
    if( str.equals("ADD")) {
        return x + x;
    }
    else if( str.equals("MUL")) {
        return x * x;
    }
    else
        return x; // not sure what you want to do in this case
}
Другие вопросы по тегам