Heim  >  Artikel  >  Java  >  Lernen Sie Lomboks Fallstricke kennen

Lernen Sie Lomboks Fallstricke kennen

coldplay.xixi
coldplay.xixinach vorne
2020-10-09 17:00:022649Durchsuche

Die Spalte „Einfaches Java-Tutorial“ führt Sie in die Fallstricke von Lombok ein, das einfach zu verwenden ist.

VorwortLernen Sie Lomboks Fallstricke kennen

Das Lombok-Plugin wurde letztes Jahr in das Projekt eingeführt, wodurch die Hände wirklich frei wurden und einige sich wiederholende einfache Arbeiten (Schreiben von Getter, Setter, toString und anderen Methoden) ersetzt wurden Bei der Verwendung habe ich einige Fallstricke entdeckt. Zuerst war mir nicht klar, dass es sich um ein Lombok-Problem handelte. Später habe ich den Quellcode der entsprechenden anderen Komponenten verfolgt und herausgefunden, dass es sich um ein Lombok-Problem handelte.

Fallstricke der Setter-Getter-Methode

Problemerkennung

Wir verwenden im Projekt hauptsächlich die Annotationen der Setter-Getter-Methode von Lombok, also die kombinierte Annotation @Data, aber beim Einfügen von Daten mit Mybatis trat ein Problem auf Das Problem wird wie folgt beschrieben:

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

Wenn wir Mybatis zum Einfügen von Daten verwenden, stellen wir fest, dass andere Attribute normal eingefügt werden können, das nMetaType-Attribut jedoch in der Datenbank immer null ist.

Lösung

Wenn ich den Projektcode debugge, um die Methode aufzurufen, die dem Einfügen von SQL in Mybatis entspricht, sehe ich, dass das nMetaType-Attribut des NMetaVerify-Objekts noch Daten enthält, das nMetaType-Feld der Datenbank jedoch nach dem Ausführen des Einfügens immer vorhanden ist Null. Ursprünglich dachte ich, mein Aufzählungstyp sei falsch geschrieben. Ich habe mir andere Felder angesehen, die ebenfalls Aufzählungstypen haben, und das hat mich noch mehr verwirrt. Also habe ich den Quellcode von Mybatis verfolgt und festgestellt, dass Mybatis Reflektion verwendet hat, um das nMetaType-Attribut abzurufen, und dabei die getxxxx-Methode verwendet hat. Ich habe jedoch festgestellt, dass sich die get-Methode von nMetaType ein wenig von der erforderlichen getxxxx-Methode unterscheidet von Mybatis. Problem gefunden!

Grund

Die von Lombok generierte Get-Set-Methode für Attribute mit dem ersten Buchstaben in Kleinbuchstaben und dem zweiten Buchstaben in Großbuchstaben unterscheidet sich von der von Mybatis und Idea oder dem offiziell anerkannten Java generierten Get-Set-Methode:

#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方法也是大写
    }
}复制代码
#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;
    }
}复制代码

Mybatis (Version 3.4.6) Analysieren Sie die Get-Set-Methode, um den Quellcode des Attributnamens zu erhalten:

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");
        }

}复制代码

Mybatis analysiert die Get-Set-Methode für den Attributnamentest

    @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复制代码

Solution

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

@Accessor(chain = true ) Anmerkungsproblem

Problemerkennung

Beim Exportieren mit easyexcel (github.com/alibaba/eas…) stellte ich fest, dass die vorherigen Entitätsklassenexporte normal waren, die neu hinzugefügten Entitätsklassen jedoch nicht normal sind. Es wurde festgestellt, dass die neu hinzugefügten Entitätsklassen nicht normal waren. Die Annotation @Accessor(chain = true) wurde der Entitätsklasse hinzugefügt. Unser Zweck besteht hauptsächlich darin, uns das Aufrufen der Set-Methode in einer Kette zu erleichtern:

new UserDto()
.setUserName("")
.setAge(10)
........
.setBirthday(new Date());复制代码

Reason

Die unterste Ebene von easyexcel verwendet cglib als Reflexions-Toolkit:

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

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

abstract public Object put(Object bean, Object key, Object value);复制代码

Aber cglib Die Methode der Introspector-Klasse in Javas rt.jar wird verwendet:

# 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);
    }
}复制代码

Lösung

1.去掉Accessor注解
2.要么就等待easyexcel的作者替换掉底层的cglib或者是其他,反正是支持获取返回值不是void的setxxx方法就行复制代码

Verwandte kostenlose Lernempfehlungen:

Java grundlegendes Tutorial

Das obige ist der detaillierte Inhalt vonLernen Sie Lomboks Fallstricke kennen. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Stellungnahme:
Dieser Artikel ist reproduziert unter:juejin.im. Bei Verstößen wenden Sie sich bitte an admin@php.cn löschen