浅谈java字符串比较到底应该用==还是equals

当用new String(“aaaa”)对字符串做定义时,==会返回false,equals可以返回正常结果。

比如

 System.out.println("a" == "a"); //true
 System.out.println("a".equals("a")); //true
 System.out.println("a".equals(new String("a"))); //true
 System.out.println("a" == new String("a")); //false
 System.out.println(new String("a") == new String("a")); //false
 System.out.println(new String("a").equals(new String("a"))); //true

以上是网上几乎全部文章所提到过的,也是它们仅仅提到的。但是,在实际项目中,并没有什么参考价值。

我从来没见有人用new String的方式定义过字符串,但是当它们在使用==做判断后,依然可能返回错误的结果。

原因就在于,你不用new String,也有人替你用。

在一般的开发中,字符串总是伴随着String类的方法使用的,这些方法中大量使用了new String。一旦使用了这些方法,==就会产生各种各样的bug。

 System.out.println("a".concat("a") == "aa"); //false
 System.out.println("aa".replaceAll("a", "b") == "bb"); //false
 System.out.println("A".toLowerCase() == "a"); //false
 System.out.println("1;2".split(";")[0] == "1"); //false

更要命的是,这些方法在特定情况下也是能够返回true的,具体原因可以通过阅读源码获知。

 System.out.println("a".concat("") == "a"); //ture
 //replaceAll由于用到了StringBuffer的toString方法,会一直返回false
 System.out.println("a".toLowerCase() == "a"); //ture
 System.out.println("1;2".split("3")[0] == "1;2"); //ture

这会导致一些问题:

  1. 测试时没bug,上线后突然出现一堆bug。
  2. 模块一直处于有时正确有时错误的状态。
  3. 做扩展的时候,在上层添加了一些String类的方法,导致下层出现了问题。
  4. …等等

无论哪种,都是很令人郁闷的bug。

你以为==很简单?等出了bug后你就不会这么想了!

当用equals时,需要注意的是调用者不可为空,否则会报空指针异常。一般可以通过常量调用equals的方式避免。若两边都是未知量,那就只能先判断一下是否为空了。虽然麻烦点,但总比出bug要好。

 String str1 = "a";
 String str2 = "a";
 System.out.println("a".equals(str1)); //正确,不会出现空指针
 System.out.println(str1.equals("a")); //不建议,若str1为空,会报空指针
 System.out.println(((str1 != null) && str1.equals(str2)) || (str1 == null && str2 == null));//正确,但是建议拆成两句写。你难道不觉得这么多逻辑判断很头疼么?

结论

就当==不存在,无脑使用equals。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • java求数组元素重复次数和java字符串比较大小示例

    复制代码 代码如下: /** * Name: 求数组中元素重复次数对多的数和重复次数 * Description:  * 数组中的元素可能会重复,这个方法可以找出重复次数最多的数,同时可以返回重复了多少次. * 但需要知道这个数组中最大的元素是多少,如果无法确定,就悲剧啦~ * * @param array目标数组: *           max数组中数据的最大值: * @return 返回一个包含重复次数最多的数(value)和重复次数(maxCount)的map集合: *         

  • Java字符串比较方法equals的空指针异常的解决

    在Java语言中字符串比较有两种方式:== 和equals(). "=="比较的是针对两个String类型变量的引用,当两个String类型的变量指向同一个String对象(即同一个内存堆),则返回true.而equals()方法是对String对象封装的字符串内容进行比较,相同返回true. 在用equals方法与其他值做比较的时候,有可能会导致抛出空指针异常.写一个小程序来举例说明: public class StringEqual { public static void equ

  • java字符串比较获取字符串出现次数的示例

    比如:javascriptjavasejavaeejavame 思路:定义一个计数器获取java第一次出现的位置从第一次出现位置后剩余的字符串中继续获取java出现的位置每获取一次就计数一次当获取不到时,计数完成 复制代码 代码如下: class StringCount{    public static void main(String[] args){        String s = "javascriptjavasejavaeejavame";        int coun

  • 浅谈java字符串比较到底应该用==还是equals

    当用new String("aaaa")对字符串做定义时,==会返回false,equals可以返回正常结果. 比如 System.out.println("a" == "a"); //true System.out.println("a".equals("a")); //true System.out.println("a".equals(new String("a"

  • 浅谈java 字符串,字符数组,list间的转化

    1.关于java.lang.string.split xxx.split()方法可以将一个字符串分割为子字符串,然后将结果作为字符串数组返回. 2.字符串转字符数组 String str =" aa.png,a2.png,a3.png"; String[] arrayStr =new String[]{}; arrayStr = str.split(","); 3.字符数组转list List list = java.util.Arrays.asList(array

  • 浅谈Java finally语句到底是在return之前还是之后执行(必看篇)

    网上有很多人探讨Java中异常捕获机制try...catch...finally块中的finally语句是不是一定会被执行?很多人都说不是,当然他们的回答是正确的,经过我试验,至少有两种情况下finally语句是不会被执行的: (1)try语句没有被执行到,如在try语句之前就返回了,这样finally语句就不会执行,这也说明了finally语句被执行的必要而非充分条件是:相应的try语句一定被执行到. (2)在try块中有System.exit(0);这样的语句,System.exit(0);

  • 浅谈JAVA字符串匹配算法indexOf函数的实现方法

    前言 相信每个学习过Java的人都使用过indexOf函数,indexOf函数我们可以查找一个字符串(模式串)是否在另一个字符串(主串)出现过,返回结果表示出现位置的下标,如果返回-1,表示模式串在主串中不存在,那么,你可曾想过这些查找函数又是如何实现的呢? 从indexOf源码看起 首先我们先来看一下indexOf的源码,indexOf的使用方式比较多,这是我们以一个形参的为例. static String mainString = "Hello my name is HuangLinqing

  • 浅谈java String.split丢失结尾空字符串的问题

    java中的split函数用于将字符串分割为字符数组是很方便的,但由于不是很熟悉,犯了错误 如下: String strtest = "1,2,"; String arry[] = strtest.split(","); 这样得到的数组元素个数只是2两个,为什么呢,最后一个","后没有内容,它没有作为空字符串成为第三个数组元素,结尾的空字符串被丢弃了! 这个函数还有另一种重载方式 :public String [] split (String 

  • 浅谈java中字符串数组、字符串、整形之间的转换

    字符串数组转字符串(只能通过for循环): String[] str = {"abc", "bcd", "def"}; StringBuffer sB = new StringBuffer(); for (int i = 0; i < str.length;i++) { sB.append(str[i]); } String s = sB.toString(); 字符数组转字符串可以通过下面的方式: char[] data = {"

  • 浅谈Java到底是值传递还是引用传递呢

    一.前言 最近在看Java核心卷一,也就是这本书: 在这本书里面也看到了这个问题,Java是值传递还是引用传递,这个问题其实也是很有意思的,之前也看到过这个问题,但是只是依稀记得是值传递,而且网上也有在讨论这个问题的.所以就先说结论吧:是值传递. 二.值传递与引用传递 既然讨论是值传递还是引用传递,那肯定是要知道啥是值传递.引用传递的. 值传递:是指在调用函数时将实际参数复制一份传递到函数中,这样在函数中如果对参数进行修改,将不会影响到实际参数. 引用传递:是指在调用函数时将实际参数的地址直接传

  • 浅谈Java多线程实现及同步互斥通讯

    Java多线程深入理解本文主要从三个方面了解和掌握多线程: 1. 多线程的实现方式,通过继承Thread类和通过实现Runnable接口的方式以及异同点. 2. 多线程的同步与互斥中synchronized的使用方法. 3. 多线程的通讯中的notify(),notifyAll(),及wait(),的使用方法,以及简单的生成者和消费者的代码实现. 下面来具体的讲解Java中的多线程: 一:多线程的实现方式 通过继承Threa类来实现多线程主要分为以下三步: 第一步:继承 Thread,实现Thr

  • 浅谈Java实现回溯算法之八皇后问题

    目录 一.前言 二.浅谈递归 三.回溯算法 四.八皇后问题 五.八皇后变种 六.总结 一.前言 说起八皇后问题,它是一道回溯算法类的经典问题,也可能是我们大部分人在上数据结构或者算法课上遇到过的最难的一道题-- 二.浅谈递归 对于递归算法,我觉得掌握递归是入门数据结构与算法的关键,因为后面学习很多操作涉及到递归,例如链表的一些操作.树的遍历和一些操作.图的dfs.快排.归并排序等等. 递归的实质还是借助栈实现一些操作,利用递归能够完成的操作使用栈都能够完成,并且利用栈的话可以很好的控制停止,效率

  • 浅谈Java中Unicode的编码和实现

    Unicode的编码和实现 大概来说,Unicode编码系统可分为编码方式和实现方式两个层次. 编码方式 字符是抽象的最小文本单位.它没有固定的形状(可能是一个字形),而且没有值."A"是一个字符,"€"也是一个字符.字符集是字符的集合.编码字符集是一个字符集,它为每一个字符分配一个唯一数字. Unicode 最初设计是作为一种固定宽度的 16 位字符编码.也就是每个字符占用2个字节.这样理论上一共最多可以表示216(即65536)个字符.上述16位统一码字符构成基

随机推荐