统一设置VO中字段默认值工具类

背景:将VO中值为空的字符串字段,设置一个默认值.

如果逐个对字段进行判断, 或者在VO中中get方法判断,都会造成太多重复代码. 故通过工具类进行统一设置达到代码简化优雅.

工具类如下:

import com.gexin.fastjson.JSON;
import org.apache.commons.lang3.StringUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.beans.BeanInfo;
import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.Map;
import java.util.Objects;

public class ClassUtils {
    private static Logger logger = LoggerFactory.getLogger(ClassUtils.class);

    //空数据暂无
    private final static String defaultStringData = "暂无";

    /**
     * 设置bean中String类型的空字符串字段的默认值"暂无"
     * @methodName setStringDefaultValue
     * @param bean
     * @author admin
     * @date 2022/12/12
     * @return void
     */
    public static <T> T setStringDefaultValue(T bean){
        try {
            Class<?> type = bean.getClass();
            Map<String, Object> returnMap = new HashMap<>();
            BeanInfo beanInfo = Introspector.getBeanInfo(type);
            PropertyDescriptor[] propertyDescriptors = beanInfo.getPropertyDescriptors();
            for (PropertyDescriptor descriptor : propertyDescriptors) {
                String propertyName = descriptor.getName();//字段名
                if (!Objects.equals(propertyName,"class")) {
                    Class<?> propertyType = descriptor.getPropertyType();
                    String name = propertyType.getName();//字段类型
                    //字段类型为字符串
                    if (Objects.equals(name, "java.lang.String")){
                        Method readMethod = descriptor.getReadMethod();
                        Object result = readMethod.invoke(bean);//字段值
                        if (StringUtils.isBlank((String)result)){
                            Method writeMethod = descriptor.getWriteMethod();
                            //设置默认值
                            writeMethod.invoke(bean, defaultStringData);
                        }
                    }
                }
            }

            return bean;
        } catch (Exception e) {
            e.printStackTrace();
        }

        return bean;
    }


    public static class TestVO{

        @ApiModelProperty("id")
        private Long id;

        @ApiModelProperty("名称")
        private String name;

        public Long getId() {
            return this.id;
        }

        public void setId(Long id) {
            this.id = id;
        }

        public String getName() {
            return this.name;
        }

        public void setName(String name) {
            this.name = name;
        }
    }

    public static void main(String[] args) {
        TestVO vo = new TestVO();
        vo.setId(1L);
        //vo.setName("a");
        System.out.println(JSON.toJSONString(vo));
        System.out.println("--------------------");
        //设置默认值
        TestVO newVO = ClassUtils.setStringDefaultValue(vo);
        System.out.println(JSON.toJSONString(newVO));

    }

}

<think>好的,我现在要分析用户的问题,关于MyBatis-Plus中@TableField注解的insertStrategy和updateStrategy都设置为FieldStrategy.IGNORED时,什么情况下会导致payType字段设置为空的情况。 首先,我需要回忆一下MyBatis-Plus的字段策略。FieldStrategy有几个选项,比如IGNORED、NOT_NULL、NOT_EMPTY等。IGNORED表示不管字段值是否为null,都会拼接到SQL语句中。所以当使用这个策略时,即使字段值是null,也会被插入或更新到数据库里。 接下来,用户的问题是,设置这两个策略为IGNORED后,什么时候会导致payType字段设置为空。也就是在什么情况下,这个字段会被赋值为null,从而覆盖数据库中的现有值。 可能的场景包括: 1. **显式设置字段为null**:如果在Java对象中,payType字段被明确设置为null,那么在插入或更新时,数据库中的对应列会被设置为NULL。比如,当创建一个新的对象时,payType设为null,插入操作就会将数据库中的pay_type设为NULL。或者在更新时,将payType设为null,更新后数据库中的该字段也会变为NULL。 2. **部分更新操作**:比如使用updateById方法时,如果传入的对象中payType字段为null,由于updateStrategy是IGNORED,这个null值会被更新到数据库,覆盖原有的值。这时候即使原本数据库有值,也会被设为NULL。 3. **表单提交或接口传参时未提供该字段**:如果前端提交的数据中缺少payType字段,或者传入了null,而Java对象中该字段未被正确初始化,可能会保留为null。此时进行插入或更新操作,就会导致数据库中的pay_type被设为NULL。 需要注意的情况是,当使用如BeanUtils.copyProperties这样的工具类时,如果源对象没有该字段的值,目标对象的payType可能被设置为null,进而导致更新时为null的情况。 可能用户的实际场景是,他们担心在某些情况下,即使没有显式操作payType字段,它也会被意外地设为null。比如在部分更新的情况下,传入了部分字段的对象,而其他字段可能被默认设置为null,导致覆盖数据库中的值。 另外,用户可能需要了解如何避免这种情况,比如使用其他字段策略,或者在业务逻辑中确保不会传递null值。或者使用DTO来过滤掉不需要更新的字段,防止null的传递。 总结来说,当字段策略设为IGNORED时,任何情况下只要payType字段Java对象中为null,无论是显式设置还是隐式未赋值,都会导致数据库中的对应列被更新为NULL。因此,需要特别注意在业务逻辑中处理该字段的值,避免意外的null覆盖。</think>--- ### 一、核心结论 当字段配置`insertStrategy = updateStrategy = FieldStrategy.IGNORED`时,**以下三种情况会导致`pay_type`字段被设为`NULL`**: | 触发场景 | 插入操作 | 更新操作 | 典型风险案例 | |-------------------------------|----------|----------|-----------------------------| | 主动设置`payType=null` | ✅ | ✅ | 接口参数显式传空值 | | 未初始化字段默认`null` | ✅ | ✅ | 前端未传参导致DTO字段为null | | 对象拷贝覆盖原始值 | - | ✅ | BeanUtils.copyProperties误覆盖 | --- ### 二、策略机制解析 #### MyBatis-Plus字段策略对照表 ```java public enum FieldStrategy { IGNORED, // 无视字段值,直接参与SQL NOT_NULL, // 非null时才参与 NOT_EMPTY, // 非空(字符串还需非空串)时参与 NEVER // 绝对不参与 } ``` #### `IGNORED`策略核心逻辑 ```java // 伪代码演示处理逻辑 if (fieldStrategy == IGNORED) { sql.append("pay_type = #{payType}"); // 无条件拼接字段 } // 生成类似:UPDATE table SET pay_type=null WHERE ... ``` --- ### 三、危险场景示例 #### 场景1:显式传递null值 ```java // 创建对象时主动赋null CollegeSetting setting = new CollegeSetting(); setting.setPayType(null); // 明确设空 mapper.insert(setting); // 插入null到pay_type列 ``` #### 场景2:未传参导致字段默认null ```java // 请求参数未包含payType字段 @PostMapping("/update") public void update(@RequestBody CollegeSettingDTO dto) { // dto.getPayType() == null service.updateById(dto.toEntity()); // 更新时覆盖为null } ``` #### 场景3:对象拷贝污染 ```java CollegeSetting dbEntity = getFromDB(); // 原pay_type="ALIPAY" CollegeSetting param = new CollegeSetting(); param.setName("新名称"); // 拷贝时未排除payType字段 BeanUtils.copyProperties(param, dbEntity); // dbEntity.payType变为null! mapper.updateById(dbEntity); // pay_type被置空 ``` --- ### 四、防御方案 #### 方案1:严格校验入参(Controller层) ```java @PatchMapping("/settings") public Result updateSetting(@Valid @RequestBody SettingUpdateVO vo) { // 使用校验注解约束 if(vo.getPayType() == null) { throw new IllegalArgumentException("支付类型不可为空"); } // ... } ``` #### 方案2:使用非全量更新方法 ```java // 只更新指定字段 new LambdaUpdateWrapper<CollegeSetting>() .set(CollegeSetting::getName, "新名称") // 明确指定更新字段 .eq(CollegeSetting::getId, 1) .update(); ``` #### 方案3:自定义类型处理器 ```java @TableField(typeHandler = EmptyStringAsNullHandler.class) private String payType; // 自定义处理逻辑 public class EmptyStringAsNullHandler extends BaseTypeHandler<String> { @Override public void setNonNullParameter(...) { // 空字符串转为null的逻辑 } } ``` --- ### 五、配置建议等级 | 方案 | 实施难度 | 安全性 | 适用场景 | |--------------------------|----------|--------|----------------------| | 入参校验+DTO隔离 | ★☆☆☆☆ | ★★★★☆ | 所有更新操作 | | 动态字段更新 | ★★★☆☆ | ★★★★☆ | 局部更新场景 | | 数据库默认值约束 | ★★☆☆☆ | ★★★☆☆ | 表结构可控时 | | 字段策略改为NOT_NULL | ★☆☆☆☆ | ★★★★★ | 需要严格防空的字段 | --- 通过理解这些机制,可以有效避免因策略配置不当导致的数据异常覆盖问题。关键点在于:**明确区分"未传递值"与"主动赋空值"的业务语义**,根据实际需求选择匹配的字段策略。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值