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

com.scudata.expression.fn.Between Maven / Gradle / Ivy

Go to download

SPL(Structured Process Language) A programming language specially for structured data computing.

The newest version!
package com.scudata.expression.fn;

import com.scudata.array.ArrayUtil;
import com.scudata.array.BoolArray;
import com.scudata.array.ConstArray;
import com.scudata.array.IArray;
import com.scudata.array.IntArray;
import com.scudata.common.MessageManager;
import com.scudata.common.ObjectCache;
import com.scudata.common.RQException;
import com.scudata.dm.Context;
import com.scudata.expression.Expression;
import com.scudata.expression.Function;
import com.scudata.expression.IParam;
import com.scudata.expression.Relation;
import com.scudata.resources.EngineMessage;
import com.scudata.util.Variant;

/**
 * ?ж?ֵ?Ƿ??ڸ?????????
 * between(x,a:b)
 * @author RunQian
 *
 */
public class Between extends Function {
	private Expression exp;
	private Expression startExp;
	private Expression endExp;
	
	/**
	 * ??????ʽ????Ч?ԣ???Ч???׳??쳣
	 */
	public void checkValidity() {
		if (param == null) {
			MessageManager mm = EngineMessage.get();
			throw new RQException("between" + mm.getMessage("function.missingParam"));
		} else if (param.getSubSize() != 2) {
			MessageManager mm = EngineMessage.get();
			throw new RQException("between" + mm.getMessage("function.invalidParam"));
		}
		
		IParam sub0 = param.getSub(0);
		IParam sub1 = param.getSub(1);
		if (sub0 == null || sub1 == null || sub1.getSubSize() != 2) {
			MessageManager mm = EngineMessage.get();
			throw new RQException("between" + mm.getMessage("function.invalidParam"));
		}
		
		exp = sub0.getLeafExpression();
		IParam startParam = sub1.getSub(0);
		IParam endParam = sub1.getSub(1);
		if (startParam == null || endParam == null) {
			MessageManager mm = EngineMessage.get();
			throw new RQException("between" + mm.getMessage("function.invalidParam"));
		}
		
		startExp = startParam.getLeafExpression();
		endExp = endParam.getLeafExpression();
	}
	
	public Object calculate(Context ctx) {
		Object val = exp.calculate(ctx);
		Object startVal = startExp.calculate(ctx);
		int cmp = Variant.compare(val, startVal, true);
		
		if (option == null || option.indexOf('b') == -1) {
			if (cmp > 0) {
				Object endVal = endExp.calculate(ctx);
				cmp = Variant.compare(val, endVal, true);
				if (cmp < 0) {
					return Boolean.TRUE;
				} else if (cmp == 0) {
					return Boolean.valueOf(option == null || option.indexOf('r') == -1);
				} else {
					return Boolean.FALSE;
				}
			} else if (cmp == 0) {
				return Boolean.valueOf(option == null || option.indexOf('l') == -1);
			} else {
				return Boolean.FALSE;
			}
		} else {
			if (cmp > 0) {
				Object endVal = endExp.calculate(ctx);
				cmp = Variant.compare(val, endVal, true);
				if (cmp < 0) {
					return ObjectCache.getInteger(0);
				} else if (cmp == 0) {
					if (option.indexOf('r') == -1) {
						return ObjectCache.getInteger(0);
					} else {
						return ObjectCache.getInteger(1);
					}
				} else {
					return ObjectCache.getInteger(1);
				}
			} else if (cmp == 0) {
				if (option.indexOf('l') == -1) {
					return ObjectCache.getInteger(0);
				} else {
					return ObjectCache.getInteger(-1);
				}
			} else {
				return ObjectCache.getInteger(-1);
			}
		}
	}
	

	/**
	 * ??????????еĽ??
	 * @param ctx ??????????
	 * @return IArray
	 */
	public IArray calculateAll(Context ctx) {
		boolean isBinary = false;
		int leftRelation = Relation.GREATER_EQUAL;
		int rightRelation = Relation.LESS_EQUAL;
		
		if (option != null) {
			if (option.indexOf('b') != -1) isBinary = true;
			if (option.indexOf('l') != -1) leftRelation = Relation.GREATER;
			if (option.indexOf('r') != -1) rightRelation = Relation.LESS;
		}
		
		IArray array = exp.calculateAll(ctx);
		IArray startArray = startExp.calculateAll(ctx);
		
		// ????ֵ??????бȽ?
		BoolArray result = array.calcRelation(startArray, leftRelation);
		IArray endArray = endExp.calculateAll(ctx, result, true);
		
		if (isBinary) {
			int size = result.size();
			int []values = new int[size + 1];
			boolean []signs = result.getDatas();
			for (int i = 1; i <= size; ++i) {
				values[i] = signs[i] ? 0 : -1;
			}
			
			array.calcRelations(endArray, rightRelation, result, true);
			for (int i = 1; i <= size; ++i) {
				if (values[i] == 0 && !signs[i]) {
					values[i] = 1;
				}
			}
			
			return new IntArray(values, null, size);
		} else {
			array.calcRelations(endArray, rightRelation, result, true);
			return result;
		}
	}

	/**
	 * ????signArray??ȡֵΪsign????
	 * @param ctx
	 * @param signArray ?б?ʶ????
	 * @param sign ??ʶ
	 * @return IArray
	 */
	public IArray calculateAll(Context ctx, IArray signArray, boolean sign) {
		boolean isBinary = false;
		int leftRelation = Relation.GREATER_EQUAL;
		int rightRelation = Relation.LESS_EQUAL;
		
		if (option != null) {
			if (option.indexOf('b') != -1) isBinary = true;
			if (option.indexOf('l') != -1) leftRelation = Relation.GREATER;
			if (option.indexOf('r') != -1) rightRelation = Relation.LESS;
		}
		
		BoolArray result = ArrayUtil.booleanValue(signArray, sign);
		IArray array = exp.calculateAll(ctx, result, true);
		IArray startArray = startExp.calculateAll(ctx, result, true);
		
		// ????ֵ??????бȽ?
		array.calcRelations(startArray, leftRelation, result, true);
		IArray endArray = endExp.calculateAll(ctx, result, true);
		
		if (isBinary) {
			int size = result.size();
			int []values = new int[size + 1];
			boolean []signs = result.getDatas();
			for (int i = 1; i <= size; ++i) {
				values[i] = signs[i] ? 0 : -1;
			}
			
			array.calcRelations(endArray, rightRelation, result, true);
			for (int i = 1; i <= size; ++i) {
				if (values[i] == 0 && !signs[i]) {
					values[i] = 1;
				}
			}
			
			return new IntArray(values, null, size);
		} else {
			array.calcRelations(endArray, rightRelation, result, true);
			return result;
		}
	}
	
	/**
	 * ?????߼????????&&???Ҳ????ʽ
	 * @param ctx ??????????
	 * @param leftResult &&??????ʽ?ļ?????
	 * @return BoolArray
	 */
	public BoolArray calculateAnd(Context ctx, IArray leftResult) {
		int leftRelation = Relation.GREATER_EQUAL;
		int rightRelation = Relation.LESS_EQUAL;
		
		if (option != null) {
			if (option.indexOf('l') != -1) leftRelation = Relation.GREATER;
			if (option.indexOf('r') != -1) rightRelation = Relation.LESS;
		}
		
		BoolArray result = leftResult.isTrue();
		IArray array = exp.calculateAll(ctx, result, true);
		IArray startArray = startExp.calculateAll(ctx, result, true);
		
		// ????ֵ??????бȽ?
		array.calcRelations(startArray, leftRelation, result, true);
		IArray endArray = endExp.calculateAll(ctx, result, true);
		array.calcRelations(endArray, rightRelation, result, true);
		return result;
	}
	
	/**
	 * ?жϸ?????ֵ??Χ?Ƿ????㵱ǰ????????ʽ
	 * @param ctx ??????????
	 * @return ȡֵ????Relation. -1??ֵ??Χ??û????????????ֵ??0??ֵ??Χ??????????????ֵ??1??ֵ??Χ??ֵ??????????
	 */
	public int isValueRangeMatch(Context ctx) {
		IArray array = exp.calculateRange(ctx);
		if (array == null) {
			return Relation.PARTICALMATCH;
		}
		
		IArray startArray = startExp.calculateRange(ctx);
		if (!(startArray instanceof ConstArray)) {
			return Relation.PARTICALMATCH;
		}
		
		IArray endArray = endExp.calculateRange(ctx);
		if (!(endArray instanceof ConstArray)) {
			return Relation.PARTICALMATCH;
		}
		
		boolean containLeft = true, containRight = true;
		if (option != null) {
			if (option.indexOf('l') != -1) containLeft = false;
			if (option.indexOf('r') != -1) containRight = false;
		}
		
		Object startVal = startArray.get(1);
		Object endVal = endArray.get(1);
		Object minVal = array.get(1);
		int cmp = Variant.compare(minVal, startVal, true);
		
		if (cmp > 0 || (cmp == 0 && containLeft)) {
			cmp = Variant.compare(minVal, endVal, true);
			if (cmp < 0 || (cmp == 0 && containRight)) {
				Object maxValue = array.get(2);
				cmp = Variant.compare(maxValue, endVal, true);
				if (cmp < 0 || (cmp == 0 && containRight)) {
					return Relation.ALLMATCH;
				} else {
					return Relation.PARTICALMATCH;
				}
			} else {
				return Relation.UNMATCH;
			}
		} else {
			Object maxValue = array.get(2);
			cmp = Variant.compare(maxValue, startVal, true);
			if (cmp > 0 || (cmp == 0 && containLeft)) {
				return Relation.PARTICALMATCH;
			} else {
				return Relation.UNMATCH;
			}
		}
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy