【Java面试宝典】常用类中的方法重写|equals方法与逻辑运算符==的区别

简介: 【Java面试宝典】常用类中的方法重写|equals方法与逻辑运算符==的区别

1、常用类(Object)概述

常用类是所有类的父类


一个类如果如果没有显式声明父类,则一定直接继承自Object

Object类中存放着所有类都必须拥有的内容,子类可以根据需要对继承的方法进行重写

2、五种常用方法与部分方法重写示例

1.getClass():获取引用的实际对象类型


2.int hashCode():获取对象的哈希码值


每个对象地址的哈希码值都不同

重写原因:

由于该方法默认关注对象地址,无法满足对象内容相同被认为是同一对象的需求

重写规则:

所有基本数据类型的属性值相加

这里会存在误差:

因为该方法返回值类型是整型,遇到浮点型属性则需进行类型强转抹去小数部分

如果存在String,则加上String类型属性名.hashCode()

如果存在自定义对象类型,则该类内部也需要重写hashCode()方法并调用相加:

public class Student extends Person{
    private String name;
  private int age;
  private double score;
  private Teacher tea;
  //省略getter、setter、构造
  public int hashCode(){
    return age+(int)score+name.hashCode()+tea.hashCode();
  }
}

3.boolean equals(Object o):判断两个对象内容是否相同


默认比较堆地址

重写原因:与hashCode一致

重写规则:

 public boolean equals(Object obj){
  if(this==obj){//自反性
      return true;//自己跟自己比,一定相同
  }
  if (obj==null){//非空判断
      return false;//跟null值比,一定不同
  }
  if (this.getClass()!=obj.getClass()){//类型比较
      return false;//实际类型不同,一定不同
  }
  当前类名 s=(当前类名) obj;//类型强转
  if (基本类型使用==比较,String类型调用equals方法比较,多个比较用&&连接){
      //属性值比较
      return true;
  }
  return  false;
}
//例如:
public boolean equals(Object obj){
  if(this==obj){
      return true;
  }
  if (obj==null){
      return false;
  }
  if (this.getClass()!=obj.getClass()){
      return false;
  }
  Student s=(Student) obj;
  if (this.age==s.getAge() && this.score==s.getScore() 
  && this.name.equals(s.getName())){
      return true;
  }
  return  false;
}

 

4.String toString():以字符串的形式返回对象内容


输出引用时会默认调用该方法,返回对象地址

重写原因:

实战开发中,需要经常查看对象信息,可以通过该方法直接返回对象属性值

重写规则:按照需求格式拼接返回属性值即可

public String toString(){
  return name+","+age+","+score;
}

5.void finalize():用来被虚拟机调用,进行垃圾对象回收


垃圾回收机制:

当内存已经满到不足以支撑新对象的创建时,虚拟机会调用垃圾对象的该方法对其进行回收销毁,从而释放一定空间

垃圾对象:

当一个对象没有任何引用指向时,虚拟机就会认定其为垃圾对象(零引用机制)

手动进行垃圾回收:

借助垃圾回收器GC,在程序中调用System.gc()方法手动完成回收

3、逻辑运算符==与equals的比较

3.1、==操作符特点

==是逻辑运算符,用来判断两个变量是否相等:


对于基本数据类型:比较的是他们的值是否相同

对于引用数据类型:比较的是他们的堆地址是否相同

引用类型对象变量其实是一个存放在栈中的引用,它们的值是指向对象所在的堆地址,堆地址中存放对象本身的数据。


引用类型内存简易图示:


0a75018e5ff5451cabe7008121e73c7b.png


stu是栈中的引用,指向堆中的一个地址,在此例中该堆地址中的数据就是名字:张三,年龄:18


注意:如果是两个不同名字的字符串进行==判断,结果也是true,这是因为第一个字符串创建后会放入常量池中,当第二个字符串相同时不会再次创建字符串对象,而是指向常量池中的堆地址。


3.2、二者的异同点

未重写equals方法的类:

Object中的equals方法实际使用的也是==操作符,比较的是他们的内存地址是否相同。

重写了equals方法的类:

从上面对equals方法重写的示例来看,重写后比较的是二者的内容,比如:

String :比较字符串具体的内容

Integer:相当于比较包装类对应的基本数据类型的值是否相同

String、Date、File、包装类都重写了Object类的equals方法

equals方法与==操作符的比较是个经典面试问题了,重写equals的方法我也已经举例提出,希望能够帮助到大家。


目录
相关文章
|
8天前
|
人工智能 安全 Java
Java并发包下Atomic相关类的使用
本文介绍了 `java.util.concurrent.atomic` 包下的各类原子类及其使用场景,包括基本类型原子类(如 `AtomicInteger`、`AtomicLong`)、数组类型原子类(如 `AtomicIntegerArray`)、引用类型原子类(如 `AtomicReference`)、对象属性修改原子类(如 `AtomicIntegerFieldUpdater`)以及原子操作增强类(如 `LongAdder` 和 `LongAccumulator`)。同时,详细对比了不同原子类在高并发场景下的性能表现,展示了 `LongAdder` 的高效性。
66 31
|
1月前
|
Java 数据安全/隐私保护
Java 类和对象
本文介绍了Java编程中类和对象的基础知识,作为面向对象编程(OOP)的核心概念。类是对象的蓝图,定义实体类型;对象是具体实例,包含状态和行为。通过示例展示了如何创建表示汽车的类及其实例,并说明了构造函数、字段和方法的作用。同时,文章还探讨了访问修饰符的使用,强调封装的重要性,如通过getter和setter控制字段访问。最后总结了类与对象的关系及其在Java中的应用,并建议进一步学习继承等概念。
|
1月前
|
Java
java中一个接口A,以及一个实现它的类B,一个A类型的引用对象作为一个方法的参数,这个参数的类型可以是B的类型吗?
本文探讨了面向对象编程中接口与实现类的关系,以及里氏替换原则(LSP)的应用。通过示例代码展示了如何利用多态性将实现类的对象传递给接口类型的参数,满足LSP的要求。LSP确保子类能无缝替换父类或接口,不改变程序行为。接口定义了行为规范,实现类遵循此规范,从而保证了多态性和代码的可维护性。总结来说,接口与实现类的关系天然符合LSP,体现了多态性的核心思想。
38 0
|
2月前
|
Java
java常见的集合类有哪些
Map接口和Collection接口是所有集合框架的父接口: 1. Collection接口的子接口包括:Set接口和List接口 2. Map接口的实现类主要有:HashMap、TreeMap、Hashtable、ConcurrentHashMap以及 Properties等 3. Set接口的实现类主要有:HashSet、TreeSet、LinkedHashSet等 4. List接口的实现类主要有:ArrayList、LinkedList、Stack以及Vector等
|
3月前
|
存储 监控 Java
【Java并发】【线程池】带你从0-1入门线程池
欢迎来到我的技术博客!我是一名热爱编程的开发者,梦想是编写高端CRUD应用。2025年我正在沉淀中,博客更新速度加快,期待与你一起成长。 线程池是一种复用线程资源的机制,通过预先创建一定数量的线程并管理其生命周期,避免频繁创建/销毁线程带来的性能开销。它解决了线程创建成本高、资源耗尽风险、响应速度慢和任务执行缺乏管理等问题。
241 60
【Java并发】【线程池】带你从0-1入门线程池
|
14天前
|
Java
java 多线程异常处理
本文介绍了Java中ThreadGroup的异常处理机制,重点讲解UncaughtExceptionHandler的使用。通过示例代码展示了当线程的run()方法抛出未捕获异常时,JVM如何依次查找并调用线程的异常处理器、线程组的uncaughtException方法或默认异常处理器。文章还提供了具体代码和输出结果,帮助理解不同处理器的优先级与执行逻辑。
|
1月前
|
Java 中间件 调度
【源码】【Java并发】从InheritableThreadLocal和TTL源码的角度来看父子线程传递
本文涉及InheritableThreadLocal和TTL,从源码的角度,分别分析它们是怎么实现父子线程传递的。建议先了解ThreadLocal。
72 4
【源码】【Java并发】从InheritableThreadLocal和TTL源码的角度来看父子线程传递
|
2月前
|
存储 网络协议 安全
Java网络编程,多线程,IO流综合小项目一一ChatBoxes
**项目介绍**:本项目实现了一个基于TCP协议的C/S架构控制台聊天室,支持局域网内多客户端同时聊天。用户需注册并登录,用户名唯一,密码格式为字母开头加纯数字。登录后可实时聊天,服务端负责验证用户信息并转发消息。 **项目亮点**: - **C/S架构**:客户端与服务端通过TCP连接通信。 - **多线程**:采用多线程处理多个客户端的并发请求,确保实时交互。 - **IO流**:使用BufferedReader和BufferedWriter进行数据传输,确保高效稳定的通信。 - **线程安全**:通过同步代码块和锁机制保证共享数据的安全性。
117 23
|
1月前
|
数据采集 存储 网络协议
Java HttpClient 多线程爬虫优化方案
Java HttpClient 多线程爬虫优化方案
|
2月前
|
Java 调度
【源码】【Java并发】【线程池】邀请您从0-1阅读ThreadPoolExecutor源码
当我们创建一个`ThreadPoolExecutor`的时候,你是否会好奇🤔,它到底发生了什么?比如:我传的拒绝策略、线程工厂是啥时候被使用的? 核心线程数是个啥?最大线程数和它又有什么关系?线程池,它是怎么调度,我们传入的线程?...不要着急,小手手点上关注、点赞、收藏。主播马上从源码的角度带你们探索神秘线程池的世界...
185 0
【源码】【Java并发】【线程池】邀请您从0-1阅读ThreadPoolExecutor源码

热门文章

最新文章