面试题 | 一道有趣的 Java 基础题

简介: 面试题 | 一道有趣的 Java 基础题

一、问题

       在一个 Java 群里有位群友分享了一道关于 Java 的题目,问代码是否抛异常。代码如下:

public class Hello {
    static String a, b;
    public static void main(String[] argc) {
        a = a + b;
        System.out.println(a);
    }
}


       对于该问题,我只知道是不会抛出异常的,但是对于输出的结果我认为是空字符串。不知道大家是怎么考虑的。后来这位群友,给出的答案是不会抛出异常,输出的结果是 nullnull,且计算长度是 8 。感觉比较有意思,就自己分析了一下。

二、测试

       在对这个结果产生兴趣时,我先进行了测试,将代码进行编译并运行,输出情况如下:

PS C:\Users\Administrator\Desktop> java Hello
nullnull

     肉眼直接观察,输出的结果的确是 nullnul,而长度也肯定是 8。那么为什么呢?还是需要从 Java 的内部去进行了解。

三、分析

       分析的最直接的方法应该是看 JDK 的源码,但是 JDK 的代码浩如烟海,不知从何看起。那么就直接看其反汇编代码。其反汇编代码如下:

0: new           #2                  // class java/lang/StringBuilder
 3: dup
 4: invokespecial #3                  // Method java/lang/StringBuilder."<init>":()V
 7: getstatic     #4                  // Field a:Ljava/lang/String;
10: invokevirtual #5                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
13: getstatic     #6                  // Field b:Ljava/lang/String;
16: invokevirtual #5                  // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder;
19: invokevirtual #7                  // Method java/lang/StringBuilder.toString:()Ljava/lang/String;
22: putstatic     #4                  // Field a:Ljava/lang/String;
25: getstatic     #8                  // Field java/lang/System.out:Ljava/io/PrintStream;
28: getstatic     #4                  // Field a:Ljava/lang/String;
31: invokevirtual #9                  // Method java/io/PrintStream.println:(Ljava/lang/String;)V
34: return

       我没有了解过 JVM 的指令,但是我了解过安卓 Dalvik 的 Smali 指令。两者在设计上虽然有所区别,但是实际还是比较相似的。不过本身这种虚拟机的指令抽象程度较高,也算是比较好理解。那么,这里我们就来看 StringBuilder 的 append 方法的源码(如果不看反汇编代码,我根本想不到这里要看的是 StringBuilder 类的 append 方法)。虽然,StringBuilder 有多个重载的 append 方法,但是根据反汇编指令也能够知道具体调用的是哪个方法。其源码如下:

@Override
public StringBuilder append(String str) {
    super.append(str);
    return this;
}

       这里直接调用了其父类的方法,接着往上看它的代码,代码如下:

public AbstractStringBuilder append(String str) {
    if (str == null)
        return appendNull();
    int len = str.length();
    ensureCapacityInternal(count + len);
    str.getChars(0, len, value, count);
    count += len;
    return this;
}

       该代码在 AbstractStringBuilder 类中,通过上面的代码可以知道,当 str 为 null 时,会调用 appendNull 方法,该方法的代码如下:

private AbstractStringBuilder appendNull() {
    int c = count;
    ensureCapacityInternal(c + 4);
    final char[] value = this.value;
    value[c++] = 'n';
    value[c++] = 'u';
    value[c++] = 'l';
    value[c++] = 'l';
    count = c;
    return this;
}

     从以上代码就不难看出,输出结果为什么是 nullnull 的情况了。

四、小结

       这样的问题意义何在呢?我个人觉得,在项目中总会出现各种奇奇怪怪的问题,而一些奇奇怪怪的问题却是我们平时疏忽的基础或细节造成的,因此也在空闲之余,多多关注基础知识和技术细节,会有助于自己解决很多奇奇怪怪的问题。

相关文章
|
2月前
|
存储 Java
【IO面试题 四】、介绍一下Java的序列化与反序列化
Java的序列化与反序列化允许对象通过实现Serializable接口转换成字节序列并存储或传输,之后可以通过ObjectInputStream和ObjectOutputStream的方法将这些字节序列恢复成对象。
|
2月前
|
Java 编译器 C++
【Java基础面试一】、为什么Java代码可以实现一次编写、到处运行?
这篇文章解释了Java能够实现“一次编写,到处运行”的原因,主要归功于Java虚拟机(JVM),它能够在不同平台上将Java源代码编译成的字节码转换成对应平台的机器码,实现跨平台运行。
【Java基础面试一】、为什么Java代码可以实现一次编写、到处运行?
|
2月前
|
Java
【Java基础面试四】、介绍一下Java的数据类型
这篇文章介绍了Java的数据类型,包括8种基本数据类型(整数、浮点、字符、布尔)和3类引用数据类型(数组、类、接口),并提供了基本数据类型所占内存空间和数据范围的详细信息。
|
2月前
|
Java C++
【Java基础面试十七】、Java为什么是单继承,为什么不能多继承?
这篇文章讨论了Java单继承的设计原因,指出Java不支持多继承主要是为了避免方法名冲突等混淆问题,尽管Java类不能直接继承多个父类,但可以通过接口和继承链实现类似多继承的效果。
【Java基础面试十七】、Java为什么是单继承,为什么不能多继承?
|
2月前
|
Java
【Java基础面试三】、说一说你对Java访问权限的了解
这篇文章介绍了Java中的四种访问权限:private、default(无修饰符时的访问权限)、protected和public,以及它们分别在修饰成员变量/方法和类时的不同访问级别和规则。
【Java基础面试三】、说一说你对Java访问权限的了解
|
2月前
|
存储 缓存 网络协议
复盘女朋友面试4个月的Java基础题
这篇文章是关于Java基础面试题的复盘,涵盖了HashMap原理、对象序列化作用等高频面试问题,并强调了Java基础知识的重要性。
复盘女朋友面试4个月的Java基础题
|
2月前
|
Java
【Java基础面试二】、个Java文件里可以有多个类吗(不含内部类)?
这篇文章讨论了Java文件中类的定义规则,指出一个Java文件可以包含多个类(不包含内部类),但其中最多只能有一个public类,且如果有public类,它的名称必须与文件名一致。
|
2月前
|
XML 存储 JSON
【IO面试题 六】、 除了Java自带的序列化之外,你还了解哪些序列化工具?
除了Java自带的序列化,常见的序列化工具还包括JSON(如jackson、gson、fastjson)、Protobuf、Thrift和Avro,各具特点,适用于不同的应用场景和性能需求。
|
2月前
|
Java
【Java基础面试三十七】、说一说Java的异常机制
这篇文章介绍了Java异常机制的三个主要方面:异常处理(使用try、catch、finally语句)、抛出异常(使用throw和throws关键字)、以及异常跟踪栈(异常传播和程序终止时的栈信息输出)。
|
2月前
|
Java
【Java基础面试三十八】、请介绍Java的异常接口
这篇文章介绍了Java的异常体系结构,主要讲述了Throwable作为异常的顶层父类,以及其子类Error和Exception的区别和处理方式。
下一篇
无影云桌面