区间工具类

工具类:

import javax.script.ScriptEngine;
import javax.script.ScriptEngineManager;
import javax.script.ScriptException;
import java.math.BigDecimal;

/**
 * 区间工具类
 *
 * @author zql
 * @version 1.0
 * @createTime 2021-06-11 1:18:00
 * @modifyLog
 */
public class IntervalUtil {

    /**
     * 判断numberStr是否在interval区间范围内
     *
     * @param numberStr 数值类型的字符串
     * @param interval 正常的数学区间,包括无穷大等,如:(0, 100]、<50、(-∞,8]、(10%,50%)U(80%,100%)
     * @return true:表示numberStr在区间interval范围内,false:表示numberStr不在区间interval范围内
     */
    public static boolean isInTheInterval(String numberStr, String interval) throws ScriptException {
        // 将区间和numberStr转化为可计算的表达式
        String formula = getFormulaByAllInterval(numberStr, interval, "||");
        System.out.println(formula);
        ScriptEngine jse = new ScriptEngineManager().getEngineByName("JavaScript");
        try {
            // 计算表达式
            return (Boolean) jse.eval(formula);
        } catch (ScriptException e) {
            System.out.println("计算表达式为:" + formula);
            throw e;
        }
    }

    /**
     * 将所有阀值区间转化为公式:如
     * <pre>
     * [60,100] 转为 numberStr >= 60 && numberStr <= 100
     * (18%,56%)U(86%,100%) 转为 (numberStr > 0.18 && numberStr < 0.56) || (numberStr > 0.86 && numberStr < 1)
     * </pre>
     *
     * @param numberStr 数值类型的字符串
     * @param interval 数学区间
     * @param connector 连接符
     */
    private static String getFormulaByAllInterval(String numberStr, String interval, String connector) {
        StringBuffer sb = new StringBuffer();
        for (String limit : interval.split("U")) {
            sb.append("(");
            String formula = getFormulaByInterval(numberStr, limit, " && ");
            sb.append(formula);
            sb.append(")");
            sb.append(connector);
        }
        String allLimitInvel = sb.toString();
        int index = allLimitInvel.lastIndexOf(connector);
        allLimitInvel = allLimitInvel.substring(0, index);
        return allLimitInvel;
    }

    /**
     * 将整个阀值区间转化为公式:如
     * <pre>
     * [60, 100] 转为 numberStr >= 60 && numberStr <= 100
     * </pre>
     *
     * @param numberStr 数值类型的字符串
     * @param interval 数学区间
     * @param connector 连接符
     */
    private static String getFormulaByInterval(String numberStr, String interval, String connector) {
        StringBuffer sb = new StringBuffer();
        // 如:[60, 100]
        for (String halfInterval : interval.split(",")) {
            // 将半个阀值区间转化为公式
            String formula = getFormulaByHalfInterval(halfInterval, numberStr);
            sb.append(formula).append(connector);
        }
        String limitInvel = sb.toString();
        int index = limitInvel.lastIndexOf(connector);
        limitInvel = limitInvel.substring(0, index);
        return limitInvel;
    }

    /**
     * 将半个阀值区间转化为公式:如
     * <pre>
     * 60) 转为 numberStr < 60
     * ≥88% 转为 numberStr >= 0.88
     * [60 转为 numberStr >= 60
     * <50% 转为 numberStr < 0.5
     * </pre>
     *
     * @param halfInterval 半个阀值区间,如:60)
     * @param numberStr 数值类型的字符串
     * @return
     */
    private static String getFormulaByHalfInterval(String halfInterval, String numberStr) {
        halfInterval = halfInterval.trim();
        // 包含无穷大则不需要公式
        if (halfInterval.contains("∞")) {
            return "1 == 1";
        }
        StringBuffer formula = new StringBuffer();
        String number = "";
        String opera = "";
        // 表示判断方向(如>)在前面 如:≥88%
        String regex = "^([<>≤≥\\[\\(]{1}(-?\\d+.?\\d*\\%?))$";
        if (halfInterval.matches(regex)) {
            opera = halfInterval.substring(0, 1);
            number = halfInterval.substring(1);
        } else {
            opera = halfInterval.substring(halfInterval.length() - 1);
            number = halfInterval.substring(0, halfInterval.length() - 1);
        }
        BigDecimal value = dealPercent(number);
        formula.append(numberStr).append(" ").append(opera).append(" ").append(value);
        String res = formula.toString();
        // 转化特定字符
        res = res.replace("[", ">=");
        res = res.replace("(", ">");
        res = res.replace("]", "<=");
        res = res.replace(")", "<");
        res = res.replace("≤", "<=");
        res = res.replace("≥", ">=");
        return res;
    }

    /**
     * 去除百分号,转为小数
     *
     * @param str 可能含百分号的数字
     * @return
     */
    private static BigDecimal dealPercent(String str) {
        BigDecimal bigDecimal = BigDecimal.ZERO;
        if (str.contains("%")) {
            str = str.substring(0, str.length() - 1);
            bigDecimal = new BigDecimal(str).divide(new BigDecimal(100));
        } else {
            bigDecimal = new BigDecimal(str);
        }
        return bigDecimal;
    }

}

测试类:

import org.junit.Test;

import javax.script.ScriptException;

/**
 * 区间工具测试类
 *
 * @author zql
 * @version 1.0
 * @createTime 2021-06-11 2:24:00
 * @modifyLog
 */
public class IntervalUtilTest {

    @Test
    public void isInTheInterval() throws ScriptException {
        System.out.println(IntervalUtil.isInTheInterval("40", "(0, 100]"));
        System.out.println(IntervalUtil.isInTheInterval("51", "<50"));
        System.out.println(IntervalUtil.isInTheInterval("8", "(-∞,8]"));
        System.out.println(IntervalUtil.isInTheInterval("0.9", "(10%,50%)U(80%,100%)"));
        System.out.println(IntervalUtil.isInTheInterval("88", "[60,100]"));
        System.out.println(IntervalUtil.isInTheInterval("0.79", "≥88%"));
        System.out.println(IntervalUtil.isInTheInterval("0.44", "<50%"));
    }
}
上一篇:力扣中的726题 原子的数量


下一篇:执行数学公式的函数 - 回复 "heyongan" 的问题