返回顶部

[小程序] Java判等问题:细节决定成败

[复制链接]
超级玛丽 显示全部楼层 发表于 2020-11-30 21:07:34 |阅读模式 打印 上一主题 下一主题
本帖最后由 超级玛丽 于 2020-11-30 21:09 编辑

判等问题,在我们代码中就是一句话的事情,但是这一行代码如果处理不好,不仅会出现致命的bug,下面我们就以Java中 equals、compareTo 和 Java 的数值缓存、字符串驻留等问题展开讨论



1. 注意 equals 和 == 的区别

在业务代码中,我们通常使用 equals 或 == 进行判等操作。equals是方法而 ==是操作符
: 1.对基本类型,比如 int 、long、进行判断,只能使用 == ,比较对是直接值,因为基本类型对值就是其数值
: 2.对引用类型,比如Integer 、Long 和 String 进行判等,需要使用 equals 进行内容判等。因为引用类型,需要使用equals进行内容判等。因为饮用类型等直接值是指针,使用 == 的话,比较的是指针,也就是两个对象在内存中的地址,即比较他们是不是同一个对象,而不是比较对象内容

结论:
  • 比较值的内容,除了基本类型只能使用 ==外,其他类型都需要使用 equals。
案例:
  1. public static void main(String[] args) throws Exception {
  2.         Integer a = 127;
  3.         Integer b = 127;
  4.         System.out.println(" a == b " +(a == b));

  5.         Integer c = 128;
  6.         Integer d = 128;
  7.         System.out.println(" c == d " + (c == d));

  8.         Integer g = new Integer(127);
  9.         Integer h = new Integer(127);
  10.         System.out.println(" g == h " + (g == h));

  11.         Integer i = 128;
  12.         int j = 128;
  13.         System.out.println(" i == j " + (i == j));
  14.     }
复制代码


结果 :
  1. a == b true
  2. c == d false
  3. g == h false
  4. i == j true
复制代码


在 a == b 中,编译器会把 a = 127 转换为 Integer.valueOf(127),源码可以发现,这个转换是内部其实做了缓存,使得两个 Integer 指向同一个对象 所以返回true
  1. public static Integer valueOf(int i) {
  2.         if (i >= IntegerCache.low && i <= IntegerCache.high)
  3.             return IntegerCache.cache[i + (-IntegerCache.low)];
  4.         return new Integer(i);
  5.     }
复制代码


在 c == d 中使用128 返回false ,Integer 当不符合-128 127值范围时候。记住用的:new,开辟新的内存空间,不属于IntergerCache管理区


  1. private static class IntegerCache {
  2.         static final int low = -128;
  3.         static final int high;
  4.         static final Integer cache[];

  5.         static {
  6.             // high value may be configured by property
  7.             int h = 127;
  8.             String integerCacheHighPropValue =
  9.                 sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
  10.             if (integerCacheHighPropValue != null) {
  11.                 try {
  12.                     int i = parseInt(integerCacheHighPropValue);
  13.                     i = Math.max(i, 127);
  14.                     // Maximum array size is Integer.MAX_VALUE
  15.                     h = Math.min(i, Integer.MAX_VALUE - (-low) -1);
  16.                 } catch( NumberFormatException nfe) {
  17.                     // If the property cannot be parsed into an int, ignore it.
  18.                 }
  19.             }
  20.             high = h;

  21.             cache = new Integer[(high - low) + 1];
  22.             int j = low;
  23.             for(int k = 0; k < cache.length; k++)
  24.                 cache[k] = new Integer(j++);

  25.             // range [-128, 127] must be interned (JLS7 5.1.7)
  26.             assert IntegerCache.high >= 127;
  27.         }
复制代码


  • 在g == h案例中,New 出来的 Integer 始终是不走缓存的新对象。比较两个新对象,
    或者比较一个新对象和一个来自缓存的对象,结果肯定不是相同的对象,因此返回 false。
2. equals 没有这么简单

如果看过 Object 类源码,你可能就知道,equals 的实现其实是比较对象引用:


  1. public boolean equals(Object obj) {
  2.         return (this == obj);
  3.     }
复制代码

重点(注意点):
  • 不重写equals方法与“ == ”一样,用于比较对象的引用是否相等。

之所以 Integer 或 String 能通过 equals 实现内容判等,是因为它们都重写了这个方法。

String 的 equals 的实现:


  1. /**
  2.      * Compares this string to the specified object.  The result is {@code
  3.      * true} if and only if the argument is not {@code null} and is a {@code
  4.      * String} object that represents the same sequence of characters as this
  5.      * object.
  6.      *
  7.      * @param  anObject
  8.      *         The object to compare this {@code String} against
  9.      *
  10.      * @return  {@code true} if the given object represents a {@code String}
  11.      *          equivalent to this string, {@code false} otherwise
  12.      *
  13.      * @see  #compareTo(String)
  14.      * @see  #equalsIgnoreCase(String)
  15.      */
  16.     public boolean equals(Object anObject) {
  17.         if (this == anObject) {
  18.             return true;
  19.         }
  20.         if (anObject instanceof String) {
  21.             String anotherString = (String)anObject;
  22.             int n = value.length;
  23.             if (n == anotherString.value.length) {
  24.                 char v1[] = value;
  25.                 char v2[] = anotherString.value;
  26.                 int i = 0;
  27.                 while (n-- != 0) {
  28.                     if (v1[i] != v2[i])
  29.                         return false;
  30.                     i++;
  31.                 }
  32.                 return true;
  33.             }
  34.         }
  35.         return false;
  36.     }
复制代码


本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

星点互联 成立于2014年8月,是目前国内优秀的开源技术社区,拥有超过300万会员,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作