24.仿简道云公式函数实战-数学函数-COUNTIF

1. COUNTIF函数

2. 函数用法

COUNTIF(value1, value2, …, criteria)

3. 函数示例

统计满足条件的参数个数,value 代表不同的值,criteria 为判断条件。

COUNTIF([1,2,3,4], ">2"),返回结果为2

COUNTIF([1,2,3,4], "!=2"),返回结果为3

COUNTIF([1,2,3,4], ">=1"),返回结果为4

COUNTIF(["a", "b", "c"], "d"),返回结果为0

4. 代码实战

首先我们在function包下创建math包,在math包下创建CountIfFunction类,代码如下:

package com.ql.util.express.self.combat.function.math;

import com.ql.util.express.Operator;
import com.ql.util.express.self.combat.exception.FormulaException;

import java.util.Arrays;
import java.util.function.Predicate;

/**
 * 类描述: COUNTIF函数
 *
 * @author admin
 * @version 1.0.0
 * @date 2023/11/23 10:40
 */
public class CountIfFunction extends Operator {

    public CountIfFunction(String name) {
        this.name = name;
    }

    @Override
    public Object executeInner(Object[] objects) throws Exception {

        // 参数校验
        if (objects.length == 0) {
            throw new FormulaException("操作数异常");
        }

        Predicate<Object> predicate = null;

        // 截取条件
        String criteria = objects[objects.length-1].toString();

        String op = criteria.replaceAll("[0-9]", "");

        // 值
        Object[] values = copyV(objects);

        if (">".equals(op)) {
            // 提取数字值
            String conditionV = criteria.replaceAll("[^0-9]", "");
            predicate = object -> {

                double param = Double.parseDouble(object.toString());
                double p = Double.parseDouble(conditionV);

                return param > p;
            };

        } else if ("<".equals(op)) {// >=
            // 提取数字值
            String conditionV = criteria.replaceAll("[^0-9]", "");
            predicate = object -> {
                double param = Double.parseDouble(object.toString());
                double p = Double.parseDouble(conditionV);
                return param < p;
            };
        } else if ("<=".equals(op)) {// >=
            // 提取数字值
            String conditionV = criteria.replaceAll("[^0-9]", "");
            predicate = object -> {
                double param = Double.parseDouble(object.toString());
                double p = Double.parseDouble(conditionV);
                return param <= p;
            };
        } else if (">=".equals(op)) {
            // 提取数字值
            String conditionV = criteria.replaceAll("[^0-9]", "");
            predicate = object -> {

                double param = Double.parseDouble(object.toString());
                double p = Double.parseDouble(conditionV);

                return param >= p;
            };

        } else if ("!=".equals(op)) {
            // 提取数字值
            String conditionV = criteria.replaceAll("[^0-9]", "");
            predicate = object -> {

                double param = Double.parseDouble(object.toString());
                double p = Double.parseDouble(conditionV);

                return param != p;
            };

        } else if ("==".equals(op)) {
            // 提取数字值
            String conditionV = criteria.replaceAll("[^0-9]", "");
            predicate = object -> {

                double param = Double.parseDouble(object.toString());
                double p = Double.parseDouble(conditionV);

                return param == p;
            };
        } else {
            predicate = object -> object.toString().equals(criteria);
        }
        final long count = Arrays.stream(values).filter(predicate).count();
        return count;
    }

    public Object[] copyV(Object[] src) {

        Object[] rst = new Object[src.length-1];

        for (int i=0;i<rst.length;i++) {
            rst[i] = src[i];
        }

        return rst;

    }

}

把CountIfFunction类注册到公式函数入口类中,代码如下:

package com.ql.util.express.self.combat.ext;

import com.ql.util.express.ExpressRunner;
import com.ql.util.express.IExpressResourceLoader;
import com.ql.util.express.parse.NodeTypeManager;
import com.ql.util.express.self.combat.function.logic.*;
import com.ql.util.express.self.combat.function.math.*;

/**
 * 类描述: 仿简道云公式函数实战入口类
 *
 * @author admin
 * @version 1.0.0
 * @date 2023/11/21 15:29
 */
public class FormulaRunner extends ExpressRunner {

    public FormulaRunner() {
        super();
    }

    public FormulaRunner(boolean isPrecise, boolean isTrace) {
        super(isPrecise,isTrace);
    }

    public FormulaRunner(boolean isPrecise, boolean isStrace, NodeTypeManager nodeTypeManager) {
        super(isPrecise,isStrace,nodeTypeManager);
    }

    public FormulaRunner(boolean isPrecise, boolean isTrace, IExpressResourceLoader iExpressResourceLoader, NodeTypeManager nodeTypeManager) {
        super(isPrecise,isTrace,iExpressResourceLoader,nodeTypeManager);
    }

    @Override
    public void addSystemFunctions() {
        // ExpressRunner 的内部系统函数
        super.addSystemFunctions();
        // 扩展公式函数
        this.customFunction();
    }
    /***
     * 自定义公式函数
     */
    public void customFunction() {

        // 逻辑公式函数
        this.addLogicFunction();

        // 数学公式函数
        this.addMathFunction();
    }

    public void addLogicFunction() {
        // AND函数
        this.addFunction("AND",new AndFunction("AND"));

        // IF函数
        this.addFunction("IF",new IfFunction("IF"));

        // IFS函数
        this.addFunction("IFS",new IfsFunction("IFS"));

        // XOR函数
        this.addFunction("XOR",new XorFunction("XOR"));

        // TRUE函数
        this.addFunction("TRUE",new TrueFunction("TRUE"));

        // FALSE函数
        this.addFunction("FALSE",new FalseFunction("FALSE"));

        // NOT函数
        this.addFunction("NOT",new NotFunction("NOT"));

        // OR函数
        this.addFunction("OR",new OrFunction("OR"));
    }

    public void addMathFunction() {
        // ABS函数
        this.addFunction("ABS",new AbsFunction("ABS"));

        // AVERAGE函数
        this.addFunction("AVERAGE",new AvgFunction("AVERAGE"));

        // CEILING函数
        this.addFunction("CEILING",new CeilingFunction("CEILING"));

        // RADIANS函数
        this.addFunction("RADIANS",new RadiansFunction("RADIANS"));

        // COS函数
        this.addFunction("COS",new CosFunction("COS"));

        // COT函数
        this.addFunction("COT",new CotFunction("COT"));

        // COUNT函数
        this.addFunction("COUNT",new CountFunction("COUNT"));

        // COUNTIF函数
        this.addFunction("COUNTIF",new CountIfFunction("COUNTIF"));

    }
}

创建测试用例

package com.ql.util.express.self.combat;

import com.ql.util.express.DefaultContext;
import com.ql.util.express.self.combat.ext.FormulaRunner;
import org.junit.Test;

/**
 * 类描述: 实战测试类
 *
 * @author admin
 * @version 1.0.0
 * @date 2023/11/21 15:45
 */
public class CombatTest {

    @Test
    public void COUNTIF() throws Exception{

        FormulaRunner formulaRunner = new FormulaRunner(true,true);
        // 创建上下文
        DefaultContext<String, Object> context = new DefaultContext<>();
        String express = "COUNTIF(1,2,3,4,5,'!=2')";
        Object object = formulaRunner.execute(express, context, null, true, true);
        System.out.println(object);

    }

}

运行结果

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值