Java面试题及答案汇总(一)
java 基础
1. jdk 和 jre 有什么区别?
-
jdk:java development kit 的简称,java 开发工具包,提供了 java 的开发环境和运行环境。
-
jre:java runtime environment 的简称,java 运行环境,为 java 的运行提供了所需环境。
具体来说 jdk 其实包含了 jre,同时还包含了编译 java 源码的编译器 javac,还包含了很多 java 程序调试和分析的工具。简单来说:如果你需要运行 java 程序,只需安装 jre 就可以了,如果你需要编写 java 程序,需要安装 jdk。
2. == 和 equals 的区别是什么?
== 解读
对于基本类型和引用类型 == 的作用效果是不同的,如下所示:
-
基本类型:比较的是值是否相同;
-
引用类型:比较的是引用是否相同;
代码示例:
1 string x = "string"; 2 string y = "string"; 3 string z = new string("string"); 4 system.out.println(x==y); // true 5 system.out.println(x==z); // false 6 system.out.println(x.equals(y)); // true 7 system.out.println(x.equals(z)); // true
代码解读:因为 x 和 y 指向的是同一个引用,所以 == 也是 true,而 new string()方法则重写开辟了内存空间,所以 == 结果为 false,而 equals 比较的一直是值,所以结果都为 true。
equals 解读
equals 本质上就是 ==,只不过 string 和 integer 等重写了 equals 方法,把它变成了值比较。看下面的代码就明白了。
首先来看默认情况下 equals 比较一个有相同值的对象,代码如下:
1 class cat { 2 public cat(string name) { 3 this.name = name; 4 } 5 6 private string name; 7 8 public string getname() { 9 return name; 10 } 11 12 public void setname(string name) { 13 this.name = name; 14 } 15 } 16 17 cat c1 = new cat("王磊"); 18 cat c2 = new cat("王磊"); 19 system.out.println(c1.equals(c2)); // false
输出结果出乎我们的意料,竟然是 false?这是怎么回事,看了 equals 源码就知道了,源码如下:
1 public boolean equals(object obj) { 2 return (this == obj); 3 }
原来 equals 本质上就是 ==。
那问题来了,两个相同值的 string 对象,为什么返回的是 true?代码如下:
1 string s1 = new string("老王"); 2 string s2 = new string("老王"); 3 system.out.println(s1.equals(s2)); // true
同样的,当我们进入 string 的 equals 方法,找到了答案,代码如下:
1 public boolean equals(object anobject) { 2 if (this == anobject) { 3 return true; 4 } 5 if (anobject instanceof string) { 6 string anotherstring = (string)anobject; 7 int n = value.length; 8 if (n == anotherstring.value.length) { 9 char v1[] = value; 10 char v2[] = anotherstring.value; 11 int i = 0; 12 while (n-- != 0) { 13 if (v1[i] != v2[i]) 14 return false; 15 i++; 16 } 17 return true; 18 } 19 } 20 return false; 21 }
原来是 string 重写了 object 的 equals 方法,把引用比较改成了值比较。
总结 :== 对于基本类型来说是值比较,对于引用类型来说是比较的是引用;而 equals 默认情况下是引用比较,只是很多类重新了 equals 方法,比如 string、integer 等把它变成了值比较,所以一般情况下 equals 比较的是值是否相等。
3. 两个对象的 hashcode()相同,则 equals()也一定为 true,对吗?
不对,两个对象的 hashcode()相同,equals()不一定 true。
代码示例:
1 string str1 = "通话"; 2 string str2 = "重地"; 3 system.out.println(string.format("str1:%d | str2:%d", str1.hashcode(),str2.hashcode())); 4 system.out.println(str1.equals(str2));
执行的结果:
str1:1179395 | str2:1179395 false
代码解读:很显然“通话”和“重地”的 hashcode() 相同,然而 equals() 则为 false,因为在散列表中,hashcode()相等即两个键值对的哈希值相等,然而哈希值相等,并不一定能得出键值对相等。
4. final 在 java 中有什么作用?
-
final 修饰的类叫最终类,该类不能被继承。
-
final 修饰的方法不能被重写。
-
final 修饰的变量叫常量,常量必须初始化,初始化之后值就不能被修改。
5. java 中的 math.round(-1.5) 等于多少?
等于 -1,因为在数轴上取值时,中间值(0.5)向右取整,所以正 0.5 是往上取整,负 0.5 是直接舍弃。
6. string 属于基础的数据类型吗?
string 不属于基础类型,基础类型有 8 种:byte、boolean、char、short、int、float、long、double,而 string 属于对象。
7. java 中操作字符串都有哪些类?它们之间有什么区别?
操作字符串的类有:string、stringbuffer、stringbuilder。
string 和 stringbuffer、stringbuilder 的区别在于 string 声明的是不可变的对象,每次操作都会生成新的 string 对象,然后将指针指向新的 string 对象,而 stringbuffer、stringbuilder 可以在原有对象的基础上进行操作,所以在经常改变字符串内容的情况下最好不要使用 string。
stringbuffer 和 stringbuilder 最大的区别在于,stringbuffer 是线程安全的,而 stringbuilder 是非线程安全的,但 stringbuilder 的性能却高于 stringbuffer,所以在单线程环境下推荐使用 stringbuilder,多线程环境下推荐使用 stringbuffer。
8. string str="i"与 string str=new string("i")一样吗?
不一样,因为内存的分配方式不一样。string str="i"的方式,java 虚拟机会将其分配到常量池中;而 string str=new string("i") 则会被分到堆内存中。
9. 如何将字符串反转?
使用 stringbuilder 或者 stringbuffer 的 reverse() 方法。
示例代码:
1 // stringbuffer reverse 2 stringbuffer stringbuffer = new stringbuffer(); 3 stringbuffer.append("abcdefg"); 4 system.out.println(stringbuffer.reverse()); // gfedcba 5 // stringbuilder reverse 6 stringbuilder stringbuilder = new stringbuilder(); 7 stringbuilder.append("abcdefg"); 8 system.out.println(stringbuilder.reverse()); // gfedcba
10. string 类的常用方法都有那些?
-
indexof():返回指定字符的索引。
-
charat():返回指定索引处的字符。
-
replace():字符串替换。
-
trim():去除字符串两端空白。
-
split():分割字符串,返回一个分割后的字符串数组。
-
getbytes():返回字符串的 byte 类型数组。
-
length():返回字符串长度。
-
tolowercase():将字符串转成小写字母。
-
touppercase():将字符串转成大写字符。
-
substring():截取字符串。
-
equals():字符串比较。
11. 抽象类必须要有抽象方法吗?
不需要,抽象类不一定非要有抽象方法。
示例代码:
1 abstract class cat { 2 public static void sayhi() { 3 system.out.println("hi~"); 4 } 5 }
上面代码,抽象类并没有抽象方法但完全可以正常运行。
12. 普通类和抽象类有哪些区别?
-
普通类不能包含抽象方法,抽象类可以包含抽象方法。
-
抽象类不能直接实例化,普通类可以直接实例化。
13. 抽象类能使用 final 修饰吗?
不能,定义抽象类就是让其他类继承的,如果定义为 final 该类就不能被继承,这样彼此就会产生矛盾,所以 final 不能修饰抽象类,如下图所示,编辑器也会提示错误信息:
14. 接口和抽象类有什么区别?
-
实现:抽象类的子类使用 extends 来继承;接口必须使用 implements 来实现接口。
-
构造函数:抽象类可以有构造函数;接口不能有。
-
main 方法:抽象类可以有 main 方法,并且我们能运行它;接口不能有 main 方法。
-
实现数量:类可以实现很多个接口;但是只能继承一个抽象类。
-
访问修饰符:接口中的方法默认使用 public 修饰;抽象类中的方法可以是任意访问修饰符。
15. java 中 io 流分为几种?
按功能来分:输入流(input)、输出流(output)。
按类型来分:字节流和字符流。
字节流和字符流的区别是:字节流按 8 位传输以字节为单位输入输出数据,字符流按 16 位传输以字符为单位输入输出数据。
16. bio、nio、aio 有什么区别?
-
bio:block io 同步阻塞式 io,就是我们平常使用的传统 io,它的特点是模式简单使用方便,并发处理能力低。
-
nio:new io 同步非阻塞 io,是传统 io 的升级,客户端和服务器端通过 channel(通道)通讯,实现了多路复用。
-
aio:asynchronous io 是 nio 的升级,也叫 nio2,实现了异步非堵塞 io ,异步 io 的操作基于事件和回调机制。
17. files的常用方法都有哪些?
-
files.exists():检测文件路径是否存在。
-
files.createfile():创建文件。
-
files.createdirectory():创建文件夹。
-
files.delete():删除一个文件或目录。
-
files.copy():复制文件。
-
files.move():移动文件。
-
files.size():查看文件个数。
-
files.read():读取文件。
-
files.write():写入文件。