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数字作为乘法因子。