欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

JDK基础必备面试十问

程序员文章站 2022-12-21 10:26:29
1. new一个对象在Java内部做了哪些工作? 从静态角度来看,new一个对象表示创建一个类的对象实例。 从JVM运行角度来看,当JVM执行到new字节码时,首先会去查看类有没有被加载到内存以及初始化,如果是第一次使用该类,则首先加载该类。加载完成后便会在堆内存分配该对象实例的内存空间,虚拟机栈分 ......

1. new一个对象在java内部做了哪些工作?

从静态角度来看,new一个对象表示创建一个类的对象实例。

从jvm运行角度来看,当jvm执行到new字节码时,首先会去查看类有没有被加载到内存以及初始化,如果是第一次使用该类,则首先加载该类。加载完成后便会在堆内存分配该对象实例的内存空间,虚拟机栈分配对象实例的应用内存。

2. 抽象类是否可以定义构造函数?如果能,是否能new一个抽象类?

抽象类同样也可以定义构造函数,但是它不能new一个抽象类。

3. 既然不能new一个抽象类,那它定义构造函数有什么意义呢?

抽象类中的构造函数只能通过构造函数链调用,也就是从其他类中的构造函数调用,它的作用可以初始化抽象类中的一些初始值。

4. string是否是基本数据类型?它与stringbuilder、stringbuffer有什么区别?

string不是基本数据类型。

string是不可变的,尽管它能在程序中多次赋值以及拼接,但实际上每一次赋值都是在内存中重新开辟一块内存空间。

stringbuilder和stringbuffer是可变的,多次对它们赋值不会在内存中开辟一块内存空间,stringbuilder不是线程安全,stringbuffer是线程安全的。

5. stringbuilder与stringbuffer的内部实现原理是什么?

stringbuffer与stringbuilder的不同点在于stringbuffer在append方法加了synchronized关键字,它是线程安全的。

它们都是继承自abstractbuilder,内部实现都是一个可变数组,数组初始长度为16。当调用append方法拼接字符串时,其内部实际上是调用了system.arraycopy将字符串拷贝进了可变数组。

6. stringbuilder的扩容机制是什么?

stringbuilder在内部是一个字符数组,默认大小为16,当容量超过16时,会进行扩容,新的数组大小是之前数组大小的2倍+2,也就是第一次扩容大小为34。扩容后将以前的数组拷贝到新数组中。

7. string str = "a"与string str = new string("b")有什么区别?

string str = "a",首先会去常量池中查找是否有"a"字符串,如果有则直接指向它,没有则在常量池中创建并指向它。

string str = new string("b")则会在堆内存中创建一个string对象实例,并指向它,同时它也会在常量池中创建"b"对象。

8. ==与equals比较有什么区别?

==比较的是引用地址,

equals通常比较的是值,equals在object中的实现仍然是==,所以如果要通过equals比较值就必须重写equals。

9. 重写equals方法需要注意什么?

在重写equals方法时,一定要重写hashcode方法,hashcode方法是计算对象的hash值。

在java中规定:

equals等于true,则它们的hashcode一定相等;

equals等于false,则它们的hashcode可能相等可能不相等,也就是如果hashcode相等,则equals不一定相等。

之所以要重写hashcode方法,主要是应用在集合中的判断。

如果没有重写类的hashcode方法,只重写了equals方法,当两个对象equals等于true时,它们的hashcode不相等。此时如果将它们作为key放到map集合中,由于它们的hash值不相等,所以map认为它们是不相等的key,此时在map中将会在逻辑上存在两个相等的key值,不符合我们对程序的预期。所以在重写equals方法时必须重写hashcode方法。

10. 重写hashcode方法需要注意什么?

在设计散列函数时,应该尽量避免冲突。如果频繁的产生散列冲突,在将对象作为key存放在map中时,会将不同的key值散列到一个位置,对map的性能会有所影响。可以参考string的hashcode实现,将质数31数字作为乘法因子。

关注coderbuff公众号,回复“面试”获取更多
JDK基础必备面试十问