All Downloads are FREE. Search and download functionalities are using the official Maven repository.

weka.core.expressionlanguage.common.MathFunctions Maven / Gradle / Ivy

/*
 *   This program is free software: you can redistribute it and/or modify
 *   it under the terms of the GNU General Public License as published by
 *   the Free Software Foundation, either version 3 of the License, or
 *   (at your option) any later version.
 *
 *   This program is distributed in the hope that it will be useful,
 *   but WITHOUT ANY WARRANTY; without even the implied warranty of
 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *   GNU General Public License for more details.
 *
 *   You should have received a copy of the GNU General Public License
 *   along with this program.  If not, see .
 */

/*
 *    MathFunctions.java
 *    Copyright (C) 2015 University of Waikato, Hamilton, New Zealand
 *
 */

package weka.core.expressionlanguage.common;

import java.util.Map;
import java.util.HashMap;

import weka.core.expressionlanguage.core.Node;
import weka.core.expressionlanguage.core.Macro;
import weka.core.expressionlanguage.core.MacroDeclarations;
import weka.core.expressionlanguage.core.SemanticException;
import weka.core.expressionlanguage.common.Primitives.DoubleExpression;

/**
 * Macro declarations for common mathematical functions.

* * The following functions are being exposed through macros:
*
    *
  • {@link java.lang.Math.abs(double)} as abs
  • *
  • {@link java.lang.Math.sqrt(double)} as sqrt
  • *
  • {@link java.lang.Math.log(double)} as log
  • *
  • {@link java.lang.Math.exp(double)} as exp
  • *
  • {@link java.lang.Math.sin(double)} as sin
  • *
  • {@link java.lang.Math.cos(double)} as cos
  • *
  • {@link java.lang.Math.tan(double)} as tan
  • *
  • {@link java.lang.Math.rint(double)} as rint
  • *
  • {@link java.lang.Math.floor(double)} as floor
  • *
  • {@link java.lang.Math.ceil(double)} as ceil
  • *
  • {@link java.lang.Math.pow(double)} as pow
  • *
* * @author Benjamin Weber ( benweber at student dot ethz dot ch ) * @version $Revision: 1000 $ */ public class MathFunctions implements MacroDeclarations { /** the macros to be exposed */ private static Map macros = new HashMap(); static { macros.put("abs", new DoubleUnaryMacro(AbsFunction.class)); macros.put("sqrt", new DoubleUnaryMacro(SqrtFunction.class)); macros.put("log", new DoubleUnaryMacro(LogFunction.class)); macros.put("exp", new DoubleUnaryMacro(ExpFunction.class)); macros.put("sin", new DoubleUnaryMacro(SinFunction.class)); macros.put("cos", new DoubleUnaryMacro(CosFunction.class)); macros.put("tan", new DoubleUnaryMacro(TanFunction.class)); macros.put("rint", new DoubleUnaryMacro(RintFunction.class)); macros.put("floor", new DoubleUnaryMacro(FloorFunction.class)); macros.put("ceil", new DoubleUnaryMacro(CeilFunction.class)); macros.put("pow", new PowMacro()); } /** * Whether the macro is declared * * @param name of the macro * @return whether the macro is declared */ @Override public boolean hasMacro(String name) { return macros.containsKey(name); } /** * Tries to fetch the macro

* * The same invariant of {@link MacroDeclarations} applies here too. * * @param name name of the macro * @return a macro */ @Override public Macro getMacro(String name) { if (macros.containsKey(name)) return macros.get(name); throw new RuntimeException("Macro '" + name + "' undefined!"); } private static class DoubleUnaryMacro implements Macro { private final Class func; public DoubleUnaryMacro(Class func) { this.func = func; } private String name() { return func.getSimpleName(); } @Override public Node evaluate(Node... params) throws SemanticException { if (params.length != 1) throw new SemanticException("'" + name() + "' takes exactly one argument!"); if (!(params[0] instanceof DoubleExpression)) throw new SemanticException("'" + name() + "'s first argument must be double!"); try { Node node =(Node) func.getConstructor(DoubleExpression.class).newInstance(params[0]); return node; } catch (Exception e) { throw new RuntimeException("Failed to instantiate '" + name() + "'!", e); } } } private static abstract class DoubleUnaryFunction implements DoubleExpression { final DoubleExpression expr; DoubleUnaryFunction(DoubleExpression expr) { this.expr = expr; } } private static class AbsFunction extends DoubleUnaryFunction { public AbsFunction(DoubleExpression expr) { super(expr); } @Override public double evaluate() { return Math.abs(expr.evaluate()); } } private static class SqrtFunction extends DoubleUnaryFunction { public SqrtFunction(DoubleExpression expr) { super(expr); } @Override public double evaluate() { return Math.sqrt(expr.evaluate()); } } private static class LogFunction extends DoubleUnaryFunction { public LogFunction(DoubleExpression expr) { super(expr); } @Override public double evaluate() { return Math.log(expr.evaluate()); } } private static class ExpFunction extends DoubleUnaryFunction { public ExpFunction(DoubleExpression expr) { super(expr); } @Override public double evaluate() { return Math.exp(expr.evaluate()); } } private static class SinFunction extends DoubleUnaryFunction { public SinFunction(DoubleExpression expr) { super(expr); } @Override public double evaluate() { return Math.sin(expr.evaluate()); } } private static class CosFunction extends DoubleUnaryFunction { public CosFunction(DoubleExpression expr) { super(expr); } @Override public double evaluate() { return Math.cos(expr.evaluate()); } } private static class TanFunction extends DoubleUnaryFunction { public TanFunction(DoubleExpression expr) { super(expr); } @Override public double evaluate() { return Math.tan(expr.evaluate()); } } private static class RintFunction extends DoubleUnaryFunction { public RintFunction(DoubleExpression expr) { super(expr); } @Override public double evaluate() { return Math.rint(expr.evaluate()); } } private static class FloorFunction extends DoubleUnaryFunction { public FloorFunction(DoubleExpression expr) { super(expr); } @Override public double evaluate() { return Math.floor(expr.evaluate()); } } private static class CeilFunction extends DoubleUnaryFunction { public CeilFunction(DoubleExpression expr) { super(expr); } @Override public double evaluate() { return Math.ceil(expr.evaluate()); } } private static class PowMacro implements Macro { @Override public Node evaluate(Node... params) throws SemanticException { if (params.length != 2) throw new SemanticException("pow takes exactly two arguments!"); if (!(params[0] instanceof DoubleExpression)) throw new SemanticException("pow's first argument must be double!"); if (!(params[1] instanceof DoubleExpression)) throw new SemanticException("pow's second argument must be double!"); return new PowFunction((DoubleExpression) params[0], (DoubleExpression) params[1]); } } private static class PowFunction implements DoubleExpression { private final DoubleExpression base; private final DoubleExpression exponent; public PowFunction(DoubleExpression base, DoubleExpression exponent) { this.base = base; this.exponent = exponent; } @Override public double evaluate() { return Math.pow(base.evaluate(), exponent.evaluate()); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy