java通过正则进行语法分析实现表达式的逻辑判断和复杂计算实现

首先功能展示:

  

相关功能实现,实现功能类似js,弱语言类型:

  1、核销语法解析使用正则校验和匹配实现处理,每一行是一个完整表达式

  2、有系统变量使用,内置的变量可直接获取值进行相关逻辑和条件计算

  3、有内容函数可供实现,具体函数可看枚举FuncEnum的定义和说明,如要扩展跟多函数,可再functionCalculateFactory中实现逻辑

  4、可以通过如果...则,否则如果...则,否则,如果完完成逻辑控制

  5、可以通过加(+)、减(-)、乘(*)、除(/)、求余(%)实现表达式计算

  6、采用变量池存储所有自定义变量和产生的临时变量

  /**
     * 变量池,用于存储当前自定义变量和数据
     */
    protected HashMap<String, VariateDto> VARIABLE_POOL = new HashMap<>();

  7、执行过程会打印所有执行的详细步骤

  8、前端编写语法高亮使用codemirror的插件改写,具体文件可看github里,使用方式如下:

// 1、定义
<textarea ref="formula-mirror-code" name="code"></textarea>
// 2、引用控件
require('../../components/qd-project/codemirror/codemirror.css');
import CodeMirror from '../../components/qd-project/codemirror/codemirror.js';
// 使用
this.$nextTick(function () {        
    var te = this.$refs["formula-mirror-code"]; // document.getElementById("formula-mirror-code");
    this.editor = CodeMirror.fromTextArea(te, {
        mode: "lu-code",
        lineNumbers: true,
        lineWrapping: false, // 自动换行
        //extraKeys: {"Ctrl-Q": function(cm){ cm.foldCode(cm.getCursor()); }},
        foldGutter: true,
        gutters: ["CodeMirror-linenumbers", "CodeMirror-foldgutter"]
    });
    //editor.foldCode(CodeMirror.Pos(13, 0));
    this.editor.setValue(this.content || '');
    this.editor.setSize('100%', '400px')
    this.editor.on('change', ()=>{            
        this.$emit('input', this.editor.getValue());
        this.$emit('change', this.editor.getValue());                    
        // 直接修改pop里的属性会被视作反模式操作,会出现警告
        // 由于在新的渲染机制中,每当父组件重新渲染时,子组件都会被覆盖,所以应该把props看做是不可变对象
        // 应该这个组件提交个事件给父组件
        //this.value = this.editor.getValue();
    })
    if (this.editor.getValue().length <= 0) {                    
        this.editor.setValue("数值 分摊比例\n"
                    + "数值 分摊单价\n"
                    + "\n"
                    + "分摊单价 = 公区表_单价\n"
                    + "// 条件判断户数大于10\n"
                    + "如果 分摊户数<=10 则\n"
                    + "    分摊用量 =  公区表_合计用量/分摊户数*0.8\n"
                    + "否则如果 分摊户数>10 && 分摊户数<20 则\n"
                    + "    分摊用量 =  公区表_合计用量 /分摊户数\n"
                    + "否则如果 分摊户数>=20 && 分摊户数<=100 则\n"
                    + "    分摊用量 =  公区表_合计用量 /分摊户数*1.2\n"
                    + "否则\n"
                    + "    分摊用量 =  公区表_合计用量 /分摊户数*1.5\n"
                    + "如果完\n"
                    + "分摊用量 = round(分摊用量, 2)\n"
                    + "// 计算金额\n"
                    + "分摊金额 = round(分摊用量* 公区表_单价, 2)\n"
                    + "分摊比例 =  round(公区表_应收金额/分摊金额, 2)");
    }
})

 

 具体功能调用方式:

 static String content = "数值 加压电上次读数\n" +
        "数值 加压电本次读数\n" +
        "数值 加压用电\n" +
        "数值     用水量\n" +
        "用水量=标准值+1\n" +
        "如果 !!(!(max(43.3,99)-2<=66) || 1==2 && min(用水量,4)>3 || (max(44,33) > 10 &&  7>9)) != true 则\n"+
        "用水量=MOD(5,2)\n" +
        "否则如果 用水量==26 || 1==1 则\n"+
        "   加压用电=MAX(4.0038,4)+2\n" +
        "   用水量=334\n" +
        "否则如果 用水量==28 则\n"+
        "   加压用电=MAX(4.0038,4)+3\n" +
        "   用水量=335\n" +
        "否则如果 1==1 则\n"+
        "   加压用电=MAX(4.0038,4)+36\n" +
        "   用水量=3366\n" +
        "否则\n" +
        "   用水量=QUOTIENT(5,2)\n" +
        "   加压用电=2\n" +
        "如果完\n"+
        "用水量=sqrt(-43.1)\n" +
        "用水量=TROWC(-43.1)\n" +
        "加压用电=ROUND(((加压电本次读数+5-加压电上次读数*(加压电上次读数+加压电本次读数*(MIN(加压电本次读数,0.5)/2)))-(加压电上次读数+3)*2)*3,2)\n" ;
     
// 设置系统变量 HashMap<String, Object> systemVariable = Maps.newHashMap(); systemVariable.put("金额", 0.5); systemVariable.put("编号", "SNG-001-23"); MeterCalculate meterCalculate = new MeterCalculate(content, systemVariable); // 实例化补充系统变量 meterCalculate.putSystemVariable("标准值", 66); // 验证表达式是否正确 meterCalculate.verifyContent(); // 执行编译,可得到相关结果 meterCalculate.compile(); // 可以得到相关变量的值 logger.info(">>>>>{}", JSON.toJSONString(meterCalculate.VARIABLE_POOL()));

Github地址:https://github.com/wanglu05/simple-calculate.git

posted @ 2019-08-22 15:00  三月路  阅读(1789)  评论(0编辑  收藏  举报