Java SE基础知识详解:源于技术书籍的深度解读

news2025/1/23 21:21:31

写在前面

⭐️在无数次的复习巩固中,我逐渐意识到一个问题:面对同样的面试题目,不同的资料来源往往给出了五花八门的解释,这不仅增加了学习的难度,还容易导致概念上的混淆。特别是当这些信息来自不同博主的文章或是视频教程时,它们之间可能存在的差异性使得原本清晰的概念变得模糊不清。更糟糕的是,许多总结性的面试经验谈要么过于繁复难以记忆,要么就是过于简略,对关键知识点一带而过,常常在提及某项技术时,又引出了更多未经解释的相关术语和实例,例如,在讨论ReentrantLock时,经常会提到这是一个可重入锁,并存在公平与非公平两种实现方式,但对于这两种锁机制背后的原理以及使用场景往往语焉不详。

⭐️正是基于这样的困扰与思考,我决定亲自上阵,撰写一份与众不同的面试指南。这份指南不仅仅是对现有资源的简单汇总,更重要的是,它融入了我的个人理解和解读。我力求回归技术书籍本身,以一种层层递进的方式剖析复杂的技术概念,让那些看似枯燥乏味的知识点变得生动起来,并在我的脑海中构建起一套完整的知识体系。我希望通过这种方式,不仅能帮助自己在未来的技术面试中更加从容不迫,也能为同行们提供一份有价值的参考资料,使大家都能在这个过程中有所收获。

JavaSE相关面试题

面试官: 面向对象和面向过程的区别

候选人:

  • ⾯向过程⾯向过程性能⽐⾯向对象⾼。 ⽐如单⽚机、嵌⼊式开发、Linux/Unix 等⼀般采⽤⾯向过程开发。但是,⾯向过程没有⾯向对象易维护、易复⽤、易扩展。

  • ⾯向对象⾯向对象易维护、易复⽤、易扩展。 因为⾯向对象有封装、继承、多态性的特性,所以可以设计出低耦合的系统,使系统更加灵活、更加易于维护。但是,⾯向对象性能⽐⾯向过程低

拓展:为什么⾯向过程性能⽐⾯向对象⾼?

⾯向过程也需要分配内存,计算内存偏移量,Java 性能差的主要原因并不是因为它是⾯向对象语⾔,⽽是 Java 是半编译语⾔,最终的执⾏代码并不是可以直接被 CPU 执⾏的⼆进制机械码。

⽽⾯向过程语⾔⼤多都是直接编译成机械码在电脑上执⾏,并且其它⼀些⾯向过程的脚本语⾔性能也并不⼀定⽐ Java 好。

面试官:Java语⾔有哪些特点?

候选人:

  1. 简单易学(Java语言的语法与C语言和C++语言很接近);

  2. ⾯向对象(封装,继承,多态);

  3. 跨平台性( Java 虚拟机实现平台⽆关性);

  4. 健壮性(Java语言的强类型机制、异常处理、垃圾的自动收集等);

  5. 安全性;

  6. ⽀持多线程( C++ 语⾔没有内置的多线程机制,因此必须调⽤操作系统的多线程功能来进⾏多线程程序设计,⽽ Java 语⾔却提供了多线程⽀持);

  7. ⽀持⽹络编程( Java 语⾔诞⽣本身就是为简化⽹络编程设计的);

  8. 编译与解释并存

面试官:说说JVM 、 JDK 和 JRE的区别?

候选人:

  • Java Virtual Machine(JVM) 是运⾏ Java 字节码的虚拟机,Java程序需要运行在虚拟机上,不同的平台有自己的虚拟机,因此Java语言可以实现跨平台。

拓展:什么是字节码?采⽤字节码的好处是什么?

在 Java 中,JVM 可以理解的代码就叫做 (即扩展名为 .class 的⽂件),它不⾯向任何特定的处理器,只⾯向虚拟机。Java 语⾔通过字节码的⽅式,在⼀定程度上解决了传统解释型语⾔执⾏效率低的问题,同时⼜保留了解释型语⾔可移植的特点。

  • Java Runtime Environment(JRE) 包括Java虚拟机和Java程序所需的核心类库等。核心类库主要是java.lang包:包含了运行Java程序必不可少的系统类,如基本数据类型、基本数学函数、字符串处理、线程、异常处理类等。
  • Java Development Kit(JDK) 是提供给Java开发人员使用的,其中包含了Java的开发工具,也包括了JRE。所以安装了JDK,就无需再单独安装JRE了。其中的开发工具:编译工具(javac.exe),打包工具(jar.exe)等。

img

面试官:Java 程序从源代码到运⾏分为哪几步?

候选人: Java程序从源代码到运⾏⼀般有下⾯3步:

在这里插入图片描述

我们需要格外注意的是 .class->机器码 这⼀步。在这⼀步 JVM 类加载器⾸先加载字节码⽂件,然后通过解释器逐⾏解释执⾏,这种⽅式的执⾏速度会相对⽐慢。⽽且,有些⽅法和代码块是经常需要被调⽤的(也就是所谓的热点代码),所以后⾯引进了 JIT 编译器,⽽ JIT 属于运⾏时编译。当 JIT 编译器完成第⼀次编译后,其会将字节码对应的机器码保存下来,下次可以直接使⽤。⽽我们知道,机器码的运⾏效率肯定是⾼于 Java 解释器的。这也解释了我们为什么经常会说 Java 是编译与解释共存的语⾔。

面试官:什么是跨平台性?原理是什么?

候选人:

所谓跨平台性,是指 java 语言编写的程序,一次编译后,可以在多个系统平台上运行。

实现原理:Java程序是通过java虚拟机在系统平台上运行的,只要该系统可以安装相应的java虚拟机,该系统就可以运行java程序。

面试官:Java和C++的区别

候选人:

  • 都是面向对象的语言,都支持封装、继承和多态。
  • Java不提供指针来直接访问内存,程序内存更加安全。
  • Java的类是单继承的,C++支持多重继承;虽然Java的类不可以多继承,但是接口可以有多个实现类。
  • Java有自动内存管理机制,不需要程序员手动释放无用内存。
  • 在 C 语⾔中,字符串或字符数组最后都会有⼀个额外的字符‘\0’来表示结束。但是,Java 语⾔中没有结束符这⼀概念。

深度思考:

在C语言中,字符串是以空字符(‘\0’)结尾的字符数组。这种设计的好处是简单直接,但缺点是在访问字符串时每次都需要检查空字符来确定字符串的长度,这可能会导致额外的计算开销。在Java中,字符串是一个类(String),它包含了字符串的值以及其长度信息。这种设计使得字符串的长度可以在创建时确定,并且可以通过内置的方法(如length())轻松获取。这种方法提高了效率,减少了因字符串处理不当而导致的安全隐患。

面试官: 字符型常量和字符串常量的区别?

候选人:

  1. 形式上: 字符型常量是单引号引起的⼀个字符; 字符串常量是双引号引起的若⼲个字符

  2. 含义上: 字符型常量相当于⼀个整型值( ASCII 值),可以参加表达式运算; 字符串常量代表⼀个地址值(该字符串在内存中存放位置)

  3. 占内存⼤⼩:字符型常量只占 2 个字节; 字符串常量占若⼲个字节 (注意: char 在 Java 中占两个字节)

面试官: 重载和重写的区别

候选人:

重载(编译时多态)是同一个类中方法之间的关系,是水平关系。

重写(运行时多态)是子类和父类之间的关系,是垂直关系。

在这里插入图片描述

面试官:面向对象的特征有哪些方面?

候选人:

封装:隐藏对象的属性和实现细节,仅对外提供公共访问方式,将变化隔离,便于使用,提高复用性和安全性。

继承:继承是使用已存在的类的定义作为基础建立新类的技术,新类的定义可以增加新的数据或新的功能,也可以用父类的功能,但不能选择性地继承父类。通过使用继承可以提高代码复用性。继承是多态的前提。

关于继承如下 3 点请记住:

  1. 子类拥有父类非 private 的属性和方法。
  2. 子类可以拥有自己属性和方法,即子类可以对父类进行扩展。
  3. 子类可以用自己的方式实现父类的方法。

多态父类或接口定义的引用变量可以指向子类或具体实现类的实例对象。提高了程序的拓展性。

在Java中有两种形式可以实现多态:继承(多个子类对同一方法的重写)和接口(实现接口并覆盖接口中同一方法)。

面试官: String 、StringBuffer 和 StringBuilder 的区别是什么?(重要)

候选人:

可变性:

  • String对象是不可变类,也就是说String对象一旦被创建,其值将不能被改变。String 类中使⽤ final 关键字修饰字符数组来保存字符串, private final char[] value ,所以 String 对象是不可变的。

拓展:在 Java 9 之后,String 类的实现改⽤ byte 数组存储字符串

private final byte[] value

  • StringBuilder 与 StringBuffer 都继承⾃ AbstractStringBuilder 类,在 AbstractStringBuilder

    也是使⽤字符数组保存字符串 char[] value 但是没有⽤ final 关键字修饰,所以这两种对象都是可

    变的。

    abstract class AbstractStringBuilder implements Appendable, CharSequence {
     /**
     * The value is used for character storage.
     */
     char[] value;
     /**
     * The count is the number of characters used.
     */
     int count;
     AbstractStringBuilder(int capacity) {
     	value = new char[capacity];
     }
    

线程安全性

  • String 中的对象是不可变的,也就可以理解为常量,线程安全。
  • StringBuffer 对AbstractStringBuilder⽅法加了同步锁,所以是线程安全的。StringBuilder 并没有对⽅法加同步锁。

性能

  • 每次对 String 类型进⾏改变的时候,都会⽣成⼀个新的 String 对象,然后将指针指向新的 String对象。

  • StringBuffer 每次都会对对象本身进⾏操作,⽽不是⽣成新的对象并改变对象引⽤。

  • 相同情况下使⽤ StringBuilder 相⽐使⽤ StringBuffer 仅能获得 10%~15% 左右的性能提升,但却要冒多线程不安全的⻛险。(如下)

实例化/初始化:

《Java程序员面试笔试宝典》第二版 1.18节

当实例化String的时候,可以利用构造方法(String s1 = new String(“world”))的方式来对其初始化,也可以使用赋值(String s1 = “Hello”)的方式来初始化,而StringBuffer和StringBuilder只能使用构造方法(StringBuffer s = new StringBuffer(“world”))的方式来初始化。

String 字符串修改实现的原理为:当用 String 类型来对字符串进行修改时,其实现方法是首先创建一个 StringBuilder,然后调用 StingBuilder 的append 方法,最后调用 StingBuilder 的 toString方法把结果返回。举例如下:

String s = "Hello";
s += "World";

以上代码等价于下述代码:

String s = "Hello";
StringBuilder sb = new StringBuilder(s);
s.append("World");
s = sb.toString();

由此可以看出,上述过程比使用 StingBuilder 多了一些附加的操作,同时也生成了一些临时的对象,导致程序的执行效率降低。

对于三者使⽤的总结:

  1. 操作少量的数据: 适⽤ String
  2. 单线程操作字符串缓冲区下操作⼤量数据: 适⽤ StringBuilder
  3. 多线程操作字符串缓冲区下操作⼤量数据: 适⽤ StringBuffer

面试官:接⼝和抽象类的区别是什么?

候选人:

参数抽象类(从属关系is-a)接口(特定功能的实现has-a)
声明抽象类使用abstract关键字声明接口使用interface关键字声明
实现子类使用extends关键字来继承抽象类子类使用implements关键字来实现接口
构造器抽象类可以有构造器接口不能有构造器
访问修饰符抽象类中的方法可以是任意访问修饰符接口方法默认修饰符是public。并且不允许定义为private 或者 protected
多继承一个类最多只能继承一个抽象类一个类可以实现多个接口
字段声明抽象类的字段声明可以是任意的接口的字段默认都是 static 和 final 的

面试官:成员变量与局部变量的区别有哪些?

候选人:

属性成员变量局部变量
作用域针对整个类有效只在某个范围内有效,通常在方法或语句体内
存储位置随着对象的创建而存在,存储在堆内存中在方法被调用或语句被执行的时候存在,存储在栈内存中
生命周期随着对象的创建而存在,随着对象的消失而消失当方法调用完,或者语句结束后,就自动释放
初始值有默认初始值,如数字类型为0,布尔类型为false,引用类型为null没有默认初始值,使用前必须显式赋值
使用原则就近原则(首先在局部范围找,有就使用;接着在成员位置找)就近原则

面试官: == 与 equals(重要)

候选人:

== : 它的作⽤是判断两个对象的地址是不是相等。(基本数据类型 == ⽐的是值,引⽤数据类型 == ⽐的是内存地址)。

equals() : 它的作⽤也是判断两个对象是否相等。但它⼀般有两种使⽤情况:

  • 情况 1:类没有覆盖 equals() ⽅法。则通过 equals() ⽐较该类的两个对象时,等价于通过“==”⽐这两个对象地址。

  • 情况 2:类覆盖了 equals() ⽅法。⼀般我们都覆盖 equals() ⽅法来⽐两个对象的内容是否相等;若它们的内容相等,则返回 true 。

public class test1 {
    public static void main(String[] args) {
        String a = new String("ab"); // a 为一个引用
        String b = new String("ab"); // b为另一个引用,对象的内容一样
        String aa = "ab"; // 放在常量池中
        String bb = "ab"; // 从常量池中查找
        if (aa == bb) // true
            System.out.println("aa==bb");
        if (a == b) // false,非同一对象
            System.out.println("a==b");
        if (a.equals(b)) // true
            System.out.println("aEQb");
        if (42 == 42.0) { // true
            System.out.println("true");
        }
    }
}

面试官:你重写过 hashcode 和 equals 么,为什么重写equals时必须重写hashCode方法?(重要)

候选人:

1)hashCode()介绍:

hashCode() 的作⽤是获取哈希码,也称为散列码;它实际上是返回⼀个 int 整数。这个哈希码的作⽤是确定该对象在哈希表中的索引位置。 hashCode() 定义在 JDK 的 Object 类中,这就意味着 Java 中的任何类都包含有 hashCode() 函数。

public native int hashCode();

2)以“HashSet 如何检查重复”为例子来说明为什么要有 hashCode

当你把对象加入 HashSet 时,HashSet 会先计算对象的 hashcode 值来判断对象加入的位置,同时也会与其他已经加入的对象的 hashcode 值作比较,如果没有相符的hashcode,HashSet会假设对象没有重复出现。但是如果发现有相同 hashcode 值的对象,这时会调用 equals()方法来检查 hashcode 相等的对象是否真的相同。如果两者相同,HashSet 就不会让其加入操作成功。如果不同的话,就会重新散列到其他位置。(摘自《Head first java》第二版)。这样我们就大大减少了 equals 的次数,相应就大大提高了执行速度。

3)为什么重写equals 时必须重写 hashCode ⽅法?

如果两个对象相等,则 hashcode ⼀定也是相同的。两个对象相等,对两个对象分别调⽤ equals⽅法都返回 true。但是,两个对象有相同的 hashcode 值,它们也不⼀定是相等的 。因此,equals ⽅法被覆盖过,则 hashCode ⽅法也必须被覆盖。

面试官:值传递和引用传递有什么区别?(重要)

候选人: Java程序设计语言对对象采用的不是引用调用,实际上,对象引用是按值传递的。(《Java 核⼼技术 卷Ⅰ基础知识》第⼗版 4.5 节)

  • 值传递:指的是在方法调用时,传递的参数是按值的拷贝传递,传递的是值的拷贝,也就是说传递后就互不相关了。Java程序设计语言总是采用按值调用。也就是说,方法得到的是所有参数值的一个拷贝,也就是说,方法不能修改传递给它的任何参数变量的内容。
public static void main(String[] args) {
    int num1 = 10;
    int num2 = 20;

    swap(num1, num2);

    System.out.println("num1 = " + num1);
    System.out.println("num2 = " + num2);
}

public static void swap(int a, int b) {
    int temp = a;
    a = b;
    b = temp;

    System.out.println("a = " + a);
    System.out.println("b = " + b);
}

// 输出结果
a = 20
b = 10
num1 = 10
num2 = 20

在这里插入图片描述

解析:在swap方法中,a、b的值进行交换,并不会影响到 num1、num2。因为,a、b中的值,只是从 num1、num2 的复制过来的。也就是说,a、b相当于num1、num2 的副本,副本的内容无论怎么修改,都不会影响到原件本身。

  • 引用传递:指的是在方法调用时,传递的参数是按引用进行传递,其实传递的引用的地址,也就是变量所对应的内存空间的地址。传递的是值的引用,也就是说传递前和传递后都指向同一个引用(也就是同一个内存空间)。
    public static void main(String[] args) {
        int[] arr = { 1, 2, 3, 4, 5 };
        System.out.println(arr[0]);
        change(arr);
        System.out.println(arr[0]);
    }

    public static void change(int[] array) {
        // 将数组的第一个元素变为0
        array[0] = 0;
    }

// 输出结果
1
0

在这里插入图片描述

解析:array 被初始化 arr 的拷贝也就是一个对象的引用,也就是说 array 和 arr 指向的时同一个数组对象。 因此,外部对引用对象的改变会反映到所对应的对象上。

很多程序设计语言(特别是,C++和Pascal)提供了两种参数传递的方式:值调用和引用调用。有些程序员(甚至本书的作者)认为Java程序设计语言对对象采用的是引用调用,实际上,这种理解是不对的。由于这种误解具有一定的普遍性,所以下面给出一个反例来详细地阐述一下这个问题。

public class Test {

    public static void main(String[] args) {
        // TODO Auto-generated method stub
        Student s1 = new Student("小张");
        Student s2 = new Student("小李");
        Test.swap(s1, s2);
        System.out.println("s1:" + s1.getName());
        System.out.println("s2:" + s2.getName());
    }

    public static void swap(Student x, Student y) {
        Student temp = x;
        x = y;
        y = temp;
        System.out.println("x:" + x.getName());
        System.out.println("y:" + y.getName());
    }
}

// 输出结果
x:小李
y:小张
s1:小张
s2:小李

在这里插入图片描述

在这里插入图片描述

解析:通过上面两张图可以很清晰的看出: 方法并没有改变存储在变量 s1 和 s2 中的对象引用。swap方法的参数x和y被初始化为两个对象引用的拷贝,这个方法交换的是这两个拷贝。

面试官:final 、finally 、finalize区别

候选人:

  • final可以修饰类、变量、方法。修饰类表示该类不能被继承、修饰方法表示该方法不能被重写、修饰变量表
    示该变量是一个常量不能被重新赋值。(如果是基本数据类型的变量,则其数值⼀旦在初始化之后便不能更改;如果是引⽤类型的变量,则在对其初始化之后便不能再让其指向另⼀个对象。)
  • finally一般作用在try-catch代码块中,在处理异常的时候,通常我们将一定要执行的代码方法finally代码块
    中,表示不管是否出现异常,该代码块都会执行,一般用来存放一些关闭资源的代码。
  • finalize是一个方法,属于Object类的一个方法,而Object类是所有类的父类,该方法一般由垃圾回收器来调
    用,当我们调用System.gc() 方法的时候,由垃圾回收器调用finalize(),回收垃圾。

面试官:this与super的区别

候选人:

  • super()和this()类似,区别是,super()在子类中调用父类的构造方法,this()在本类内调用本类的其它构造方法。
  • super()和this()均需放在构造方法内第一行。
  • 可以用this调用一个构造器,但却不能调用两个。
  • this和super不能同时出现在一个构造函数里面,因为this必然会调用其它的构造函数,其它的构造函数必然也会有super语句的存在,所以在同一个构造函数里面有相同的语句,就失去了语句的意义,编译器也不会通过。
  • this()和super()都指的是对象,所以,均不可以在static环境中使用。包括:static变量,static方法,static语句块。
  • 从本质上讲,this是一个指向本对象的指针, 然而super是一个Java关键字。

面试官:Java程序初始化的顺序可以说下吗?

候选人:(《Java程序员面试笔试宝典》第二版 1.1节)

Java程序的初始化工作可以在许多不同的代码块中来完成(例如:静态代码块、构造函数等),它们执行的顺序为:父类静态变量→父类静态代码块→子类静态变量→子类静态代码→父类非静态变量→父类非静态代码块→父类构造方法→子类非静态变量→子类非静态代码块→子类构造方法。下面给出一个不同模块初始化时执行顺序的例子。

class Base {
    static {
        System.out.println("Base static block");
    }

    {
        System.out.println("Base block");
    }

    public Base() {
        System.out.println("Base constructor");
    }
}

public class Derived extends Base {
    static {
        System.out.println("Derived static block");
    }

    {
        System.out.println("Derived block");
    }

    public Derived() {
        System.out.println("Derived constructor");
    }

    public static void main(String[] args) {
        new Derived();
    }
}

// 输出结果
Base static block
Derived static block
Base block
Base constructor
Derived block
Derived constructor

这里需要注意的是,(静态)非静态成员域在定义时初始化和(静态)非静态块中初始化的优先级是平级的,也就是说按照从上到下初始化,最后一次初始化为最终的值(不包括非静态的成员域在构造器中初始化)。所以在(静态)非静态块中初始化的域甚至能在该域声明的上方,因为分配存储空间在初始化之前就完成了。如下例所示:

public class testStatic {
    static {
        a = 2;
    }

    static int a = 1;
    static int b = 3;

    static {
        b = 4;
    }

    public static void main(String[] args) {
        System.out.println(a);
        System.out.println(b);
    }
}

// 输出结果
1
4

面试官:说下构造方法(构造器)吧!

候选人:(《Java程序员面试笔试宝典》第二版 1.2节)

Java 语言中,构造方法具有以下特点:

1)构造方法必须与类的名字相同,并且不能有返回值(返回值也不能为 void)。

2)每个类可以有多个构造方法。

3)构造方法可以有0个、1 个或1个以上的参数。

4)构造方法总是伴随着 new 操作一起调用。

5)构造方法的主要作用是完成对象的初始化工作。

6)构造方法不能被继承,因此就不能被重写(Override),但是构造方法能够被重载(Overload)。

7)当父类和子类都没有定义构造方法的时候,编译器会为父类生成一个默认的无参数的构造方法,给子类也生成一个默认的无参数的构造方法。

面试官:break、continue以及return的区别

候选人:

  • break:跳出总上一层循环,不再执行循环(结束当前的循环体)。所以,当多层循环嵌套,break 语句出现在嵌套循环中的内层循环,它将仅仅只是终止了内层循环的执行,而不影响外层循环的执行。

拓展:由于 break 只能跳出当前的循环,那么如何才能实现跳出多重循环呢?可以在多重循环的外面定义一个标识,然后在循环体里使用带有标识的 break 语句即可跳出多重循环。

public class Break {
 public static void main(String[] args) {
     out:
     for (int i = 0; i < 5; i++) {
         for (int j = 0; j < 5; j++) {
             if (j >= 2) break out;
             System.out.println(j);
         }
     }
     System.out.println("break");
 }
}

// 输出结果
0
1
break

在 C/C++中,goto 常被用作跳出多重循环,在 Java 语言中,可以使用 break 和 continue 来达到同样的效果。那么既然 goto 没有在 Java 语言中使用,为什么还要作为保留字呢?其中一个可能的原因就是这个关键字有可能会在将来被使用。这里需要注意的是,在 Java 语言中,虽然没有 goto 语句,但是却能使用标识符加冒号 (:)的形式定义标签,如“mylabel:”,其目的主要是在多重循环中方便使用 break 和 continue。

  • continue:跳出本次循环,继续执行下次循环(结束正在执行的循环 进入下一个循环条件)。简单地说,continue 只是中断一次循环的执行而己。
  • return:程序返回,不再执行下面的代码(结束当前的方法 直接返回)。

面试官:Java的基本数据类型

候选人: Java 语言一共提供了八种原始的数据类型(byte、short、int、long、float、double、char、boolean)。

(图片源自《Java程序员面试笔试宝典》第二版 1.12节)

在这里插入图片描述

以上这些基本类型可以分为如下四种类型:

1)int 长度数据类型:byte(8bits)、short(16bits)、int(32bits)、long(64bits)。

2)float 长度数据类型:单精度(32bits float)、双精度(64bits double)。

3)boolean 类型变量的取值:true、false。对于 boolean 占用空间的大小,从理论上讲,只需要1 bit 就够了,但在设计的时候为了考虑字节对齐等因素,一般会考虑使其占用一个字节。

4)char 数据类型:Unicode 字符,16 位。

面试官: 说说你对Java 中的异常处理的理解?

候选人: 在 Java 中,所有的异常都有⼀个共同的祖先 java.lang 包中的 Throwable 类。 Throwable 类有两个重要的⼦类 Exception (异常)和 Error (错误)。 Exception 能被程序本身处理( trycatch ), Error 是⽆法处理的(只能尽量避免)。Exception 和 Error ⼆者都是 Java 异常处理的重要⼦类,各⾃都包含⼤量⼦类。

  • Exception:程序本身可以处理的异常,可以通过 catch 来进⾏捕获。 Exception ⼜可以分为 受检查异常(必须处理) 和 不受检查异常(可以不处理)。

受检查异常:Java 代码在编译过程中,如果受检查异常没有被 catch / throw 处理的话,就没办法通过编译。除了 RuntimeException 及其⼦类以外,其他的 Exception 类及其⼦类都属于检查异常 。常⻅的受检查异常有: IO 相关的异常、 ClassNotFoundException 、 SQLException …。

不受检查异常:Java 代码在编译过程中 ,我们即使不处理不受检查异常也可以正常通过编译。RuntimeException 及其⼦类都统称为⾮受检查异常,例如: NullPointExecrption(空指针异常) 、 NumberFormatException (字符串转换为数字)、 ArrayIndexOutOfBoundsException (数组越界)、 ClassCastException (类型转换错误)等。

  • Error:Error 属于程序⽆法处理的错误 ,我们没办法通过 catch 来进⾏捕获 。例如,Java 虚拟机运⾏错误( Virtual MachineError )、虚拟机内存不够错误( OutOfMemoryError )、类定义错误(NoClassDefFoundError )等 。这些异常发⽣时,Java虚拟机(JVM)⼀般会选择线程终⽌。

在这里插入图片描述

面试官:finally块中的代码什么时候被执行?

候选人:(《Java程序员面试笔试宝典》第二版 1.19节)
在 Java 语言的异常处理中,finally 语句块的作用就是保证无论出现什么情况,finally 块里的代码一定会被执行。由于当程序执行 return 的时候就意味着结束对当前方法的调用并跳出这个方法体,任何语句要执行都只能在 return 前执行(除非碰到 exit 函数),因此 fnally 块里的代码也是在return 前执行的。此外,如果 try-finally 或者 catch-finally 中都有 return,则 finally 块中的 retumn 语句将会覆盖别处的 retumn 语句,最终返回到调用者的是 finally 中 return 的值。下面通过两个例子来说明这个问题:

public class Test {
    public static int testFinally() {
        try {
            return 1;
        } catch (Exception e) {
            return 0;
        } finally {
            System.out.println("execute finally");
        }
    }
    public static void main(String[] args) {
        int result = testFinally();
        System.out.println(result);
    }
}

// 输出结果
execute finally
1

从上面这个例子中可以看出,在执行 return 前确实执行了 finally 中的代码。紧接着,在 finally块里面放置 return 语句:

public class Test {
    public static int testFinally() {
        try {
            return 1;
        } catch (Exception e) {
            return 0;
        } finally {
            System.out.println("execute finally");
            return 3;
        }
    }
    public static void main(String[] args) {
        int result = testFinally();
        System.out.println(result);
    }
}

// 输出结果
execute finally
3

在以下 3 种特殊情况下, finally 块不会被执⾏:

  1. 在 try 或 finally 块中⽤了 System.exit(0) 退出程序。

  2. 当程序在进入try语句块之前就出现异常的时候(int i = 5/0;)。

  3. 程序所在的线程死亡。

  4. 关闭 CPU。

面试官: BIO,NIO,AIO 有什么区别?(重要)

候选人:

  • BIO (Blocking I/O): 同步阻塞 I/O 模式,数据的读取写⼊必须阻塞在⼀个线程内等待其完成(每个线程只能处理一个连接)。在活动连接数不是特别⾼(⼩于单机 1000)的情况下,这种模型不错的,可以让每⼀个连接专注于⾃⼰的 I/O 并且编程模型简单,也不⽤过多考虑系统的过载、限流等问题。但是,当⾯对⼗万甚⾄百万级连接的时候,这种方式需要创建大量的线程,而系统的资源都是有限的,大量的线程会降低系统的性能。

  • NIO (Non-blocking I/O): 同步⾮阻塞的 I/O 模型,在 Java 1.4 中引⼊NIO 框架之前,Java通过传统的Socket来实现基本的网络通信功能的。NIO通过 Channels , Selector,Buffers 来实现非阻塞的IO操作。NIO 中的 N 可以理解为 Non-blocking,不单纯是 New。它⽀持⾯向缓冲的,基于通道的 I/O 操作⽅法。它最主要的特点是,提供了基于Selector的异步网络I/O,使得一个线程可以管理多个连接。(图片源自《Java程序员面试笔试宝典》第二版 2.4节)

在这里插入图片描述

扩展:Channel(通道) , Selector(选择器),Buffer(缓冲区)

(1) Channel(通道)
为了更容易地理解什么是 Channel,这里以 InputStream 为例来介绍什么是 Channel。传统的 IO 中经常使用下面的代码来读取文件(此处忽略异常处理):

File file = new File("imput.txt");
InputStream is = new FileInputStream(file);
byte[] tempbyte = new byte[1024];
while((tempbyte=is.read())!=-1){
//处理读取到的数据
}
is.close();

InputStream 其实就是一个用来读取文件的通道。只不过 InputStream 是一个单向的通道,只能用来读取数据。而 NIO 中的 Channel 是一个双向的通道,不仅能读取数据,而且还能写入数据。

(2) Buffer(缓冲区)
在上面的示例代码中,InputSteam 把读取到的数据放在了 byte 数组中,如果用 OutputSteam 写数据,那么也可以把 byte 数组中的数据写到文件中。而在 NIO 中,数据只能被写到 Buffer 中,同理读取的数据也只能放在 Buffer 中,由此可见 Bufer 是 Channel 用来读写数据的非常重要的一个工具。

在这里插入图片描述

(3) Selector(选择器)
Selector 是 NIO 中最重要的部分,是实现一个线程管理多个连接的关键,它的作用就是轮询所有被注册的Channel,一旦发现 Channel 上被注册的事件发生,就可以对这个事件进行处理。

  • AIO (Asynchronous I/O): AIO 也就是 NIO 2。在 Java 7 中引⼊了 NIO 的改进版 NIO 2,它是异步⾮阻塞的 IO 模型。虽然 NIO 在⽹络操作中,也提供了⾮阻塞的⽅法,但它仍需要使用阻塞的方式读取数据,显然这种情况下这个线程就被阻塞了。在高并发环境下,也会有一定的性能问题。造成这个问题的主要原因就是NIO仍然使用了同步的IO。而异步 IO (AIO)是基于事件和回调机制实现的,也就是应⽤操作之后会直接返回,不会堵塞在那⾥,当后台处理完成,操作系统会通知相应的线程进⾏后续的操作。查阅⽹上相关资料,我发现就⽬前来说 AIO 的应⽤还不是很⼴泛,Netty 之前也尝试使⽤过 AIO,不过⼜放弃了。

面试官:深拷⻉ vs 浅拷⻉

候选人:

  1. 浅拷⻉:对基本数据类型进⾏值传递,对引⽤数据类型进⾏引⽤传递般的拷⻉,此为浅拷⻉。

  2. 深拷⻉:对基本数据类型进⾏值传递,对引⽤数据类型,创建⼀个新的对象,并复制其内容,此为深拷⻉。

在这里插入图片描述

面试官: Java反射机制以及获取反射对象的几种方式

候选人:(《Java程序员面试笔试宝典》第二版 1.4节)

在Java语言中,反射机制是指对于运行时类,都能够动态地获取到这个类的所有属性和方法。对于任意的一个对象,都能够调用它的任意一个方法以及访问它的属性;这种动态地获取类或对象的属性以及方法从而完成调用功能被称为Java语言的反射机制。

反射机制中Class是一个非常重要的类,在Java语言中获取Class对象主要有如下几种方法。

  1. 通过className.class来获取。
class A{
    static {
        System.out.println("static block");
    }
    {
        System.out.println("dynamic block");
    }
}

class Test{
    public static void main(String[] args) {
        Class<?> clazz = A.class;
        System.out.println("className:" + clazz.getName());
    }
}

// 执行结果
className:A
  1. 通过Class.forName()来获取。
class A {
    static {
        System.out.println("static block");
    }

    {
        System.out.println("dynamic block");
    }
}

class Test {
    public static void main(String[] args) {
        Class<?> clazz = null;
        try {
            clazz = Class.forName("A");
        } catch (Exception e) {
            e.printStackTrace();
        }
        System.out.println("className:" + clazz.getName());
    }
}

// 执行结果
static block
className:A
  1. 通过Object.getClass()来获取。
class A {
    static {
        System.out.println("static block");
    }

    {
        System.out.println("dynamic block");
    }
}

class Test {
    public static void main(String[] args) {
        Class<?> clazz = new A().getClass();
        System.out.println("className:" + clazz.getName());
    }
}

// 执行结果
static block
dynamic block
className:A

从上面的例子可知,虽然这三种方式都可以获得类的Class对象,但是它们还是有区别的,主要区别如下所示:

方法1)不执行静态块和动态构造块

方法2)只执行静态块,而不执行普通代码块

方法3)因为需要创建对象,所以会执行静态块和普通代码块

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

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

相关文章

Day02Day03

1. 为什么拦截器不会去拦截/admin/login上&#xff0c;是因为在SpringMvc中清除了这种可能。 2.使用自己定义注解&#xff0c;实现AOP&#xff08;insert ,update&#xff09; 3.使用update最好使用动态语句&#xff0c;可以使用多次 4.使用阿里云的OSS存储。用common类 5.在写…

Python 课程16-Pygame

前言 Pygame 是一个基于 Python 的游戏开发库&#xff0c;专门用于多媒体应用程序开发&#xff0c;特别是 2D 游戏。它提供了处理图像、声音、键盘、鼠标等交互功能的 API&#xff0c;并且能够与 OpenGL 集成&#xff0c;用于更复杂的图形操作。Pygame 是初学者和业余开发者学…

erlang学习:mnesia数据库与ets表1

Mnesia 和 ETS 都是 Erlang 提供的表管理工具&#xff0c;用于存储和检索数据&#xff0c;但它们之间有一些重要的区别和共同点。 共同点 都是Erlang提供的表存储机制&#xff1a;ETS 和 Mnesia 都允许你在内存中创建表&#xff0c;并且可以用来存储键值对或者更复杂的数据结…

实战16-RVP定义完成适配

新增文件 //设计搞总宽度 const DRAFT_WIDTH 360//将元素的设计搞大小转化为真机中的大小 export default function rvp(val: number) {/*计算元素真正的大小&#xff1b;* 元素在设计稿的大小 / 设计搞总宽度 x / 真机宽度 (保证元素在不同设备占比相同)x 元素在设计稿的大…

论文不会写?分享6款AI论文写作免费一键生成网站!

在当今学术研究和写作领域&#xff0c;AI论文写作工具的出现极大地提高了写作效率和质量。这些工具不仅能够帮助研究人员快速生成论文草稿&#xff0c;还能进行内容优化、查重和排版等操作。本文将分享6款免费一键生成AI论文写作网站&#xff0c;并重点推荐千笔-AIPassPaper。 …

uniapp富文本editor输入二次扩展兼容微信小程序

在uni-app中开发富文本输入功能&#xff0c;并使其兼容微信小程序&#xff0c;需要注意一些特定的限制和解决方案。由于微信小程序本身对HTML的支持有限&#xff0c;直接在小程序中实现像Web那样完整的富文本编辑功能&#xff08;如使用CKEditor、Quill等&#xff09;是不可能的…

算法笔记/USACO Guide GOLD金组DP 3. Paths on Grids

今天学习背包DP&#xff08;Knapsack DP) 是USACO Guide的DP章节中第三点 What is grid DP? -Summary DP problems often involve a 2D grid where paths are analyzed. Movement is restricted to one direction on the x-axis and y-axis, typically starting from one c…

AI修手有救了?在comfyui中使用Flux模型实现局部重绘案例

&#x1f431;‍&#x1f409;背景 局部重绘相关的话题我们已经讨论和测试过很多次了&#xff0c;比如说inpaint模型、brushnet模型、powerpaint模型等等&#xff0c;最近对于flux模型重绘画面的案例也越来越多了&#xff0c;那我们就结合flux模型的重绘来试试看效果。 &…

体验几款AI论文写作工具后,我认为这个最值得尝试!

开学随之而来的论文写作肯定又让你头疼了吧&#xff0c;而现如今随着AI技术的飞快发展&#xff0c;许多人巧妙地借助AI论文辅助工具&#xff0c;迅速搭建起论文的基本框架&#xff0c;然后一键生成万字正文内容&#xff0c;准确获得大量文献引用&#xff0c;使得整个论文创作过…

[深度学习]Pytorch框架

1 深度学习简介 应用领域&#xff1a;语音交互、文本处理、计算机视觉、深度学习、人机交互、知识图谱、分析处理、问题求解 2 发展历史 1956年人工智能元年2016年国内开始关注深度学习2017年出现Transformer框架2018年Bert和GPT出现2022年&#xff0c;chatGPT出现&#xff0…

基于python+django+vue的美术馆预约系统

作者&#xff1a;计算机学姐 开发技术&#xff1a;SpringBoot、SSM、Vue、MySQL、JSP、ElementUI、Python、小程序等&#xff0c;“文末源码”。 专栏推荐&#xff1a;前后端分离项目源码、SpringBoot项目源码、SSM项目源码 系统展示 【2025最新】基于协同过滤pythondjangovue…

WSL中使用AMBER GPU串行版

前提是已经安装过wsl 1 在 WSL 2 中启用 NVIDIA CUDA 参考在 WSL 2 上启用 NVIDIA CUDA | Microsoft Learn 注意&#xff1a;勿在 WSL 中安装任何 Linux 显示驱动程序。Windows 显示驱动程序将同时安装本机 Windows 和 WSL 支持的常规驱动程序组件。 2 在WSL2中配置Cuda 不安…

SEO之页面优化(一-页面标题2)

初创企业搭建网站的朋友看1号文章&#xff1b;想学习云计算&#xff0c;怎么入门看2号文章谢谢支持&#xff1a; 1、我给不会敲代码又想搭建网站的人建议 2、“新手上云”能够为你开启探索云世界的第一步 博客&#xff1a;阿幸SEO~探索搜索排名之道 &#xff08;接上一篇。。…

OpenCV_最简单的鼠标截取ROI区域

在OpenCV中也存在鼠标的操作&#xff0c;今天我们先介绍一下鼠标中的操作事件 void setMousecallback(const string& winname, MouseCallback onMouse, void* userdata0) setMousecallback参数说明&#xff1a; winname:窗口的名字 onMouse:鼠标响应函数&#xff0c;回调…

基于Springboot+vue的音乐网站

随着信息技术在管理上越来越深入而广泛的应用&#xff0c;管理信息系统的实施在技术上已逐步成熟。本文介绍了音乐网站的开发全过程。通过分析音乐网站管理的不足&#xff0c;创建了一个计算机管理音乐网站的方案。文章介绍了音乐网站的系统分析部分&#xff0c;包括可行性分析…

828华为云征文|Flexus云服务器X实例部署宝塔运维面板

本次华为云Flexus云服务器X实例部署宝塔运维面板教学&#xff0c;这次是推陈出新啊 之前的云耀云服务器L实例已经很不错了&#xff0c;大力赞叹华为云的 同时感谢华为云提供优惠卷&#xff0c;只能说白嫖真是太棒了 华为云近期正在筹办华为云828企业节活动&#xff0c;90款免…

人类行为识别系统源码分享

人类行为识别检测系统源码分享 [一条龙教学YOLOV8标注好的数据集一键训练_70全套改进创新点发刊_Web前端展示] 1.研究背景与意义 项目参考AAAI Association for the Advancement of Artificial Intelligence 项目来源AACV Association for the Advancement of Computer Vis…

【资料分析】刷题日记1

第一套 第二个是相比2019年的增长率&#xff0c;错找为同比增长率 延申&#xff1a; 当出口和进口相比2019年的增长率相同时&#xff0c;可以用盐水解决 √ 一个假设分配&#xff08;第二次是1.4取1&#xff09;加法对比选项 基期倍数&#xff1a; 求A是B的多少倍&#x…

DBeaver纵向展示一行数据

DBeaver查询结果如果列数太多&#xff0c;横着看并不方便&#xff0c;这时可以点击左下角的【记录】按钮&#xff0c;然后可看到纵向的展示结果。如图 就这么一个小功能&#xff0c;没细看的话直接上网搜&#xff0c;不知为啥出来的都是一堆错误方法。所以这里记一下。

2、HDFS编程实践

目录 1、Hadoop三种Shell方式(1)目录操作1、查看目录2、创建目录3、删除目录 &#xff08;2&#xff09;文件操作1、创建文件2、上传文件3、下载文件4、拷贝文件 2、利用Web界面管理HDFS3、利用Java API 与 HDFS 进行交互&#xff08;1&#xff09;安装eclipse包&#xff08;2&…