解决@Valid校验List失败问题

3,680 阅读2分钟

背景

在实际的项目开发中,当参数是List集合方式时,往往我们需要对List集合属性进行校验。

场景1

@PostMapping("/preOrder")
public Result<T> doSomething(@RequestBody @Valid List<OrderDTO> list) throws Exception {
    return orderService.doSomething(list);
}

如果是直接使用@Valid修饰这样的方式校验的话,经过测试,@Valid是无法发挥作用的。

场景2

@PostMapping("/preOrder")
public Result<T> doSomething(@RequestBody @Valid OrderDTOList list) throws Exception {
    return orderService.doSomething(list);
}
@Data public class OrderDTOList() { 
    @Valid private List<OrderDTO> orderDTOList; 
}

这种情况是情况1的进阶,这样设计至少@Valid能够发挥作用了。但是太麻烦,需要专门写一个OrderDTOList类,简直不优雅。同时外接的JSON需要需要多套一层,如下:

{
    "orderDTOList":[
        {
            "userId":"1",
            "productCode":"A102",
            "num":1
        },
        {
            "userId":"1",
            "productCode":"A103",
            "num":2
        }
    ]
}    

结果参数多一层"orderDTOList"对前端调用方来说也是麻烦。

类似场景1那样子设计行不通的原因是,java.util.List(ArrayList)内部通过持有一个数组来保存对象们,而作为Java官方的类,内部肯定不会在数组上声明@Valid,所以内部的对象们没有得到应有的递归校验。

所以,考虑使用一种新的java.util.List实现,来变相的达到列表校验的效果。

解决方案

/**
 * 可被校验的List
 * @param <E> 元素类型
 */  
@Data
public class ValidList <E> implements List<E> {

    @Valid
    private List<E> list = new ArrayList<>();

    @Override
    public int size() {
        return list.size();
    }

    @Override
    public boolean isEmpty() {
        return list.isEmpty();
    }

    @Override
    public boolean contains(Object o) {
        return list.contains(o);
    }

    @Override
    public Iterator<E> iterator() {
        return list.iterator();
    }

    @Override
    public Object[] toArray() {
        return list.toArray();
    }

    @Override
    public <T> T[] toArray(T[] a) {
        return list.toArray(a);
    }

    @Override
    public boolean add(E e) {
        return list.add(e);
    }

    @Override
    public boolean remove(Object o) {
        return list.remove(o);
    }

    @Override
    public boolean containsAll(Collection<?> c) {
        return list.containsAll(c);
    }

    @Override
    public boolean addAll(Collection<? extends E> c) {
        return list.addAll(c);
    }

    @Override
    public boolean addAll(int index, Collection<? extends E> c) {
        return list.addAll(index, c);
    }

    @Override
    public boolean removeAll(Collection<?> c) {
        return list.removeAll(c);
    }

    @Override
    public boolean retainAll(Collection<?> c) {
        return list.retainAll(c);
    }

    @Override
    public void clear() {
        list.clear();
    }

    @Override
    public E get(int index) {
        return list.get(index);
    }

    @Override
    public E set(int index, E element) {
        return list.set(index, element);
    }

    @Override
    public void add(int index, E element) {
        list.add(index, element);
    }

    @Override
    public E remove(int index) {
        return list.remove(index);
    }

    @Override
    public int indexOf(Object o) {
        return list.indexOf(o);
    }

    @Override
    public int lastIndexOf(Object o) {
        return list.lastIndexOf(o);
    }

    @Override
    public ListIterator<E> listIterator() {
        return list.listIterator();
    }

    @Override
    public ListIterator<E> listIterator(int index) {
        return list.listIterator(index);
    }

    @Override
    public List<E> subList(int fromIndex, int toIndex) {
        return list.subList(fromIndex, toIndex);
    }

}

可以看到具有二象性,ValidList LIKE-A java.util.List,同时也是个holder,对外的方法全部会移交给持有的list处理。

也就是说,ValidList与java.util.List的对外功能完全一致。

对于Spring参数绑定来说,JSON转换后也能够绑定到ValidList对象上(就像能绑定到java.util.ArrayList对象上一样。)

实现了上述的效果后,直接在list上声明一个@Valid就解决所有问题了。

最终的方法定义方案

@PostMapping("/preOrder")
public Result<T> doSomething(@RequestBody @Valid ValidList<OrderDTO> list) throws Exception {
    return orderService.doSomething(list);
}

参数格式:

[{"userId":"1","productCode":"A102","num":1},{"userId":"1","productCode":"A103","num":2}]