Java实现自定义字段

Java自定义字段

小黄最近在工作中遇到一个比较有意思的需求,在此跟大家分享一下

需求

这个需求是这样的,用户在添加表单时,除了一些固定信息填入之外,还可以自定义一些字段填入,例如一个用户表,默认情况下姓名和年龄是所有人都要填入的,除此之外,比如小红想多添加一个字段身高,小白想多添加一个字段体重…所以在用户填入表单时,就需要可拓展的自定义字段。

小红存在数据库中的字段如下:

{
    "name":"小红",
    "age":18,
    "height":180
}

小白存在数据库中的字段如下:

{
    "name":"小红",
    "age":18,
    "weight":65
}

分析

其实接口是非常简单的,就是一套CRUD接口,所以我们来分析一下如何解决这个问题。

首先我们先和前端约定好,如何传一个表单格式,关于自定义字段,小黄是这样定义的

  • en:表示字段名
  • cn:表示字段中文名
  • type:表示字段类型,因为前端需要靠这个渲染页面,和前端约定
  • required:表示是否必填项
  • value:表示字段值
[       
    {
        "en": "name",
        "cn": "用户名",
        "type": 1,
        "required":1,
        "value": "小红"
    }
]

思路一

使用MySQL,创建一张User表,表结构如下设计

  • ext:表示用户自定义字段内容,使用json格式存储

image-20230515085559691

使用这个方法可以将用户填入的多个自定义字段都存入在ext中,但是由于所有字段都挤在了一起,对于后续的可拓展性不是很高。

思路二

使用Mongo存储,Mongo是一个文档型数据库,每一条数据可以使用不同的字段存储,这对于我们这个需求来说就再适合不过了。

我们需要花两张表来存储信息,status用于表示逻辑删除

  • user表:用于存储字段对应的值,存储格式如下

    {
        "_id": "98c6db9985c8481a8d80f3555b6db326",
        "status": NumberInt("0"),
        "name": "yellowstar459",
        "age": NumberInt("18"),
        "height": 180.5,
        "happy": "唱跳rap"
    }
    
  • user_ext表:用于存储字段信息,存储格式如下,每个值分别对应cn,type,required

    {
        "_id": "98c6db9985c8481a8d80f3555b6db326",
        "status": NumberInt("0"),
        "name": "用户名,1,1",
        "age": "年龄,2,1",
        "height": "身高,3,1",
        "happy": "爱好,1,1"
    }
    

实现

这篇文章仅仅分享一下思路,代码写的都不是非常规范,见谅。

新增

定义一个接收前端参数的对象

@Data
public class Custom implements Serializable {
    private static final long serialVersionUID = 1L;

    private String en;

    private String cn;

    private Integer type;

    //1:必填,0:非必填
    private Integer required;

    private Object value;
}

编写业务类

/**
 * userExt对应
 * cn,type,required
 * status : 作为逻辑删除
 * @param list
 */
@PostMapping
public void add(@RequestBody List<Custom> list) {
    Document user = new Document();
    Document userExt = new Document();
    String id = UUID.randomUUID().toString().replaceAll("-", "");
    user.append("_id", id);
    user.append("status", 0);
    userExt.append("_id", id);
    userExt.append("status", 0);
    for (Custom custom : list) {
        user.append(custom.getEn(), custom.getValue());
        userExt.append(custom.getEn(), custom.getCn() + "," + custom.getType() + "," + custom.getRequired());
    }
    mongoTemplate.save(user, "user");
    mongoTemplate.save(userExt, "user_ext");
}

测试

我这边和前端约定type值

  • 1:字符串
  • 2:整型
  • 3:浮点型

通过postman发送如下请求

[       
    {
        "en": "name",
        "cn": "用户名",
        "type": 1,
        "required":1,
        "value": "yellowstar"
    },
    {
        "en": "age",
        "cn": "年龄",
        "type": 2,
        "required":1,
        "value": 18
    },
    {
        "en": "height",
        "cn": "身高",
        "type": 3,
        "required":1,
        "value": 180.5
    },
    {
        "en": "happy",
        "cn": "爱好",
        "type": 1,
        "required":1,
        "value": "唱跳rap"
    }
]

存入Mongo的数据分别为

image-20230515091648485

根据id查询

返回类

返回类需要一个id,在添加时将两张表的id设为相同

@Data
public class CustomVo implements Serializable {
    private static final long serialVersionUID = 1L;

    private String _id;

    private List<Custom> list;
}

业务类

@GetMapping("/{id}")
public CustomVo getById(@PathVariable(value = "id") String id) {
    Document user = mongoTemplate.findOne(Query.query(Criteria.where("_id").is(id).and("status").is(0)), Document.class, "user");
    Document userExt = mongoTemplate.findOne(Query.query(Criteria.where("_id").is(id).and("status").is(0)), Document.class, "user_ext");
    if (Objects.isNull(user) || Objects.isNull(userExt)) {
        throw new RuntimeException("数据不存在");
    }
    CustomVo customVo = new CustomVo();
    customVo.set_id(user.getString("_id"));
    List<Custom> list = new ArrayList<>();
    for (Map.Entry<String, Object> entry : user.entrySet()) {
        if ("_id".equals(entry.getKey()) || "status".equals(entry.getKey())) {
            continue;
        }
        Custom custom = new Custom();
        custom.setEn(entry.getKey());
        custom.setValue(entry.getValue());
        String ext = userExt.getString(entry.getKey());
        String[] split = ext.split(",");
        custom.setCn(split[0]);
        custom.setType(Integer.valueOf(split[1]));
        custom.setRequired(Integer.valueOf(split[2]));
        list.add(custom);
    }
    customVo.setList(list);
    return customVo;
}

更新

@PutMapping
public void update(@RequestBody CustomVo vo) {
    Document user = new Document();
    Document userExt = new Document();
    user.append("_id", vo.get_id());
    user.append("status", 0);
    userExt.append("_id", vo.get_id());
    userExt.append("status", 0);
    for (Custom custom : vo.getList()) {
        user.append(custom.getEn(), custom.getValue());
        userExt.append(custom.getEn(), custom.getCn() + "," + custom.getType() + "," + custom.getRequired());
    }
    Update userUpdate = Update.fromDocument(user);
    Update userExtUpdate = Update.fromDocument(userExt);
    mongoTemplate.updateFirst(Query.query(Criteria.where("_id").is(vo.get_id()).and("status").is(0)), userUpdate, "user");
    mongoTemplate.updateFirst(Query.query(Criteria.where("_id").is(vo.get_id()).and("status").is(0)), userExtUpdate, "user_ext");
}

删除

只需要修改status状态即可

@DeleteMapping("/{id}")
public void delete(@PathVariable(value = "id") String id) {
    Update update = Update.update("status", 1);
    mongoTemplate.updateFirst(Query.query(Criteria.where("_id").is(id).and("status").is(0)), update, "user");
    mongoTemplate.updateFirst(Query.query(Criteria.where("_id").is(id).and("status").is(0)), update, "user_ext");
}

列表

一般情况下,列表中肯定只显示一些通用的字段,目前只需要返回nameage即可

实体类

@Data
public class User {
    private String name;
    private Integer age;
}

业务逻辑

@GetMapping
public List<User> list(){
    Criteria criteria = Criteria.where("status").is(0);
    List<User> users = mongoTemplate.find(Query.query(criteria), User.class, "user");
    return users;
}

以上就是小黄的思路分析啦,有更好方法的小伙伴可以尽情提出想法,欢迎评论

  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值