前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >jcsv导入导出动态扩展思路

jcsv导入导出动态扩展思路

作者头像
一笠风雨任生平
发布2022-01-06 14:18:46
4220
发布2022-01-06 14:18:46
举报
文章被收录于专栏:服务化进程

我整理的jcsv工具类库简介:csv导入导出组件jcsv jcsv可以支持导入校验规则动态扩展,以及导出规则动态扩展。 下面来介绍下扩展思路

问题

我们支持文件校验、题头校验、列校验等,规则已经多样化,已经覆盖了绝大多数场景,但是难免有一些个性化的校验需要与业务挂钩,这种该怎么办呢?每次都需要在jcsv组件中加规则?这个很容易导致规则越来越多,越来越难以维护。

jcsv导入动态扩展思路

导入配置如下:

代码语言:javascript
复制
csv-config:
  importc:
    - id: aa
      desc: "通用上传"
      max-size: 30 #单位m
      separator: ","
      start-row: 2
      col-from-header: true
      check-column-size: false
      header-validcate: throngBaseTdHeaderValidator
      valicate:
        - { col: 0, name: email,required: true,validateRegex: "^[éûàçùôîèíá¡N¿UóñúäöüßàèùåäöãáéàêçíşçöüğşàâäèéêëîïôœùûüÿçÀÂÄÈÉÊËÎÏÔŒÙÛÜŸÇA-Za-z0-9_\\-\\.\\u4e00-\\u9fa5]+@[a-zA-Z0-9_-]+(\\.[a-zA-Z0-9_-]+)+$", hint: "邮箱地址错误"}
        - { col: 1, name: language ,required: true,hint: "语言错误" ,validator: throngBaseTdValidator ,td-id: 11}
        - { col: 2, name: site_id,required: true,hint: "站点错误" ,validator: throngBaseTdValidator ,td-id: 13}
        - { col: 4, name: member_id,validateRegex: "^\\d{1,10}$", hint: "请填写10位以内的数字",required: true}
        - { col: 0, name: device_id,required: true}

这里我们就需要设计一个接口来支持动态扩展,这里就是使用策略模式来进行动态扩展,规则可替换。

1、我们先定义一个动态的题头校验规则以及一个列校验规则 如下:

代码语言:javascript
复制
public interface HeaderValidator {
    public boolean validcate(String[] headerStrs, List<ColValidcateProperties> valicates);
}

@FunctionalInterface
public interface Validator {
    public boolean validcate(String value, ColValidcateProperties col, String[] row, Object params);
}

个性化规则需要实现上面的两个接口 参考上面的配置参数 importc.header-validcate 与importc.valicate[0].valicate 这里有两种方式,1、bean交给spring托管,配置beanName,2、配置类全路径,通过class.forName来实例化对象,实例化的对象交给ValidcateFactory来进行管理。 这里我们是采用方案1,下面是组件中相关伪代码

代码语言:javascript
复制
for (int i = 0; i < colValues.length; i++) {
               //上面若干代码省略
                    if (StringUtils.isNotBlank(v.getValidator())) {
                        Validator validator = (Validator) SpringContext.getSingleton().getBean(v.getValidator());
                        if (validator != null && !validator.validcate(colValues[i].trim(), v, colValues, request.getParams())) {
                            sb.append(errorRow + v.getName() + v.getHint()).append(enterLine);
                            bFlag=false;
                            continue;
                        }
                    }
                    //下面若干代码省略
 }

下面给一个详细实例来校验:我们需要根据数据库中的字典来校验规则,如下

代码语言:javascript
复制
@Service("aaValidator")
public class AAValidator implements Validator {
    private ConcurrentHashMap<String, List<String>> tdListMap=new ConcurrentHashMap<>();

    @Autowired
    private DemoMapperdemoMapper;
    @Override
    public boolean validcate(String value, ColValidcateProperties v, String[] row, Object params) {
        int product=0;
        if(params!=null){
            product=Integer.parseInt(params.toString());
        }
        List<String> values=null;
        if("language".equals(v.getName())){
            values=getTdList("11",product);
        }else if("site_id".equals(v.getName())){
            values=getTdList("13",product);
        }
        if(values!=null&&values.size()>0&&!values.contains(value)){
            return false;
        }
        return true;
    }
    public List<String> getTdList(String id, int product){
        List<String> list = demoMapper.queryValues();
        return list;
    }
}

然后配置列如下:

代码语言:javascript
复制
 - { col: 1, name: language ,required: true,hint: "语言错误" ,validator: aaValidator}

jcsv导出动态扩展

下面也介绍下导出动态扩展,这里提供了一个分页接口,如下:

代码语言:javascript
复制
public interface Paging {
    public List<Map> getList(long pageSize, int pageNum);

    public long getTotal();
}

下面就举个例子,使用游标查询es使用如下:

代码语言:javascript
复制
 int total = getGoodsCount(tspManualValRule);
String path = csvContext.export("tag_val_detail_export", new Paging() {
                private String cursor;

                @Override
                public List<Map> getList(long pageSize, int pageNum) {
            
                    List<Map> result = null;
                    if (pageNum == 1) {
                        JSONObject jsonObject = JSONObject.parseObject(aaRpc.getListPage(
                                params1, params2, (int) pageSize, null));
                        JSONObject data = (JSONObject) jsonObject.get("data");
                        JSONArray columns = (JSONArray) data.get("columns");
                        cursor = String.valueOf(data.get("cursor"));
                        if (ObjectUtils.isEmpty(columns)) {
                            return null;
                        }
                        result = converter.convert(jsonObject);
                    } else {
                        JSONObject jsonObject = JSONObject.parseObject(aaRpc.getListPageNext(cursor));
                        result = converter.convert(jsonObject);
                    }
                  
                    return result;
                }

                @Override
                public long getTotal() {
                    return total >= 500000 ? 10000 : total;
                }
            });
本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2020/07/19 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 问题
  • jcsv导入动态扩展思路
  • jcsv导出动态扩展
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档