JavaSE阶段面试题(一)

news2025/4/5 18:48:09

目录

1.int a = 1, int b = 1, Integer c = 1, Integer d = 1;四个区别和联系,以及c和d是同一个吗?

2.为什么重写HashCode必须重写euqals,两者之间的关系?

3.创建对象的方式有哪些

4.重写和重载的区别

5.抽象类和接口的区别

6.String的不可变原则

7.浅拷贝和深拷贝


本专栏全是博主自己收集的面试题,仅可参考,不能相信面试官就出这种题目。

       

1.int a = 1, int b = 1, Integer c = 1, Integer d = 1;四个区别和联系,以及c和d是同一个吗?

   a 和 b 是基本数据类型 int 的 变量,它们直接存储在栈(stack)上,并且每个变量都有自己的存储空间。

        c 和 d 是 Integer 类型的 对象,它们存储在堆(heap)上,。多个 Integer 对象被创建并且它们的值相同,Java 可能会使它们引用相同的对象,从而节省内存。因此cd 可能会引用相同的 Integer 对象。所以可能是同一个

联系c 和 d 都是通过自动装箱从基本数据类型 int 转换而来的 Integer 对象。

区别:a、b是变量,存储在栈上;c和d是对象,是Integer类的实例,可以调用方法。

2.为什么重写HashCode必须重写euqals,两者之间的关系?

        解答问题之前,回忆一下,什么是HashCode和equals,及其作用

        HashCode是在Java中用于获取对象的唯一标识符的方法。它是根据对象的内容生成的一个整数值。对象的hashCode()方法被调用时,它返回的是对象内存地址的哈希码。哈希码可以用于在哈希表等数据结构中快速定位对象

注意点:

  1. 每一次运行对象内容即使一致,但hash值不一定一致
  2. 运行过程里,new的两个对象,内容相等 ≠ 哈希码相等
  3. 不同的对象可能会生成相同的哈希码,概率极低

那么为什么需要重写hashcode呢?

我也不太清楚,但是重写hashcode可以让两个内容相同的对象拥有同一个哈希码

例子:

public class Student {
    int age;
    String sex;
    String name;
    @Override
    public int hashCode() {
        final int prime = 31;
        int result = 1;
        result = prime * result + this.age;
        result = prime * result + (this.sex == null ? 0 : this.sex.hashCode());
        result = prime * result + (this.name == null ? 0 : this.name.hashCode());
        return result;
    }

        在回忆一下equals,在Java中,equals() 方法被设计用来比较对象的用于比较对象的内容或状态是否相等!

        因此,疑问出来了,既然equals()方法比较的是内容或状态,那么是否hashCode()值相等,两个对象通过 equals() 方法被认为是相等吗?

解析:

        在 Java 中,当我们重写 hashCode() 方法时,通常需要同时重写 equals() 方法,这是为了维护 hashCode()equals() 方法之间的协定,确保对象在集合(如哈希表)中正确地工作。

  1. hashCode() 值相等

    • 如果两个对象的 hashCode() 值相等,那么根据 Java 的约定,这是 equals() 方法返回 true 的必要条件。也就是说,如果两个对象通过 equals() 方法相等,它们的 hashCode() 值必须相等。
  2. hashCode() 值不相等

    • 如果两个对象的 hashCode() 值不相等,那么按照 Java 的规定,这两个对象不应该通过 equals() 方法返回 true。即使 equals() 方法返回 true,也会违反 hashCode() 方法的契约,这可能导致在使用哈希集合(如 HashMapHashSet 等)时出现不一致的行为,甚至会导致数据结构的错误操作。

总结:HashCode()方法,HashCode方法通过内容一致得出相同的哈希码

           equals()方法,通过内容判断对象是否相同。

概述:一个类有三个属性,原本只通过一个属性重写了hashcode()和equals()方法,但是我将hashcode()方法,通过2个属性得出该对象的哈希值,可equals()方法还是根据一个属性判断,这并不安全,也会出现bug。

3.创建对象的方式有哪些

一般有5种:

第一种:常见的使用构造方法new对象

第二种:使用工厂设计模型,根据你需要的,自动创建出来。

// 定义汽车类
class Car {
    private String model;
    private int year;

    public Car(String model, int year) {
        this.model = model;
        this.year = year;
    }

    // 省略 getter 和 setter 方法
}

// 定义汽车工厂类
class CarFactory {
    // 工厂方法,根据型号和年份创建汽车对象
    public static Car createCar(String model, int year) {
        //----- 判断条件 -----
        return new Car(model, year);
    }
}

// 在主程序中使用工厂方法创建对象
public class Main {
    public static void main(String[] args) {
        // 使用工厂方法创建汽车对象
        Car myCar = CarFactory.createCar("Toyota Camry", 2023);
    }
}

第三种:克隆

克隆又分两种,一种浅克隆、一种深克隆,使用接口Cloneable和方法clone()

区别在于一个类中如果含有其他类,那么浅克隆,只能克隆表面,包含的类无法克隆,反之,深克隆可以做到!

class Teacher  implements Cloneable{
    private String name;
    private  int age;
    private Student student;
}
public class Test1 {

    public static void Shallow_cloning(){
        Teacher teacher = new Teacher();
        teacher.setAge(19);
        teacher.setName("李华");
        Student student = new Student();
        student.setName("喜喜");
        teacher.setStudent(student);

        Teacher teacher1 = teacher.clone();
}

第四种:通过反射

        反射机制允许在运行时检查类的信息,并动态地创建类的对象,可以调用类的构造方法来实现对象的创建

Class<?> clazz = Class.forName("指定的被反射的类");
类名 obj = (类名) clazz.getDeclaredConstructor().newInstance();

第五种:反序列化

        通过反序列化可以从存储设备(如文件、数据库)中读取对象的字节流,并将其转换回对象

ObjectInputStream in = new ObjectInputStream(new FileInputStream("文件名"));
MyClass obj = (MyClass) in.readObject();

4.重写和重载的区别

        重写:在面向对象的继承中,子类可以通过重写父类的方法来实现自己的版本。重写指的是子类定义了一个与父类中相同名称和参数列表(签名)的方法,并且返回类型和抛出的异常类型也必须与父类中的方法一致。重写的方法需要加上 @Override 注解(可选),这样可以让编译器帮助检查是否正确地重写了父类的方法

        重载:重载是指在同一个类中,可以定义多个方法名相同但参数列表不同(包括参数类型、参数个数或参数顺序)的方法。重载的方法彼此之间的签名必须不同,返回类型可以不同,但通常情况下只有返回类型不同是不够的。

总结

  • 重写发生在子类继承父类的过程中,用于实现多态性,要求方法签名相同。
  • 重载发生在同一个类中,用于提供多个同名方法以应对不同的参数情况,要求方法签名不同。

5.抽象类和接口的区别

抽象类:不能被实例化,都是被继承使用,可以有抽象方法(没有具体实现的方法),也可有具体的实现方法,可以有构造方法,被子类调用。

接口:定义了一组没有具体实现的方法,一个类可以实现多个接口,

  • 接口中的方法默认是 public abstract 的,属性默认是 public static final 的(Java 8 之后接口中可以有默认方法和静态方法)。
  • 类通过 implements 关键字来实现接口,并提供接口中定义的所有方法的具体实现。

6.String的不可变原则

        在Java中,String 类被设计为不可变的。有人说,我一开始让String类型指向"abc",然后再指向"wer" 不是发生了改变吗?

        不不不,实际上是创建了一个新的字符串对象,而不是修改原来的对象。旧的字符串对象将被Java的垃圾回收机制清理掉。

为什么是不可变的呢?

 线程安全角度:字符串不可变性确保了字符串对象在多线程环境下是安全的,不需要额外的同步操作。因为字符串一旦创建,它的值不会改变,所以不会出现多个线程同时修改一个字符串对象的情况。

哈希值:字符串被广泛用作 HashMap 和 HashSet 的键。由于字符串不可变,可以安全地缓存它们的哈希值,提高了哈希表的性能。

安全敏感操作:在安全敏感的环境中,不可变字符串确保了关键信息(如密码)不会被意外修改。

或许有人说,如果需要频繁更改,岂不是很麻烦!

不不不,因为字符串池(String Pool)存在。字符串池是一种特殊的内存区域,用于存储字面量字符串,以便多个字符串引用可以共享相同的实例,从而节省内存。

7.浅拷贝和深拷贝

        之前我们在描述创建对象的时候,以及提及到了浅拷贝和深拷贝

拷贝:将一组数据原封不动的转移到另容器。

例如:字符串拷贝和数组拷贝

String original = "Hello";
String copied = new String(original);

String original = "Hello";
String copied = original.substring(0); // 从索引0开始,复制到字符串末尾

int[] original = {1, 2, 3, 4, 5};
int[] copied = original.clone();  //用数组的 clone() 方法

但是对象也是一组数据,因此也可以拷贝

浅拷贝:通过对象实现了 Cloneable 接口,重写clone()方法

class Teacher2  implements Cloneable{
    private String name;
    private  int age;
    @Override
    protected Teacher2 clone(){
        Teacher2 t1 = null;
        try {
            t1 = (Teacher2) super.clone();
            // 克隆引用类型
            t1.setStudent(t1.getStudent().clone());
        } catch (CloneNotSupportedException e) {
            throw new RuntimeException(e);
        }
        return t1;
    }
}
 public static void Shallow_cloning(){
        Teacher2 teacher = new Teacher();
        teacher.setAge(19);
        teacher.setName("李华");
        //克隆
        Teacher2 teacher1 = teacher.clone();
}

而所谓的深克隆就是针对一种类的有其他对象的存在,例如:

class Teacher  implements Cloneable{
    private String name;
    private  int age;
    private Student student;   //存在一个对象Student

}

这种情况下,即使克隆,其中的student还是原来的,只是共享了数据。

解决方法有三种:

  1. 所有引用属性都实现克隆,整个对象就变成了深克隆。
  2. 使用 JDK 自带的字节流序列化和反序列化对象实现深克隆。(略)
  3. 使用第三方工具实现深克隆,比如 Apache Commons Lang 库。(略)

引用属性都实现克隆方法:

public class CloneDemo {
    public static void main(String[] args) {
        Person p1 = new Person();
        p1.setName("张三");
        p1.setAge(18);
        // 引用类型
        Address address = new Address();
        address.setCity("北京");
        p1.setAddress(address);
        // 克隆 p1 对象
        Person p2 = p1.clone();
        // 对比引用类型的地址值是否相同
        System.out.println(p1.getAddress() == p2.getAddress()); // false
    }
}

@Getter
@Setter
class Person implements Cloneable {
    private String name;
    private int age;
    private Address address; // 引用类型

    @Override
    public Person clone() {
        Person person = null;
        try {
            person = (Person) super.clone();
            // 克隆引用类型
            person.setAddress(person.getAddress().clone());
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return person;
    }
}

@Getter
@Setter
class Address implements Cloneable {
    private String city;

    @Override
    public Address clone() {
        Address address = null;
        try {
            address = (Address) super.clone();
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return address;
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1890414.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Webpack: Dependency Graph 管理模块间依赖

概述 Dependency Graph 概念来自官网 Dependency Graph | webpack 一文&#xff0c;原文解释&#xff1a; Any time one file depends on another, webpack treats this as a dependency. This allows webpack to take non-code assets, such as images or web fonts, and als…

算法day1 两数之和 两数相加 冒泡排序 快速排序

两数之和 最简单的思维方式肯定是去凑两个数&#xff0c;两个数的和是目标值就ok。这里两遍for循环解决。 两数相加 敲了一晚上哈哈&#xff0c;结果超过int范围捏&#xff0c;难受捏。 public class Test2 {public static void main(String[] args) { // ListNode l1 …

像学Excel 一样学 Pandas系列-创建数据分析维度

嗨&#xff0c;小伙伴们。又到喜闻乐见的Python 数据分析王牌库 Pandas 的学习时间。按照数据分析处理过程&#xff0c;这次轮到了新增维度的部分了。 老样子&#xff0c;我们先来回忆一下&#xff0c;一个完整数据分析的过程&#xff0c;包含哪些部分内容。 其中&#xff0c…

四十篇:内存巨擘对决:Redis与Memcached的深度剖析与多维对比

内存巨擘对决&#xff1a;Redis与Memcached的深度剖析与多维对比 1. 引言 在现代的系统架构中&#xff0c;内存数据库已经成为了信息处理的核心技术之一。这类数据库系统的高效性主要来源于其对数据的即时访问能力&#xff0c;这是因为数据直接存储在RAM中&#xff0c;而非传统…

二叉树的前中后序遍历(递归法、迭代法)leetcode144、94/145

leetcode144、二叉树的前序遍历 给你二叉树的根节点 root &#xff0c;返回它节点值的 前序 遍历。 示例 1&#xff1a; 输入&#xff1a;root [1,null,2,3] 输出&#xff1a;[1,2,3] 示例 2&#xff1a; 输入&#xff1a;root [] 输出&#xff1a;[] 示例 3&#xff1a; 输…

前端入门超级攻略:你的第一步学习指南

如果您觉得这篇文章有帮助的话&#xff01;给个点赞和评论支持下吧&#xff0c;感谢~ 作者&#xff1a;前端小王hs 阿里云社区博客专家/清华大学出版社签约作者/csdn百万访问前端博主/B站千粉前端up主/知名前端开发者/网络工程师 前言 由于前端技术的快速迭代性&#xff0c;国…

解决ps暂存盘已满的问题

点击编辑->首选项->暂存盘 ps默认暂存盘使用的是c盘&#xff0c;我们改成d盘即可 然后重启ps

STM32之五:TIM定时器(2-通用定时器)

目录 通用定时器&#xff08;TIM2~5&#xff09;框图 1、 输入时钟源选择 2、 时基单元 3 、输入捕获&#xff1a;&#xff08;IC—Input Capture&#xff09; 3.1 输入捕获通道框图&#xff08;TI1为例&#xff09; 3.1.1 滤波器&#xff1a; 3.1.2 边沿检测器&#xf…

移动智能终端数据安全管理方案

随着信息技术的飞速发展&#xff0c;移动设备已成为企业日常运营不可或缺的工具。特别是随着智能手机和平板电脑等移动设备的普及&#xff0c;这些设备存储了大量的个人和敏感数据&#xff0c;如银行信息、电子邮件等。员工通过智能手机和平板电脑访问企业资源&#xff0c;提高…

【等保2.0是什么意思?等保2.0的基本要求有哪些? 】

一、等保2.0是什么意思&#xff1f; 等保2.0又称“网络安全等级保护2.0”体系&#xff0c;它是国家的一项基本国策和基本制度。在1.0版本的基础上&#xff0c;等级保护标准以主动防御为重点&#xff0c;由被动防守转向安全可信&#xff0c;动态感知&#xff0c;以及事前、事中…

SSM玉林师范学院宿舍管理系统-计算机毕业设计源码19633

摘要 随着大学生人数的增加&#xff0c;宿舍管理成为高校管理中的重要问题。本论文旨在研究玉林师范学院宿舍管理系统&#xff0c;探讨其优势和不足&#xff0c;并提出改进建议。通过对相关文献的综述和实地调研&#xff0c;我们发现该系统在宿舍分配、卫生评分、失物招领、设施…

什么是 URL ?

统一资源定位符&#xff08;URL&#xff09;是一个字符串&#xff0c;它指定了一个资源在互联网上的位置以及如何访问它。URL 是由几部分组成的&#xff0c;每部分都有其特定的作用&#xff1a; 协议/方案&#xff1a;这是 URL 的开头部分&#xff0c;表明了用于访问资源的协议…

基于uniapp(vue3)H5附件上传组件,可限制文件大小

代码&#xff1a; <template><view class"upload-file"><text>最多上传5份附件&#xff0c;需小于50M</text><view class"" click"selectFile">上传</view></view><view class"list" v…

WPF自定义模板--TreeView 实现菜单连接线

有些小伙伴说&#xff0c;在TreeView中&#xff0c;怎么每一个都加上连接线&#xff0c;进行显示连接。 代码和效果如下&#xff1a; 其实就是在原来的模板中增加一列显示线条&#xff0c;然后绘制即可 <Window x:Class"XH.TemplateLesson.TreeViewWindow"xmln…

无法定位程序输入点Z9 qt assertPKcS0i于动态链接库F:\code\projects\06_algorithm\main.exe

解决方法&#xff1a; 这个报错&#xff0c;是因为程序在运行时没要找到所需的dll库&#xff0c;如果把这个程序方法中对应库的目录下执行&#xff0c;则可正常执行。即使将图中mingw_64\bin 环境变量上移到msvc2022_64\bin 之前也不可以。 最终的解决方法是在makefile中设置环…

vue组件深入介绍之插槽

了解插槽之前请先了解vue组件基础及注册 Vue2官网介绍 Vue3官网介绍 1、vue2插槽介绍 在2.6.0中&#xff0c;具名插槽和作用域插槽引入了一个新的统一语法&#xff08;v-slot指令&#xff09;。它将取代slot和slot-scope&#xff1b; Vue 实现了一套内容分发的 API&#xf…

等保2.0 实施方案之信息软件验证要求

一、等保2.0背景及意义 随着信息技术的快速发展和网络安全威胁的不断演变&#xff0c;网络安全已成为国家安全、社会稳定和经济发展的重要保障。等保2.0&#xff08;即《信息安全技术 网络安全等级保护基本要求》2.0版本&#xff09;作为网络安全等级保护制度的最新标准&#x…

Revit 专业实用的BIM模型设计软件下载安装,Revit 最新版下载安装

Revit&#xff0c;该软件是专门为建筑信息模型&#xff08;BIM&#xff09;量身打造的&#xff0c;不仅极大提升了建筑设计师的工作效率&#xff0c;更为他们创造了一个更加精确、高效的设计环境。 在Revit的助力下&#xff0c;建筑设计师们能够轻松地进行建筑建模&#xff0c…

AI:开发者的助力还是终结者?

作为一名科技工作研发者&#xff0c;在科技浪潮汹涌澎湃的当下&#xff0c;AI 对于开发者的角色定位成为了一个备受瞩目的焦点话题。 AI 是在助力开发者&#xff0c;还是会取而代之&#xff1f;让我们从技术的角度深入剖析。 不可否认&#xff0c;AI 为开发者带来了前所未有的便…

Django QuerySet对象,all()方法

all()方法 在Django中&#xff0c;all()方法是QuerySet对象的一个方法&#xff0c;用于获取模型的所有实例。 当你调用ModelName.objects.all()时&#xff0c;Django会生成一个SQL查询&#xff0c;从数据库中获取该模型的所有记录&#xff0c;并返回一个QuerySet对象&#xf…