ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

区间工具类

2021-06-11 12:33:55  阅读:295  来源: 互联网

标签:halfInterval String param 区间 formula 工具 numberStr append


工具类:

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%"));
    }
}

标签:halfInterval,String,param,区间,formula,工具,numberStr,append
来源: https://blog.csdn.net/mr_zql/article/details/117812440

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有