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

org.nfunk.jep.function.Comparative Maven / Gradle / Ivy

Go to download

JEP is a Java library for parsing and evaluating mathematical expressions. Use groupId org.fudaa to deploy it in maven central

The newest version!
/*****************************************************************************

 JEP 2.4.1, Extensions 1.1.1
      April 30 2007
      (c) Copyright 2007, Nathan Funk and Richard Morris
      See LICENSE-*.txt for license information.

*****************************************************************************/

package org.nfunk.jep.function;

import java.util.*;
import org.nfunk.jep.*;
import org.nfunk.jep.type.*;

/** 
 * Implements the comparative operations
 * Caverts should work where arguments are Double, Complex or String
 * for the last two only != and == work.
 * For other types care might be needed.
 * 
 * Complex numbers are compared using a tolerance which can be set
 * using setTolerance().  
 * 
 * @author N Funk and R Morris
 * @since 2.3.0 beta 1 a bit of a rewrite to make sub classing easier, now allows Complex to be compared to Double i.e. 1+0 i == 1.
 * @since 2.3.0 beta 2 changed the internal lt,gt,le,ge,ne and eq method to return boolean.
 * If this breaks anything use
 * 		if(lt(obj1,obj2)) inStack.push(new Double(1));
 *		else 	inStack.push(new Double(0));
 */
public class Comparative extends PostfixMathCommand
{
	protected int id;
	double tolerance;
	public static final int LT = 0;
	public static final int GT = 1;
	public static final int LE = 2;
	public static final int GE = 3;
	public static final int NE = 4;
	public static final int EQ = 5;

	/**
	 * Constructor. Sets the number of parameters to 2. Initializes the
	 * tolerance for comparing Complex values.
	 * @param id_in The id of the comparative operator.
	 */
	public Comparative(int id_in)
	{
		id = id_in;
		numberOfParameters = 2;
		tolerance = 1e-6;
	}
	
	public void run(Stack inStack)	throws ParseException 
	{
		checkStack(inStack);// check the stack
		
		Object param2 = inStack.pop();
		Object param1 = inStack.pop();
		boolean res=false;
		switch(id)
		{
		case LT: res = lt(param1,param2); break;
		case GT: res = gt(param1,param2); break;
		case LE: res = le(param1,param2); break;
		case GE: res = ge(param1,param2); break;
		case NE: res = ne(param1,param2); break;
		case EQ: res = eq(param1,param2); break;
		}
		if(res) inStack.push(new Double(1));
		else inStack.push(new Double(0));
	}
	
	public boolean lt(Object param1, Object param2)	throws ParseException
	{
		if ((param1 instanceof Complex) || (param2 instanceof Complex))
			throw new ParseException("< not defined for complex numbers");
		if ((param1 instanceof Number) && (param2 instanceof Number))
		{
			double x = ((Number)param1).doubleValue();
			double y = ((Number)param2).doubleValue();
			return (x not defined for complex numbers");
		
		if ((param1 instanceof Number) && (param2 instanceof Number))
		{
			double x = ((Number)param1).doubleValue();
			double y = ((Number)param2).doubleValue();
			return (x>y);
		}
		throw new ParseException("> not defined for object of type "+param1.getClass().getName()+" and "+param2.getClass().getName());
	}
	
	public boolean le(Object param1, Object param2)	throws ParseException
	{
		if ((param1 instanceof Complex) || (param2 instanceof Complex))
			throw new ParseException("<= not defined for complex numbers");
		
		if ((param1 instanceof Number) && (param2 instanceof Number))
		{
			double x = ((Number)param1).doubleValue();
			double y = ((Number)param2).doubleValue();
			return (x<=y);
		}
		throw new ParseException("<= not defined for object of type "+param1.getClass().getName()+" and "+param2.getClass().getName());
	}
	
	public boolean ge(Object param1, Object param2)	throws ParseException
	{
		if ((param1 instanceof Complex) || (param2 instanceof Complex))
			throw new ParseException(">= not defined for complex numbers");
		
		if ((param1 instanceof Number) && (param2 instanceof Number))
		{
			double x = ((Number)param1).doubleValue();
			double y = ((Number)param2).doubleValue();
			return (x>=y);
		}
		throw new ParseException(">= not defined for object of type "+param1.getClass().getName()+" and "+param2.getClass().getName());
	}

	public boolean eq(Object param1, Object param2)	throws ParseException
	{
		if ((param1 instanceof Complex) && (param2 instanceof Complex))
		{
			return ((Complex)param1).equals((Complex)param2, tolerance);
		}
		if ((param1 instanceof Complex) && (param2 instanceof Number))
		{
			return ((Complex)param1).equals(new Complex((Number)param2), tolerance);
		}
		if ((param2 instanceof Complex) && (param1 instanceof Number))
		{
			return ((Complex)param2).equals(new Complex((Number)param1), tolerance);
		}
		if ((param1 instanceof Number) && (param2 instanceof Number))
		{
			double x = ((Number)param1).doubleValue();
			double y = ((Number)param2).doubleValue();
			return (x==y);
		}
		if ((param1 instanceof Boolean) && (param2 instanceof Boolean))
		{
			boolean x = ((Boolean)param1).booleanValue();
			boolean y = ((Boolean)param2).booleanValue();
			return (x==y);
		}
		if ((param1 instanceof Number) && (param2 instanceof Boolean))
		{
			double x = ((Number)param1).doubleValue();
			double y = ((Boolean)param2).booleanValue()?1.0:0.0;
			return (x==y);
		}
		if ((param1 instanceof Boolean) && (param2 instanceof Number))
		{
			double x = ((Boolean)param1).booleanValue()?1.0:0.0;
			double y = ((Number)param2).doubleValue();
			return (x==y);
		}
		
		// if we get to here, just use the equal function
		return param1.equals(param2);
	}
	
	public boolean ne(Object param1, Object param2)	throws ParseException
	{
		if ((param1 instanceof Complex) && (param2 instanceof Complex))
		{
			return !((Complex)param1).equals((Complex)param2, tolerance);
		}
		if ((param1 instanceof Complex) && (param2 instanceof Number))
		{
			return !((Complex)param1).equals(new Complex((Number) param2), tolerance);
		}
		if ((param2 instanceof Complex) && (param1 instanceof Number))
		{
			return !((Complex)param2).equals(new Complex((Number) param1), tolerance);
		}
		if ((param1 instanceof Number) && (param2 instanceof Number))
		{
			double x = ((Number)param1).doubleValue();
			double y = ((Number)param2).doubleValue();
			return (x!=y);
		}
		if ((param1 instanceof Boolean) && (param2 instanceof Boolean))
		{
			boolean x = ((Boolean)param1).booleanValue();
			boolean y = ((Boolean)param2).booleanValue();
			return (x!=y);
		}
		if ((param1 instanceof Number) && (param2 instanceof Boolean))
		{
			double x = ((Number)param1).doubleValue();
			double y = ((Boolean)param2).booleanValue()?1.0:0.0;
			return (x!=y);
		}
		if ((param1 instanceof Boolean) && (param2 instanceof Number))
		{
			double x = ((Boolean)param1).booleanValue()?1.0:0.0;
			double y = ((Number)param2).doubleValue();
			return (x!=y);
		}
		return !param1.equals(param2);
	}


/* old code
		if ((param1 instanceof Complex) || (param2 instanceof Complex))
			throw new ParseException(">= not defined for complex numbers");
		if ((param1 instanceof Number) && (param2 instanceof Number))
		{
			double x = ((Number)param1).doubleValue();
			double y = ((Number)param2).doubleValue();
			int r = (x>=y) ? 1 : 0;
			return new Double(r);
		}
		throw new ParseException(">= not defined for object of type "+param1.getClass().getName()+" and "+param2.getClass().getName());
	}
	
		{
			int r;
			
			switch (id)
			{
				case NE:
					r = ((Complex)param1).equals((Complex)param2, tolerance) ? 0 : 1;
					break;
				case EQ:
					r = ((Complex)param1).equals((Complex)param2, tolerance) ? 1 : 0;
					break;
				default:
					throw new ParseException("Relational operator type error");
			}
			
			inStack.push(new Double(r));//push the result on the inStack
		}
		else if ((param1 instanceof Number) && (param2 instanceof Number))
		{
			double x = ((Number)param1).doubleValue();
			double y = ((Number)param2).doubleValue();
			int r;
			
			switch (id)
			{
				case LT:
					r = (xy) ? 1 : 0;
					break;
				case LE:
					r = (x<=y) ? 1 : 0;
					break;
				case GE:
					r = (x>=y) ? 1 : 0;
					break;
				case NE:
					r = (x!=y) ? 1 : 0;
					break;
				case EQ:
					r = (x==y) ? 1 : 0;
					break;
				default:
					throw new ParseException("Unknown relational operator");
			}
			
			inStack.push(new Double(r));//push the result on the inStack
		} 
		else if ((param1 instanceof String) && (param2 instanceof String))
		{
			int r;
			
			switch (id)
			{
				case NE:
					r = ((String)param1).equals((String)param2) ? 0 : 1;
					break;
				case EQ:
					r = ((String)param1).equals((String)param2) ? 1 : 0;
					break;
				default:
					throw new ParseException("Relational operator type error");
			}
			
			inStack.push(new Double(r));//push the result on the inStack
		} else
		{
			throw new ParseException("Invalid parameter type");
		}
		
		
		return;
	}
	*/
	
	/**
	 * Returns the tolerance used for comparing complex numbers
	 */
	public double getTolerance() {
		return tolerance;
	}

	/**
	 * Sets the tolerance used for comparing complex numbers
	 * @param d
	 */
	public void setTolerance(double d) {
		tolerance = d;
	}

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy