SpringBoot使用Pio-tl动态填写合同(文档)

poi-tl(poi template language)是Word模板引擎,使用Word模板和数据创建很棒的Word文档。 poi-tl官方网址

项目中有需求需要动态填充交易合同,因此想到了使用poi-tl技术来实现

一、引入依赖
 <!-- 生成word并且导出 -->
        <dependency>
            <groupId>com.deepoove</groupId>
            <artifactId>poi-tl</artifactId>
            <version>1.5.0</version>
        </dependency>
        <!--  上面需要的依赖-->
        <dependency>
            <groupId>org.apache.poi</groupId>
            <artifactId>poi-ooxml</artifactId>
            <version>4.1.2</version>
        </dependency>
        <dependency>
            <groupId>org.apache.poi</groupId>
            <artifactId>poi-ooxml-schemas</artifactId>
            <version>4.1.2</version>
        </dependency>
        <dependency>
            <groupId>org.apache.poi</groupId>
            <artifactId>poi</artifactId>
            <version>4.1.2</version>
        </dependency>
        <dependency>
            <groupId>com.deepoove</groupId>
            <artifactId>poi-tl</artifactId>
            <version>1.9.1</version>
        </dependency>
二、填充合同
1、金额转换工具类
public class MoneyUtils {

    private static final String UNIT = "万千佰拾亿千佰拾万千佰拾元角分";
    private static final String DIGIT = "零壹贰叁肆伍陆柒捌玖";
    private static final double MAX_VALUE = 9999999999999.99D;
    public static String change(double v) {
        if (v < 0 || v > MAX_VALUE){
            return "参数非法!";
        }
        long l = Math.round(v * 100);
        if (l == 0){
            return "零元整";
        }
        String strValue = l + "";
        // i用来控制数
        int i = 0;
        // j用来控制单位
        int j = UNIT.length() - strValue.length();
        String rs = "";
        boolean isZero = false;
        for (; i < strValue.length(); i++, j++) {
            char ch = strValue.charAt(i);
            if (ch == '0') {
                isZero = true;
                if (UNIT.charAt(j) == '亿' || UNIT.charAt(j) == '万' || UNIT.charAt(j) == '元') {
                    rs = rs + UNIT.charAt(j);
                    isZero = false;
                }
            } else {
                if (isZero) {
                    rs = rs + "零";
                    isZero = false;
                }
                rs = rs + DIGIT.charAt(ch - '0') + UNIT.charAt(j);
            }
        }
        if (!rs.endsWith("分")) {
            rs = rs + "整";
        }
        rs = rs.replaceAll("亿万", "亿");
        return rs;
    }

    public static void main(String[] args){
        System.out.println(MoneyUtils.change(12356789.9845));
    }

}
2、电子合同生成相关配置及其配置类
# 电子合同生成相关配置
contract:
  #生产环境合同模板存放路径
  prodTemplateUrl: https://xxxxxx-cn-beijing.aliyuncs.com/templets/64042201/20221116/cd9aeb196edd46aeacc607c18d1c481d/xxxxxx活牛交易市场肉牛买卖合同模板.docx
  #生产环境合同临时存放路径(应用服务器路径)
  prodContractPath: /ekode/offlineFile/
  #测试环境合同模板存放路径
  devTemplateUrl: https://xxxxxx-cn-beijing.aliyuncs.com/templets/64042201/20221116/cd9aeb196edd46aeacc607c18d1c481d/xxxxxx活牛交易市场肉牛买卖合同模板.docx
  #测试环境合同临时存放路径(应用服务器路径)
  devContractPath: C://Users//xxxxxx//Desktop//临时文件//
  #合同名后缀
  fileNameSuffix: 肉牛买卖合同
@Component
@ConfigurationProperties(prefix = "contract")
public class ContractConfig {

    private String prodTemplateUrl;
    private String prodContractPath;
    private String devTemplateUrl;
    private String devContractPath;
    private String fileNameSuffix;


    public String getProdTemplateUrl() {
        return prodTemplateUrl;
    }

    public void setProdTemplateUrl(String prodTemplateUrl) {
        this.prodTemplateUrl = prodTemplateUrl;
    }

    public String getProdContractPath() {
        return prodContractPath;
    }

    public void setProdContractPath(String prodContractPath) {
        this.prodContractPath = prodContractPath;
    }

    public String getDevTemplateUrl() {
        return devTemplateUrl;
    }

    public void setDevTemplateUrl(String devTemplateUrl) {
        this.devTemplateUrl = devTemplateUrl;
    }

    public String getDevContractPath() {
        return devContractPath;
    }

    public void setDevContractPath(String devContractPath) {
        this.devContractPath = devContractPath;
    }

    public String getFileNameSuffix() {
        return fileNameSuffix;
    }

    public void setFileNameSuffix(String fileNameSuffix) {
        this.fileNameSuffix = fileNameSuffix;
    }
}

3、word生成工具类
public class WordUtils {

     private static Logger logger = LoggerFactory.getLogger(WordUtils.class);

    /**
     * @param templatePath 文档模板存放路径
     * @param createFileDir 生成的文档存档地址
     * @param createFileName 生成的文件名(不带格式后缀)
     * @param paramsMap   需要填充的内容
     * @return
     */
    public static String createWord(String templatePath, String createFileDir, String createFileName, Map<String, Object> paramsMap) throws IOException {
        logger.info("【动态填充文档公共方法】被操作文档:{},填充后新文档存放地址:{},填充后新文档名称:{}," +
                "填充字段内容:{}", JSON.toJSONString(templatePath),JSON.toJSONString(createFileDir),
                JSON.toJSONString(createFileName),JSON.toJSONString(paramsMap));
        Assert.notNull(templatePath,"文档模板存放路径不能为空");
        Assert.notNull(createFileDir,"生成的文档存档地址不能为空");
        Assert.notNull(createFileName,"生成的文件名不能为空");
        Assert.notNull(paramsMap,"需要填充的内容不能为空");
        //生成的文件格式
        String formatSuffix = ".docx";
        //拼接后的文件名
        String fileName = createFileName + formatSuffix;
        logger.info("生成合同的文件名(带后缀)为:{}",JSON.toJSONString(fileName));
        //生成文件存放路径
        if(!createFileDir.endsWith("/")){
            createFileDir = createFileDir + File.separator;
        }
        File dir = new File(createFileDir);
        if(!dir.exists()){
            dir.mkdirs();
            logger.info("生成合同时存储文件的路径{}不存在,已自动创建文件夹",createFileDir);
        }
        //生成的文件全路径
        String filePath = createFileDir + fileName;
        logger.info("生成合同的存放路径(绝对路径):{}",JSON.toJSONString(filePath));
        XWPFTemplate template = null;
        //poi-tl使用网络文件作为合同模板,需要转换为文件流动态填写
        if(templatePath.contains("https://xxxxxx-cn-beijing.aliyuncs.com")){
            URL url = new URL(templatePath);
            InputStream in = url.openStream();
            //渲染表格
            HackLoopTableRenderPolicy policy = new HackLoopTableRenderPolicy();
            //此list是步骤四中动态传入的数据列表
            Configure config = Configure.newBuilder().bind("list", policy).build();
            template = XWPFTemplate.compile(in,config).render(paramsMap);
        }else {
        //hetong muban wei bendi wenjian 
            template = XWPFTemplate.compile(templatePath).render(paramsMap);
        }
        //将填写后的内容写入生成的合同中
        template.writeToFile(filePath);
        template.close();
        logger.info("【生成的电子合同本地存放路径为】:{}",JSON.toJSONString(filePath));
        return filePath;
    }
   }

4、组装合同填充的内容
Map<String,Object> paramsMap = new HashMap<>();
        paramsMap.put("bianHao",bianHao);
        paramsMap.put("jfName",contractDTO.getJfName());
        paramsMap.put("yfName",contractDTO.getYfName());
        paramsMap.put("zongJia",contractDTO.getPrice());
        paramsMap.put("dingJin",contractDTO.getDingJin());
        paramsMap.put("yuKuan",contractDTO.getWeiKuan());
        paramsMap.put("yuKuanJieQingDate",contractDTO.getJiaoFuDateEnd()+"日");
        paramsMap.put("jiaoFuDateStart",contractDTO.getJiaoFuDateStart()+"日");
        paramsMap.put("jiaoFuDateEnd",contractDTO.getJiaoFuDateEnd()+"日");
        paramsMap.put("gaoZhiDay",contractDTO.getGaoZhiDay());
        paramsMap.put("jiaoFuDiDian",contractDTO.getJiaoFuDiDian());
        paramsMap.put("yunShuFangShi",contractDTO.getYunShuFangShi());
        paramsMap.put("jfwyJin",contractDTO.getJfwyJin());
        paramsMap.put("jfwyDay",contractDTO.getJfwyDay());
        //牛只列表
        List<Map<String,Object>> cattleList=new ArrayList<Map<String,Object>>();
        for (int i = 0; i < contractDTO.getCattleLists().size(); i++) {
            Map<String,Object> cattleMap = new HashMap<>();
            cattleMap.put("index",i+1);
            cattleMap.put("eartagNo",contractDTO.getCattleLists().get(i).getEartagNo());
            cattleMap.put("pz",contractDTO.getCattleLists().get(i).getVarieties());
            cattleMap.put("sex",contractDTO.getCattleLists().get(i).getGender());
            cattleMap.put("ms",contractDTO.getCattleLists().get(i).getFulColor());
            cattleMap.put("tz",contractDTO.getCattleLists().get(i).getSecondWeight());
            cattleMap.put("jiaGe",contractDTO.getCattleLists().get(i).getFinalPrice());
            cattleList.add(cattleMap);
        }
         //金额中文大写
        String zongJiaChinese = MoneyUtils.change(Double.valueOf(contractDTO.getPrice()));
        paramsMap.put("rmb",zongJiaChinese);
        paramsMap.put("list",cattleList);
       //根据日期创建文件夹
        String nowDate = DateUtils.dateTimeNow("yyyyMMdd");
        //获取当前运行环境是生产还是测试
        String active = SpringUtils.getActiveProfile();
        String templatePath = null;  //模板存放路径  OSS地址
        String createFileDir = null; //生成的合同临时存放路径
        //生产环境
        if("prod".equals(active)){
            templatePath = contractConfig.getDevTemplateUrl();
            createFileDir = contractConfig.getProdContractPath() + nowDate + "/";
        }
        //测试环境
        if("dev".equals(active)){
            templatePath = contractConfig.getDevTemplateUrl();
            createFileDir = contractConfig.getDevContractPath() + nowDate + "//";
        }
       //生成的合同名
        String createFileName = contractDTO.getJfName() + "_" +
                contractDTO.getYfName() + "_" + nowDate + "_" + contractConfig.getFileNameSuffix();
        logger.info("--------------【开始动态填充电子合同】--------------");
        InputStream is = null;
        String ossPath = null;
        String wordPath = null;
        try {
            wordPath = WordUtils.createWord(templatePath,createFileDir,createFileName,paramsMap);
        } catch (IOException e) {
            logger.error("生成合同异常,异常信息:{}",JSON.toJSONString(e.getMessage()));
            e.printStackTrace();
            return null;
        }
三、效果预览

在这里插入图片描述
在这里插入图片描述

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
svm分类基于Matlab鸽群算法优化支持向量机(SVM)数据分类是一种利用鸽群算法来优化SVM模型参数并进行数据分类的方法。鸽群算法是一种基于自然界鸟群觅食行为的优化算法,通过模拟鸟群中鸟类之间的信息交流和协作,来求解最优化问题。 在使用鸽群算法优化SVM模型之前,我们首先需要了解SVM模型的原理。SVM是一种二分类模型,通过在特征空间中找到一个最优的超平面来实现数据的分类。在SVM模型中,支持向量是决定超平面位置和方向的关键要素。 鸽群算法优化SVM模型的过程如下: 1. 初始化鸽群规模和初始解。 2. 根据当前解,计算每个个体适应度值。适应度值反映了个体解的好坏程度。 3. 选择适应度最好的个体作为当前最佳解,并保存其对应的超平面参数。 4. 利用鸽群的信息交流和协作,更新所有鸽子的位置和速度。 5. 根据更新后的位置和速度,计算新解的适应度值。 6. 根据新解的适应度值,更新当前最佳解。 7. 重复步骤4-6,直至满足停止准则或达到最大迭代次数。 通过鸽群算法优化SVM模型,可以得到一组最佳的超平面参数,从而实现对数据的分类。这种方法能够克服传统的SVM模型由于初始解的不合理和局部最优解的问题,进而改善了分类结果的准确性和鲁棒性。 以下是一个简化的Matlab源码示例(仅供参考): ```matlab % 设置鸽群规模和最大迭代次数 N = 50; MaxIter = 100; % 初始化鸽子位置和速度 X = rand(N, 2); V = rand(N, 2); % 初始化最佳解和适应度值 BestX = zeros(1, 2); BestFitness = inf; % 迭代优化 for iter = 1:MaxIter % 计算适应度值 fitness = CalculateFitness(X); % 更新最佳解 [minFitness, minIndex] = min(fitness); if minFitness < BestFitness BestFitness = minFitness; BestX = X(minIndex, :); end % 更新速度和位置 V = UpdateVelocity(V, X, BestX); X = UpdatePosition(X, V); end % 输出最佳解和适应度值 disp('Best Solution:'); disp(BestX); disp('Best Fitness:'); disp(BestFitness); % 计算适应度值的函数 function fitness = CalculateFitness(X) % 计算每个个体的适应度值 % ... end % 更新速度的函数 function V = UpdateVelocity(V, X, BestX) % 根据鸽子当前位置和最佳解更新速度 % ... end % 更新位置的函数 function X = UpdatePosition(X, V) % 根据鸽子当前速度更新位置 % ... end ``` 以上是关于基于Matlab鸽群算法优化支持向量机(SVM)数据分类的简要介绍和示例源码。这种方法可以提高SVM模型的性能,但在实际应用中还需要根据具体情况进行调试和优化。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值