一、== 运算符

1.1 基本类型的 == 运算

对于基本数据类型来说,== 比较的是值

  1. public class Main {
  2. public static void main(String[] args) {
  3. int a = 10;
  4. int b = 20;
  5. int c = 10;
  6. System.out.println(a == b); // output: false
  7. System.out.println(a == c); // output: true
  8. }
  9. }

1.2 引用类型的 == 运算

对于引用数据类型来说,== 比较的是对象的内存地址。

1.2.1 普通对象的 == 运算

  1. public class Book {
  2. private String bookName;
  3. private String bookAuthor;
  4. public Book() {
  5. }
  6. public String getBookName() {
  7. return this.bookName;
  8. }
  9. public void setBookName(String bookName) {
  10. this.bookName = bookName;
  11. }
  12. public String getBookAuthor() {
  13. return this.bookAuthor;
  14. }
  15. public void setBookAuthor(String bookAuthor) {
  16. this.bookAuthor = bookAuthor;
  17. }
  18. }
  1. public class Main {
  2. public static void main(String[] args) {
  3. Book book1 = new Book();
  4. book1.setBookName("西游记");
  5. book1.setBookAuthor("吴承恩");
  6. Book book2 = new Book();
  7. book1.setBookName("西游记");
  8. book1.setBookAuthor("吴承恩");
  9. System.out.println(book1); // output: Book@16b98e56
  10. System.out.println(book2); // output: Book@7ef20235
  11. System.out.println(book1 == book2); // output: false
  12. }
  13. }

即使 book1 与 book2 对象的属性值相同,他们 == 的结果还是 false,就是因为内存地址不同。

1.2.2 String 对象的 == 运算

  • new 方式

    1. String a = new String("yunhu");
    2. String b = new String("yunhu");
    3. System.out.println(a == b); // output: false

    因为 a 和 b 都是 String 的对象引用,即使内容相同,但是指向的是不同的内存地址。

  • 直接赋值方式

    1. String aa = "yunhu";
    2. String bb = "yunhu";
    3. System.out.println(aa == bb); // output: true

    直接赋值,对象存储在常量池中。
    虚拟机会在常量池中寻找是否已经有与将要创建的值相同的对象,如果有直接赋值给当前的引用,如果没有,新建一个 String 对象。

    二、equals()

    equals() 只判断对象是否相等,没有重写 equals() 函数的话,判断的依据是对象的地址是否相同。

    2.1 没有重写 equals()

    ```java Book book1 = new Book(); book1.setBookName(“西游记”); book1.setBookAuthor(“吴承恩”);

Book book2 = new Book(); book2.setBookName(“西游记”); book2.setBookAuthor(“吴承恩”);

System.out.println(book1.equals(book2)); // output: false

  1. 判断对象地址,当然不同。
  2. <a name="pZ9Pl"></a>
  3. ### 2.2 重写 equals()
  4. 这边为了方便,没有重写 `hashCode`方法, 不过不影响理解。
  5. ```java
  6. public class Book {
  7. private String bookName;
  8. private String bookAuthor;
  9. public String getBookName() {
  10. return bookName;
  11. }
  12. public void setBookName(String bookName) {
  13. this.bookName = bookName;
  14. }
  15. public String getBookAuthor() {
  16. return bookAuthor;
  17. }
  18. public void setBookAuthor(String bookAuthor) {
  19. this.bookAuthor = bookAuthor;
  20. }
  21. @Override
  22. public boolean equals(Object obj) {
  23. if (this == obj) {
  24. return true;
  25. }
  26. // 判断 obj 对象是不是 Book 类的
  27. if (!(obj instanceof Book)) {
  28. return false;
  29. }
  30. Book book = (Book)obj;
  31. if(book == null) {
  32. return false;
  33. }
  34. if (this.bookName.equals(book.getBookName()) && this.bookAuthor.equals(book.getBookAuthor())) {
  35. // 属性值相同,认为是同一个对象
  36. return true;
  37. }else {
  38. return false;
  39. }
  40. }
  41. }
  1. Book book1 = new Book();
  2. book1.setBookName("西游记");
  3. book1.setBookAuthor("吴承恩");
  4. Book book2 = new Book();
  5. book2.setBookName("西游记");
  6. book2.setBookAuthor("吴承恩");
  7. System.out.println(book1.equals(book2)); // output: true

2.3 String 的 equals()

  1. String a = new String("yunhu");
  2. String b = new String("yunhu");
  3. System.out.println(a.equals(b)); // output: true

想了想,这不对啊,a、b 是对象引用,equals 比较的是地址,这两个地址不同啊,怎么会返回 true。
String 确实是对象,但是它特殊处理了,重写了 equal 函数,变成比较值,而不是内存地址。
String equals 方法的源码:

  1. public boolean equals(Object anObject) {
  2. if (this == anObject) {
  3. return true;
  4. }
  5. return (anObject instanceof String aString)
  6. && (!COMPACT_STRINGS || this.coder == aString.coder)
  7. && StringLatin1.equals(value, aString.value);
  8. }

再看下 StringLatin1 类的 equals 源码:

  1. @IntrinsicCandidate
  2. public static boolean equals(byte[] value, byte[] other) {
  3. if (value.length == other.length) {
  4. for (int i = 0; i < value.length; i++) {
  5. if (value[i] != other[i]) {
  6. return false;
  7. }
  8. }
  9. return true;
  10. }
  11. return false;
  12. }

循环比较值。