Maison > Java > javaDidacticiel > Analyse d'exemples de pièges d'Objects.equals en Java

Analyse d'exemples de pièges d'Objects.equals en Java

WBOY
Libérer: 2023-05-04 13:28:06
avant
819 Les gens l'ont consulté

    1. Scène de crime

    Supposons qu'il y ait un tel besoin : déterminez l'utilisateur actuellement connecté, si c'est quoi nous spécifions l'administrateur système, envoyons un e-mail. L'administrateur système n'a pas d'identification de champ spécial. Son ID utilisateur est 888. Cette valeur est la même dans les environnements de développement, de test et de production.

    Cette exigence est vraiment simple à mettre en œuvre :

    UserInfo userInfo = CurrentUser.getUserInfo();
    
    if(Objects.isNull(userInfo)) {
       log.info("请先登录");
       return;
    }
    
    if(Objects.equals(userInfo.getId(),888)) {
       sendEmail(userInfo):
    }
    Copier après la connexion

    Obtenez les informations utilisateur à partir du contexte de l'utilisateur actuellement connecté, jugez-les et renvoyez-les directement si les informations utilisateur est vide.

    Si les informations utilisateur obtenues ne sont pas vides, alors déterminez si l'ID utilisateur est égal à 888.

    • Si égal à 888, envoyez l'e-mail.

    • S'il n'est pas égal à 888, rien ne sera fait.

    Après nous être connectés avec le compte d'administrateur système avec l'ID=888, nous avons effectué les opérations pertinentes et nous sommes préparés à recevoir l'e-mail avec de grandes attentes, mais avons constaté que l'e-mail était solitaire. .

    Plus tard, j'ai découvert que la classe UserInfo est définie comme ceci :

    @Data
    public class UserInfo {
        private Long id;
        private String name;
        private Integer age;
        private String address;
    }
    Copier après la connexion

    À ce moment-là, certains amis peuvent dire : je ne vois aucun problème.

    Mais ce que je veux dire c'est qu'il y a effectivement un problème avec ce code.

    C'est quoi le problème ?

    Réponse : La variable membre id=888 de la classe UserInfo est de type Long, et le 888 sur le côté droit de la méthode Objects.equals est de type int Les deux sont incohérents, ce qui rend le résultat renvoyé faux. Long类型的,而Objects.equals方法右边的888是int类型的,两者不一致,导致返回的结果是false。

    这算哪门子原因?

    答:各位看官,别急,后面会细讲的。

    2. 判断相等的方法

    让我们一起回顾一下,以前判断两个值是否相等的方法有哪些。

    2.1 使用==号

    之前判断两个值是否相等,最快的方法是使用==号。

    int a = 1;
    int b = 1;
    byte c = 1;
    Integer d1 = new Integer(1);
    Integer d2 = new Integer(1);
    System.out.println(a == b); 
    //结果:true
    System.out.println(a == c); 
    //结果:true
    System.out.println(a == d1); 
    //结果:true
    System.out.println(d2 == a); 
    //结果:true
    System.out.println(d1 == d2); 
    //结果:false
    Copier après la connexion

    不知道大家有没有发现,java中的基本类型,包含:int、long、short、byte、char、boolean、float、double这8种,可以使用号判断值是否相等。如果出现了基本类型的包装类,比如:Integer,用一个基本类型和一个包装类,使用号也能正确判断,返回true。

    Integer和int比较时,会自动拆箱,这是比较值是否相等。

    但如果有两个包装类,比如:d1和d2,使用==号判断的结果可能是false。

    两个Integer比较时,比较的是它们指向的引用(即内存地址)是否相等。

    还有一个有意思的现象:

    Integer d3 = 1;
    Integer d4 = 1;
    Integer d5 = 128;
    Integer d6 = 128;
    System.out.println(d3 == d4); 
    //结果:true
    System.out.println(d5 == d6); 
    //结果:false
    Copier après la connexion

    都是给Integer类型的参数,直接赋值后进行比较。d3和d4判断的结果相等,但d5和d6判断的结果却不相等。

    小伙伴们,下巴惊掉了没?

    答:因为Integer有一个常量池,-128~127直接的Integer数据直接缓存进入常量池。所以1在常量池,而128不在。

    然而,new的Integer对象不适用常量池。从之前d1和d2例子的比较结果,就能看出这一点。

    接下来,看看字符串的判断:

    String e = "abc";
    String f = "abc";
    String g = new String("abc");
    String h = new String("abc");
    System.out.println(e == f); 
    //结果:true
    System.out.println(e == g); 
    //结果:false
    System.out.println(g == h); 
    //结果:false
    Copier après la connexion

    普通的字符串变量,使用==号判断,也能返回正确的结果。

    但如果一个普通的字符串变量,跟new出来的字符串对象使用号判断时,返回false。这一点,跟之前说过的用一个基本类型和一个包装类,使用号判断的结果有区别,字符串没有自动拆箱的功能,这一点需要特别注意。

    此外,两个new出来的字符串对象使用==号判断时,也返回false。

    2.2 使用equals方法

    使用上面的==号,可以非常快速判断8种基本数据类型是否相等,除此之外,还能判断两个对象的引用是否相等。

    但现在有个问题,它无法判断两个对象在内存中具体的数据值是否相等,比如:

    String g = new String("abc");
    String h = new String("abc");
    System.out.println(g == h); 
    //结果:false
    Copier après la connexion

    字符串对象g和h是两个不同的对象,它们使用==号判断引用是否相等时,返回的是false。

    那么,这种对象不同,但数据值相同的情况,我们该如何判断相等呢?

    答:使用equals

    De quel genre de raison s'agit-il ?

    Réponse : Chers lecteurs, ne vous inquiétez pas, je vous l'expliquerai en détail plus tard.

    2. Méthodes pour juger de l'égalité

    Revoyons ensemble quelles méthodes ont été utilisées pour juger si deux valeurs sont égales.

    2.1 Utilisez le signe ==

    Le moyen le plus rapide de déterminer si deux valeurs sont égales est d'utiliser le signe ==.

    public boolean equals(Object obj) {
        return (this == obj);
    }
    Copier après la connexion

    Je ne sais pas si vous avez remarqué que les types de base en Java incluent : int, long, short, byte, char, boolean, float, double Vous pouvez utiliser le nombre pour déterminer si. les valeurs sont égales. Si un type de base de classe wrapper apparaît, tel qu'Integer, en utilisant un type de base et une classe wrapper, le numéro d'utilisation peut être correctement jugé et true sera renvoyé.

    Lors de la comparaison d'Integer et d'int, ils seront automatiquement déballés. Il s'agit de comparer si les valeurs sont égales.

    Mais s'il existe deux classes d'emballage, telles que : d1 et d2, le résultat de l'utilisation du signe == peut être faux.

    Lors de la comparaison de deux entiers, ce qu'ils comparent est de savoir si les références vers lesquelles ils pointent (c'est-à-dire les adresses mémoire) sont égales.

    Il existe un autre phénomène intéressant :

    public boolean equals(Object anObject) {
        if (this == anObject) {
            return true;
        }
        if (anObject instanceof String) {
            String anotherString = (String)anObject;
            int n = value.length;
            if (n == anotherString.value.length) {
                char v1[] = value;
                char v2[] = anotherString.value;
                int i = 0;
                while (n-- != 0) {
                    if (v1[i] != v2[i])
                        return false;
                    i++;
                }
                return true;
            }
        }
        return false;
    }
    Copier après la connexion

    sont tous des paramètres de type Integer, qui sont directement attribués et comparés. Les résultats du jugement de d3 et d4 sont égaux, mais les résultats du jugement de d5 et d6 ne sont pas égaux. #🎜🎜##🎜🎜# Les gars, vous êtes bouche bée ? #🎜🎜##🎜🎜# Réponse : Étant donné que Integer a un pool constant, -128 ~ 127 données Integer directes sont directement mises en cache dans le pool constant. Donc 1 est dans le pool constant, mais 128 ne l’est pas. #🎜🎜##🎜🎜#Cependant, les nouveaux objets Integer ne conviennent pas aux pools constants. Cela peut être vu à partir des résultats de comparaison des exemples d1 et d2 précédents. #🎜🎜##🎜🎜#Ensuite, regardons le jugement de chaîne : #🎜🎜#
    String e = "abc";
    String f = "abc";
    System.out.println(e.equals(f)); 
    //结果:true
    Copier après la connexion
    #🎜🎜# Les variables de chaîne ordinaires peuvent également renvoyer le résultat correct en utilisant le signe ==. #🎜🎜##🎜🎜#Mais si une variable chaîne ordinaire est utilisée pour juger l'objet chaîne produit par new, false sera renvoyé. Ce point est différent de ce que j'ai dit précédemment sur l'utilisation d'un type de base et d'une classe d'empaquetage, et le résultat de l'utilisation du jugement numérique Les chaînes n'ont pas la fonction de déballage automatique, ce qui nécessite une attention particulière. #🎜🎜##🎜🎜#De plus, lorsque les deux nouveaux objets chaîne sont jugés à l'aide du signe ==, false est également renvoyé. #🎜🎜#

    2.2 Utilisez la méthode égale

    #🎜🎜#En utilisant le signe == ci-dessus, vous pouvez déterminer rapidement si les huit types de données de base sont égaux. De plus, vous pouvez également déterminer l'égalité de. deux objets sont égaux. #🎜🎜##🎜🎜#Mais maintenant, il y a un problème. Il ne peut pas déterminer si les valeurs de données spécifiques des deux objets en mémoire sont égales. Par exemple : #🎜🎜#
    int a = 1;
    Integer b = new Integer(1);
    Integer c = null;
    System.out.println(a == b); 
    //结果:true
    System.out.println(a == c); 
    //结果:NullPointerException
    Copier après la connexion
    Copier après la connexion
    #🎜🎜#Le. les objets chaîne g et h sont deux objets différents, lorsqu'ils utilisent le signe == pour déterminer si les références sont égales, false est renvoyé. #🎜🎜##🎜🎜#Alors, comment juger de l'égalité lorsque les objets sont différents mais que les valeurs des données sont les mêmes ? #🎜🎜##🎜🎜# Réponse : Utilisez la méthode equals. #🎜🎜##🎜🎜#La méthode equals est en fait une méthode de la classe Object : #🎜🎜#
    String e = null;
    String f = "abc";
    System.out.println(e.equals(f)); 
    //结果:NullPointerException
    Copier après la connexion
    Copier après la connexion
    #🎜🎜#Cette méthode est très simple et détermine uniquement si les références de deux objets sont égales. #🎜🎜##🎜🎜#Évidemment, si le type chaîne utilise directement la méthode égale de la classe parent (c'est-à-dire la classe Objet) pour juger de la situation où les objets sont différents mais les valeurs sont les mêmes, il y a un problème . #🎜🎜##🎜🎜#Ainsi, la chaîne (c'est-à-dire la classe String) réimplémentera la méthode égale : #🎜🎜#
    String e = null;
    String f = "abc";
    System.out.println(equals(e, f));
    Copier après la connexion
    Copier après la connexion
    #🎜🎜#Elle déterminera toujours d'abord si les deux références d'objet sont égales et retournera vrai s'ils sont égaux. Ensuite, les deux chaînes seront comparées caractère par caractère et true ne sera renvoyé que si tous les caractères sont égaux. #🎜🎜##🎜🎜#bien, cela peut résoudre le problème du jugement de g et h : #🎜🎜#
    private static boolean equals(String e, String f) {
        if (e == null) {
            return f == null;
        }
        return e.equals(f);
    }
    Copier après la connexion
    Copier après la connexion
    #🎜🎜#On peut voir que nous utilisons la méthode égale réécrite par la classe String pour juger deux chaînes Quand les objets sont différents mais les valeurs sont les mêmes, true sera renvoyé. #🎜🎜##🎜🎜#3. Exception de pointeur nul #🎜🎜##🎜🎜# Nous savons déjà du point précédent que pour déterminer si deux objets sont égaux, vous pouvez utiliser le signe == ou la méthode égale. #🎜🎜##🎜🎜#Mais si vous les utilisez plus profondément, vous rencontrerez un problème, c'est-à-dire que si ces deux méthodes déterminent que les valeurs sont égales, une exception de pointeur nul peut être signalée. #🎜🎜#

    先看看==号判断的情况:

    int a = 1;
    Integer b = new Integer(1);
    Integer c = null;
    System.out.println(a == b); 
    //结果:true
    System.out.println(a == c); 
    //结果:NullPointerException
    Copier après la connexion
    Copier après la connexion

    int和Integer使用==号判断是否相等时,Integer会自动拆箱成int。

    但由于c在自动拆箱的过程中,需要给它赋值int的默认值0。而给空对象,赋值0,必然会报空指针异常。

    接下来,看看equals方法:

    String e = null;
    String f = "abc";
    System.out.println(e.equals(f)); 
    //结果:NullPointerException
    Copier après la connexion
    Copier après la connexion

    由于字符串对象e是空对象,直接调用它的equals方法时,就会报空指针异常。

    那么,如何解决空指针问题呢?

    答:在代码中判空。

    String e = null;
    String f = "abc";
    System.out.println(equals(e, f));
    Copier après la connexion
    Copier après la connexion

    我们抽取了一个新的equals方法:

    private static boolean equals(String e, String f) {
        if (e == null) {
            return f == null;
        }
        return e.equals(f);
    }
    Copier après la connexion
    Copier après la connexion

    该方法可以解决空指针问题,但有没有办法封装一下,变得更通用一下,也适用于Integer或者其他类型的对象比较呢?

    答:有办法,继续往下看。

    4. Objects.equals的作用

    Objects类位于java.util包下,它是里面提供了很多对象操作的辅助方法。

    下面我们重点看看它的equals方法:

    public static boolean equals(Object a, Object b) {
        return (a == b) || (a != null && a.equals(b));
    }
    Copier après la connexion

    equals方法的判断逻辑如下:

    • 该方法先判断对象a和b的引用是否相等,如果相等则直接返回true。

    • 如果引用不相等,则判断a是否为空,如果a为空则返回false。

    • 如果a不为空,调用对象的equals方法进一步判断值是否相等。

    该方法是如何使用的?

    int a = 1;
    int b = 1;
    Integer c = null;
    
    System.out.println(Objects.equals(a, c)); 
    //结果:false
    System.out.println(Objects.equals(c, a)); 
    //结果:false
    System.out.println(Objects.equals(a, b)); 
    //结果:true
    Copier après la connexion

    从上面的列子看出,使用Objects.equals方法比较两个对象是否相等,确实可以避免空指针问题。

    但这个有个疑问:前面使用a==b这种方式比较引用是否相等,当时b为空时,程序直接抛了空指针异常。

    而Objects.equals方法内部也使用了a==b比较引用是否相等,为啥它没有抛异常?

    答:因为而Objects类的equals方法,使用了Object类型接收参数,它的默认值是null,不用进行类型转换,也不用像int类型对象赋值默认值0。

    从上面的理论可以看出,如果我们把代码改成这样,也不会抛异常:

    int a = 1;
    Integer c = null;
    System.out.println(equals(a, c));
    //结果:false
    Copier après la connexion

    新定义了一个方法:

    private static boolean equals(Object a, Object b) {
        return a == b;
    }
    Copier après la connexion

    执行之后发现,确实没有抛空指针了。

    所以Objects.equals方法再比较两个对象是否相等时,确实是一个不错的方法。

    但它有坑,不信继续往下看。

    5. Objects.equals的坑

    各位小伙们看到这里,可能有点心急了,到底是什么坑?

    废话不多说,直接上例子:

    Integer a = 1;
    long b = 1L;
    System.out.println(Objects.equals(a, b));
    //结果:false
    Copier après la connexion

    什么?返回结果是false?

    而如果你直接用==号判断:

    Integer a = 1;
    long b = 1L;
    System.out.println(a == b);
    //结果:true
    Copier après la connexion

    返回又是true。

    a和b明明都是1,为什么使用Objects.equals方法判断不相等呢?

    这就要从Integer的equals方法说起来了。

    它的equals方法具体代码如下:

    public boolean equals(Object obj) {
        if (obj instanceof Integer) {
            return value == ((Integer)obj).intValue();
        }
        return false;
    }
    Copier après la connexion

    先判断参数obj是否是Integer类型,如果不是,则直接返回false。如果是Integer类型,再进一步判断int值是否相等。

    而上面这个例子中b是long类型,所以Integer的equals方法直接返回了false。

    也就是说,如果调用了Integer的equals方法,必须要求入参也是Integer类型,否则该方法会直接返回false。

    原来坑在这里!!!

    其实,如果把代码改成这样:

    Integer a = 1;
    long b = 1L;
    System.out.println(Objects.equals(b, a));
    //结果:false
    Copier après la connexion

    执行结果也是false。

    因为Long的equals方法代码,跟之前Integer的类似:

    public boolean equals(Object obj) {
        if (obj instanceof Long) {
            return value == ((Long)obj).longValue();
        }
        return false;
    }
    Copier après la connexion

    也是判断入参,如果不是Long类型,则该方法直接返回false。

    除此之外,还有Byte、Short、Double、Float、Boolean和Character也有类似的equals方法判断逻辑。

    由此可见,我们在使用Objects.equals方法,判断两个值是否相等时,一定要保证两个入参的类型要一致。否则即使两个值相同,但其结果仍然会返回false,这是一个大坑。

    那么,如何解决上面的问题呢?

    可以将参数b的类型强制转换成int。

    Integer a = 1;
    long b = 1L;
    System.out.println(Objects.equals(a, (int)b));
    //结果:true
    Copier après la connexion

    或者将参数a的类型强制转换成long。

    Integer a = 1;
    long b = 1L;
    System.out.println(Objects.equals(b, (long)a));
    //结果:true
    Copier après la connexion

    有些情况也可以直接用==号判断:

    Integer a = 1;
    long b = 1L;
    System.out.println(a==b);
    //结果:true
    Copier après la connexion

    除了Objects.equals方法在两个入参类型不同,而会直接返回false之外,java的8种基本类型包装类的equals也会有相同的问题,需要小伙们特别注意。

    之前,如果直接使用java基本类型包装类的equals方法判断相等。

    Integer a = new Integer(1);
    long b = 1L;
    System.out.println(a.equals(b));
    Copier après la connexion

    在idea中,如果你将鼠标放在equals方法上,会出现下面的提示:

    Analyse dexemples de pièges dObjects.equals en Java

    这时你就知道方法用错了,赶紧修正。但如果直接用包装类的equals方法,有个问题就是可能存在报空指针异常的风险。

    如果你使用Objects.equals方法判断相等,在idea中就并没有错误提示。

    除此之外,我还测试了findBug、sonar等工具,Objects.equals方法两个参数类型不一致的问题,也没有标识出来。

    Les gars, jetez un coup d'œil rapide à votre code. Y a-t-il des pièges ?

    Les pièges courants incluent :

    • Comparaison entre le type Long et le type Integer, tel que : le scénario d'ID utilisateur.

    • Comparaison du type d'octet et du type entier, tel que : scénario de jugement d'état.

    • Comparaison entre le type Double et le type Entier, par exemple : scénario de jugement où le montant est de 0.

    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:yisu.com
    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