Maison > Java > JavaBase > Apprenez à connaître les pièges de Lombok

Apprenez à connaître les pièges de Lombok

coldplay.xixi
Libérer: 2020-10-09 17:00:02
avant
2726 Les gens l'ont consulté

La rubrique

Tutoriel de base Java vous présente les pièges de Lombok, qui est simple à utiliser.

Apprenez à connaître les pièges de Lombok

Préface

Le plug-in Lombok a été introduit dans le projet l'année dernière, ce qui a vraiment libéré les mains et remplacé certaines tâches simples répétitives (Getter, Setter , toString, etc. Écriture de méthodes), cependant, au cours du processus d'utilisation, certains pièges ont également été découverts. Au début, je n'ai pas réalisé qu'il s'agissait d'un problème de Lombok. Plus tard, j'ai suivi le code source des autres composants correspondants et j'ai trouvé. que c'était un problème de Lombok !

Pièges de la méthode Setter-Getter

Découverte de problèmes

Nous utilisons principalement l'annotation de la méthode Setter-Getter de Lombok dans le projet, qui est l'annotation combinée @Data, mais dans Lors du processus d'insertion de données à l'aide de Mybatis, un problème est survenu. Le problème est décrit comme suit :

我们有个实体类:
@Data
public class NMetaVerify{
    private NMetaType nMetaType;
    private Long id;
    ....其他属性
}复制代码
Copier après la connexion

Lorsque nous avons utilisé Mybatis pour insérer des données, nous avons constaté que d'autres attributs pouvaient être insérés normalement, à l'exception du nMetaType. l'attribut était dans la base de données. A toujours été nul.

Solution

Lorsque je débogue le code du projet pour appeler la méthode insert SQL de Mybatis, je vois que l'attribut nMetaType de l'objet NMetaVerify a toujours des données, mais après avoir exécuté l'insertion, la base de données Le champ nMetaType est toujours nul. Au départ, je pensais que mon type d'énumération était mal écrit. Après avoir examiné d'autres champs qui ont également des types d'énumération, ils peuvent être insérés normalement dans la base de données, ce qui me rend encore plus confus. J'ai donc retracé le code source de Mybatis et découvert que Mybatis utilisait la réflexion pour obtenir l'attribut nMetaType, en utilisant la méthode getxxxx pour l'obtenir. Cependant, j'ai trouvé que la méthode get de nMetaType semblait être un peu différente de la méthode getxxxx requise. par Mybatis Idem. Problème trouvé !

Raison

La méthode get-set générée par Lombok pour les attributs avec la première lettre en minuscule et la deuxième lettre en majuscule est générée par Mybatis et idea ou la méthode get-set officiellement reconnue par Java. La différence :

#Lombok生成的Get-Set方法
@Data
public class NMetaVerify {
    private Long id;
    private NMetaType nMetaType;
    private Date createTime;
    
    public void lombokFound(){
        NMetaVerify nMetaVerify = new NMetaVerify();
        nMetaVerify.setNMetaType(NMetaType.TWO); //注意:nMetaType的set方法为setNMetaType,第一个n字母大写了,
        nMetaVerify.getNMetaType();                                  //getxxxx方法也是大写
    }
}复制代码
Copier après la connexion
rrree

Mybatis (version 3.4.6) analyse la méthode get-set pour obtenir le code source du nom de l'attribut :

#idea,Mybatis,Java官方默认的行为为:
public class NMetaVerify {
    private Long id;
    private NMetaType nMetaType;
    private Date createTime;

    public Long getId() {
        return id;
    }

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

    public NMetaType getnMetaType() {//注意:nMetaType属性的第一个字母小写
        return nMetaType;
    }

    public void setnMetaType(NMetaType nMetaType) {//注意:nMetaType属性的第一个字母小写
        this.nMetaType = nMetaType;
    }

    public Date getCreateTime() {
        return createTime;
    }

    public void setCreateTime(Date createTime) {
        this.createTime = createTime;
    }
}复制代码
Copier après la connexion

Mybatis analyse la méthode get-set pour tester le nom de l'attribut

package org.apache.ibatis.reflection.property;

import java.util.Locale;

import org.apache.ibatis.reflection.ReflectionException;

/**
 * @author Clinton Begin
 */
public final class PropertyNamer {

        private PropertyNamer() {
            // Prevent Instantiation of Static Class
        }

        public static String methodToProperty(String name) {
            if (name.startsWith("is")) {//is开头的一般是bool类型,直接从第二个(索引)开始截取(简单粗暴)
                    name = name.substring(2);
            } else if (name.startsWith("get") || name.startsWith("set")) {//set-get的就从第三个(索引)开始截取
                    name = name.substring(3);
            } else {
                    throw new ReflectionException("Error parsing property name '" + name + "'.  Didn't start with 'is', 'get' or 'set'.");
            }
            //下面这个判断很重要,可以分成两句话开始解释,解释如下
            //第一句话:name.length()==1
            //                      对于属性只有一个字母的,例如private int x;
            //                      对应的get-set方法是getX();setX(int x);
            //第二句话:name.length() > 1 && !Character.isUpperCase(name.charAt(1)))
            //                      属性名字长度大于1,并且第二个(代码中的charAt(1),这个1是数组下标)字母是小写的
            //                      如果第二个char是大写的,那就直接返回name
            if (name.length() == 1 || (name.length() > 1 && !Character.isUpperCase(name.charAt(1)))) {
                    name = name.substring(0, 1).toLowerCase(Locale.ENGLISH) + name.substring(1);//让属性名第一个字母小写,然后加上后面的内容
            }

            return name;
        }

        public static boolean isProperty(String name) {
                return name.startsWith("get") || name.startsWith("set") || name.startsWith("is");
        }

        public static boolean isGetter(String name) {
                return name.startsWith("get") || name.startsWith("is");
        }

        public static boolean isSetter(String name) {
                return name.startsWith("set");
        }

}复制代码
Copier après la connexion

Solution

    @Test
    public void foundPropertyNamer() {
        String isName = "isName";
        String getName = "getName";
        String getnMetaType = "getnMetaType";
        String getNMetaType = "getNMetaType";

        Stream.of(isName,getName,getnMetaType,getNMetaType)
                .forEach(methodName->System.out.println("方法名字是:"+methodName+" 属性名字:"+ PropertyNamer.methodToProperty(methodName)));
    }
    
    #输出结果如下:
    方法名字是:isName 属性名字:name 
    方法名字是:getName 属性名字:name 
    方法名字是:getnMetaType 属性名字:nMetaType //这个以及下面的属性第二个字母都是大写,所以直接返回name
    方法名字是:getNMetaType 属性名字:NMetaType复制代码
Copier après la connexion

Problème avec l'annotation @Accessor(chain = true)

Problème trouvé

Exporter à l'aide d'easyexcel(github.com /alibaba/eas…) Lorsque j'ai vérifié, j'ai trouvé que les exportations de classes d'entités précédentes étaient toutes normales, mais maintenant la classe d'entités nouvellement ajoutée n'est pas normale. Après comparaison, il a été constaté que la classe d'entités nouvellement ajoutée ajoutait le @Accessor(. chain = true) annotation. Notre objectif principal est Il est pratique pour nous d'appeler la méthode set dans une chaîne :

1.修改属性名字,让第二个字母小写,或者说是规定所有的属性的前两个字母必须小写
2.如果数据库已经设计好,并且前后端接口对接好了,不想修改,那就专门为这种特殊的属性使用idea生成get-set方法复制代码
Copier après la connexion

Reason

La couche inférieure d'easyexcel utilise cglib comme boîte à outils de réflexion :

new UserDto()
.setUserName("")
.setAge(10)
........
.setBirthday(new Date());复制代码
Copier après la connexion

Mais cglib utilise la méthode rt A de Java de la classe Introspector dans .jar :

com.alibaba.excel.read.listener.ModelBuildEventListener 类的第130行
BeanMap.create(resultModel).putAll(map);

最底层的是cglib的BeanMap的这个方法调用

abstract public Object put(Object bean, Object key, Object value);复制代码
Copier après la connexion

Solution

# Introspector.java 第520行
if (int.class.equals(argTypes[0]) && name.startsWith(GET_PREFIX)) {
   pd = new IndexedPropertyDescriptor(this.beanClass, name.substring(3), null, null, method, null);
   //下面这行判断,只获取返回值是void类型的setxxxx方法
 } else if (void.class.equals(resultType) && name.startsWith(SET_PREFIX)) {
    // Simple setter
    pd = new PropertyDescriptor(this.beanClass, name.substring(3), null, method);
    if (throwsException(method, PropertyVetoException.class)) {
       pd.setConstrained(true);
    }
}复制代码
Copier après la connexion

Recommandations d'apprentissage gratuites associées : Tutoriel de base Java

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Étiquettes associées:
source:juejin.im
Déclaration de ce site Web
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn
Tutoriels populaires
Plus>
Derniers téléchargements
Plus>
effets Web
Code source du site Web
Matériel du site Web
Modèle frontal