1. 前言

参加过社招的同学都了解,进入一家公司面试开发岗位时,填写完个人信息后,一般都会让先做一份笔试题,然后公司会根据笔试题的回答结果,确定要不要继续此次面试,如果答的不好,有些公司可能会直接说“技术经理或者总监在忙,你先回去等通知吧”,有些公司可能会继续面试,了解下你的项目经验等情况。

至少在工作的前5年甚至更久,面试一般不会跳过笔试题这个环节(大牛,个别公司除外),我自己也记不清自己面试过多少家公司,做过多少份面试题了,导致现在有时逛街,总感觉很多地方似曾相识,感觉自己多年前曾经来面过试,一度自嘲,一度也怀疑,自己当年是靠什么在上海坚持下来的,所以说面试题对于求职来说,还是非常重要的。

网上搜索“Java面试题”几个关键字也是有很多很多的文章讲解,为什么我还要自己总结呢?主要有以下几个原因:

  • 文章太多,反倒不知道该看哪个(就如一本书中所说太多的资讯等于没有资讯)
  • 文章的准确性不高(曾多次发现描述不正确或代码跑不起来的情况)
  • 可以加深自己的理解和记忆
  • 一劳永逸,下次不用再从网上慢慢筛选,看自己整理的就好了

本篇主要整理下Java基础知识的面试题,主要包含以下几点:

  1. Integer和int的区别
  2. ==和equals的区别
  3. String,StringBuilder,StringBuffer的区别
  4. 装箱和拆箱
  5. Java中的值传递和引用传递

接下来一一讲解。

2. Integer和int的区别

2.1 基本概念区分

  1. Integer是int的包装类(引用类型),int是Java的一种基本数据类型(值类型)。
  2. Integer变量必须实例化后才能使用,而int变量不需要。
  3. Integer实际是对象的引用,当new一个Integer时,实际上是生成一个指针指向此对象,而int则是直接存储数据值。
  4. Integer的默认值是null,int的默认值是0。

2.2 Integer与int常见的几种比较场景

1)两个new Integer()变量相比较,永远返回false

Integer i = new Integer(100);
Integer j = new Integer(100);
System.out.println(i == j); // false

两个通过new生成的Integer变量生成的是两个对象,其内存地址不同

2)非new生成的Integer变量和new Integer()生成的变量相比较,永远返回false

Integer i = new Integer(100);
Integer j = 100;
System.out.println(i == j); // false

非new生成的Integer变量指向的是Java常量池中的对象,而new Integer()生成的变量指向堆中新建的对象,两者在内存中的地址不同

3)两个非new生成的Integer变量比较,如果两个变量的值在区间-128到127 之间,则比较结果为true,如果两个变量的值不在此区间,则比较结果为 false。

Integer i = 100;
Integer j = 100;
System.out.println(i == j); //true Integer i1 = 128;
Integer j1 = 128;
System.out.println(i1 == j1); //false

为什么会这样呢,我们来分析下原因:

Integer i = 100; 在编译时,会翻译成 Integer i = Integer.valueOf(100); ,而Java中Integer类的valueOf方法的源码如下:

public static Integer valueOf(int i) {
if (i >= IntegerCache.low && i <= IntegerCache.high)
return IntegerCache.cache[i + (-IntegerCache.low)];
return new Integer(i);
} private static class IntegerCache {
static final int low = -128;
static final int high;
static final Integer cache[]; static {
// high value may be configured by property
int h = 127;
String integerCacheHighPropValue =
sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
if (integerCacheHighPropValue != null) {
try {
int i = parseInt(integerCacheHighPropValue);
i = Math.max(i, 127);
// Maximum array size is Integer.MAX_VALUE
h = Math.min(i, Integer.MAX_VALUE - (-low) -1);
} catch( NumberFormatException nfe) {
// If the property cannot be parsed into an int, ignore it.
}
}
high = h; cache = new Integer[(high - low) + 1];
int j = low;
for(int k = 0; k < cache.length; k++)
cache[k] = new Integer(j++); // range [-128, 127] must be interned (JLS7 5.1.7)
assert IntegerCache.high >= 127;
} private IntegerCache() {}
}

从源码可以看出:

Java中的Integer对于-128到127之间的数,会进行缓存。

所以 Integer i = 100 时,会将100进行缓存,下次再写Integer j = 100时,就会直接从缓存中取,而不会new了。

4)Integer变量和int变量比较时,只要两个变量的值是向等的,则结果为true

Integer i = new Integer(100);
int j = 100;
System.out.print(i == j); //true

因为包装类Integer和基本数据类型int比较时,Java会自动拆箱为int,然后进行比较,实际上就变为两个int变量的比较

3. ==和equals的区别

3.1 基本概念区分

1)对于==,比较的是值是否相等

如果作用于基本数据类型的变量,则直接比较其存储的 值是否相等,

如果作用于引用类型的变量,则比较的是所指向的对象的地址是否相等。

其实==比较的不管是基本数据类型,还是引用数据类型的变量,比较的都是值,只是引用类型变量存的值是对象的地址

2)对于equals方法,比较的是是否是同一个对象

首先,equals()方法不能作用于基本数据类型的变量,

另外,equals()方法存在于Object类中,而Object类是所有类的直接或间接父类,所以说所有类中的equals()方法都继承自Object类,在没有重写equals()方法的类中,调用equals()方法其实和使用==的效果一样,也是比较的是引用类型的变量所指向的对象的地址,不过,Java提供的类中,有些类都重写了equals()方法,重写后的equals()方法一般都是比较两个对象的值,比如String类。

Object类equals()方法源码:

public boolean equals(Object obj) {
return (this == obj);
}

String类equals()方法源码:

public boolean equals(Object anObject) {
if (this == anObject) {
return true;
}
if (anObject instanceof String) {
String anotherString = (String)anObject;
int n = value.length;
if (n == anotherString.value.length) {
char v1[] = value;
char v2[] = anotherString.value;
int i = 0;
while (n-- != 0) {
if (v1[i] != v2[i])
return false;
i++;
}
return true;
}
}
return false;
}

3.2 示例

示例1:

int x = 10;
int y = 10;
String str1 = new String("abc");
String str2 = new String("abc");
System.out.println(x == y); // true
System.out.println(str1 == str2); // false
System.out.println(str1.equals(str2)); // true

示例2:

String str3 = "abc";
String str4 = "abc";
System.out.println(str3 == str4); // true

str3与str4相等的原因是用到了内存中的常量池,当运行到str3创建对象时,如果常量池中没有,就在常量池中创建一个对象"abc",第二次创建的时候,就直接使用,所以两次创建的对象其实是同一个对象,它们的地址值相等。

示例3:

先定义学生Student类

package com.zwwhnly.springbootaction;

public class Student {
private int age; public Student(int age) {
this.age = age;
}
}

然后创建两个Student实例来比较

Student student1 = new Student(23);
Student student2 = new Student(23); System.out.println(student1.equals(student2)); // false

此时equals方法调用的是基类Object类的equals()方法,也就是==比较,所以返回false。

然后我们重写下equals()方法,只要两个学生的年龄相同,就认为是同一个学生。

package com.zwwhnly.springbootaction;

public class Student {
private int age; public Student(int age) {
this.age = age;
} @Override
public boolean equals(Object obj) {
Student student = (Student) obj;
return this.age == student.age;
}
}

此时再比较刚刚的两个实例,返回true。

Student student1 = new Student(23);
Student student2 = new Student(23); System.out.println(student1.equals(student2)); // true

4. String,StringBuilder,StringBuffer的区别

4.1 区别讲解

1)运行速度

运行速度快慢顺序为:StringBuilder > StringBuffer > String

String最慢的原因:

String为字符串常量,而StringBuilder和StringBuffer均为字符串变量,即String对象一旦创建之后该对象是不可以更改的,但后两者的对象是变量,是可以更改的。

2)线程安全

在线程安全上,StringBuilder是线程不安全的,而StringBuffer是线程安全的(很多方法带有synchronized关键字)。

3)使用场景

String:适用于少量的字符串操作的情况。

StringBuilder:适用于单线程下在字符缓冲区进行大量操作的情况。

StringBuffer:适用于多线程下在字符缓冲区进行大量操作的情况。

4.2 示例

以拼接10000次字符串为例,我们看下三者各自需要的时间:

String str = "";
long startTime = System.currentTimeMillis();
for (int i = 0; i < 10000; i++) {
str = str + i;
}
long endTime = System.currentTimeMillis();
long time = endTime - startTime;
System.out.println("String消耗时间:" + time); StringBuilder builder = new StringBuilder("");
startTime = System.currentTimeMillis();
for (int j = 0; j < 10000; j++) {
builder.append(j);
}
endTime = System.currentTimeMillis();
time = endTime - startTime;
System.out.println("StringBuilder消耗时间:" + time); StringBuffer buffer = new StringBuffer("");
startTime = System.currentTimeMillis();
for (int k = 0; k < 10000; k++) {
buffer.append(k);
}
endTime = System.currentTimeMillis();
time = endTime - startTime;
System.out.println("StringBuffer消耗时间:" + time);

运行结果:

String消耗时间:258

StringBuilder消耗时间:0

StringBuffer消耗时间:1

也验证了上面所说的StringBuilder > StringBuffer > String。

5. 装箱和拆箱

5.1 什么是装箱?什么是拆箱?

装箱:自动将基本数据类型转换为包装器类型。

拆箱:自动将包装器类型转换为基本数据类型。

Integer i = 10; // 装箱
int j = i; // 拆箱

5.2 装箱和拆箱是如何实现的?

装箱过程是通过调用包装器的valueOf方法实现的,

而拆箱过程是通过调用包装器实例的xxxValue方法实现的(xxx代表对应的基本数据类型)。

怎么证明这个结论呢,我们新建个Main类,在主方法中添加如下代码:

package com.zwwhnly.springbootaction;

public class Main {
public static void main(String[] args) {
Integer i = 100;
int j = i;
}
}

然后打开cmd窗口,切换到Main类所在路径,执行命令:javac Main.java,会发现该目录会生成一个Main.class文件,用IDEA打开,会发现编译后的代码如下:

//
// Source code recreated from a .class file by IntelliJ IDEA
// (powered by Fernflower decompiler)
// package com.zwwhnly.springbootaction; public class Main {
public Main() {
} public static void main(String[] var0) {
Integer var1 = Integer.valueOf(100);
int var2 = var1.intValue();
}
}

注意事项:以上所讲使用的是IDEA 2017.2

而在较新版本的IDEA(2018.3.3或者2019.1.3),看到的Main.class是下面这样的:

//
// Source code recreated from a .class file by IntelliJ IDEA
// (powered by Fernflower decompiler)
// package com.zwwhnly.springbootaction; public class Main {
public Main() {
} public static void main(String[] var0) {
Integer var1 = 100;
int var2 = var1;
}
}

5.3 示例

示例1:

Double i1 = 100.0;
Double i2 = 100.0;
Double i3 = 200.0;
Double i4 = 200.0; System.out.println(i1==i2);
System.out.println(i3==i4);

输出结果:

false

false

为什么都返回false呢,我们看下Double.valueOf()方法,就知晓了:

private final double value;

public Double(double value) {
this.value = value;
} public static Double valueOf(double d) {
return new Double(d);
}

示例2:

Boolean i1 = false;
Boolean i2 = false;
Boolean i3 = true;
Boolean i4 = true; System.out.println(i1==i2);
System.out.println(i3==i4);

输出结果:

true

true

为什么都返回true呢,我们看下Boolean.valueOf()方法,就知晓了:

public static final Boolean TRUE = new Boolean(true);
public static final Boolean FALSE = new Boolean(false); public static Boolean valueOf(boolean b) {
return (b ? TRUE : FALSE);
}

6. Java中的值传递和引用传递

6.1 基本概念

值传递:传递对象的一个副本,即使副本被改变,也不会影响源对象,因为值传递的时候,实际上是将实参的值复制一份给形参。

引用传递:传递的并不是实际的对象,而是对象的引用,外部对引用对象的改变也会反映到源对象上,因为引用传递的时候,实际上是将实参的地址值复制一份给形参。

说明:对象传递(数组、类、接口)是引用传递,原始类型数据(整形、浮点型、字符型、布尔型)传递是值传递。

6.2 示例

示例1(值传递):

package com.zwwhnly.springbootaction;

public class ArrayListDemo {

    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的值并未改变。

示例2(引用类型传递):

package com.zwwhnly.springbootaction;

public class ArrayListDemo {

    public static void main(String[] args) {
int[] arr = {1, 2, 3, 4, 5}; change(arr); System.out.println(arr[0]);
} public static void change(int[] array) {
System.out.println(array[0]);
array[0] = 0;
}
}

运行结果:

1

0

可以看出,在change()方法中将数组的第一个元素改为0,主方法中数组的第一个元素也跟着变为0。

示例3(StringBuffer类型):

package com.zwwhnly.springbootaction;

public class ArrayListDemo {

    public static void main(String[] args) {
StringBuffer stringBuffer = new StringBuffer("博客园:申城异乡人");
System.out.println(stringBuffer); changeStringBuffer(stringBuffer);
System.out.println(stringBuffer);
} public static void changeStringBuffer(StringBuffer stringBuffer) {
stringBuffer = new StringBuffer("掘金:申城异乡人"); stringBuffer.append(",欢迎大家关注");
}
}

运行结果:

博客园:申城异乡人

博客园:申城异乡人

也许你会认为第2次应该输出“掘金:申城异乡人,欢迎大家关注”,怎么输出的还是原来的值呢,那是因为在changeStringBuffer中,又new了一个StringBuffer对象,此时stringBuffer变量指向的内存地址已经改变,所以主方法中的stringBuffer变量未受到影响。

如果修改changeStringBuffer()方法的代码为:

public static void changeStringBuffer(StringBuffer stringBuffer) {

    stringBuffer.append(",欢迎大家关注");
}

则运行结果会变为:

博客园:申城异乡人

博客园:申城异乡人,欢迎大家关注

示例4(String类型):

package com.zwwhnly.springbootaction;

public class ArrayListDemo {

    public static void main(String[] args) {
String str = new String("博客园:申城异乡人");
System.out.println(str); changeString(str);
System.out.println(str);
} public static void changeString(String string) {
//string = "掘金:申城异乡人";
string = new String("掘金:申城异乡人");
}
}

运行结果:

博客园:申城异乡人

博客园:申城异乡人

在changeString()方法中不管用string = "掘金:申城异乡人";还是string = new String("掘金:申城异乡人");,主方法中的str变量都不会受影响,也验证了String创建之后是不可变更的。

示例5(自定义类型):

package com.zwwhnly.springbootaction;

public class Person {
private String name; public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public Person(String name) {
this.name = name;
}
}
package com.zwwhnly.springbootaction;

public class ArrayListDemo {

    public static void main(String[] args) {
Person person = new Person("zhangsan");
System.out.println(person.getName()); changePerson(person);
System.out.println(person.getName());
} public static void changePerson(Person p) {
Person person = new Person("lisi");
p = person;
}
}

运行结果:

zhangsan

zhangsan

修改changePerson()方法代码为:

public static void changePerson(Person p) {
p.setName("lisi");
}

则运行结果为:

zhangsan

lisi

7. 参考

关于==和equals的区别和联系,面试这么回答就可以

Java中==号与equals()方法的区别

Java中的String,StringBuilder,StringBuffer三者的区别

深入剖析Java中的装箱和拆箱

Integer、new Integer() 和 int 比较的面试题

java面试题之int和Integer的区别

最最最常见的Java面试题总结-第一周

【Java面试题系列】:Java基础知识常见面试题汇总 第一篇的更多相关文章

  1. Java基础知识常见面试题汇总第一篇

    [Java面试题系列]:Java基础知识常见面试题汇总 第一篇 文中面试题从茫茫网海中精心筛选,如有错误,欢迎指正! 1.前言 ​ 参加过社招的同学都了解,进入一家公司面试开发岗位时,填写完个人信息后 ...

  2. 【Java面试题系列】:Java基础知识常见面试题汇总 第二篇

    文中面试题从茫茫网海中精心筛选,如有错误,欢迎指正! 第一篇链接:[Java面试题系列]:Java基础知识常见面试题汇总 第一篇 1.JDK,JRE,JVM三者之间的联系和区别 你是否考虑过我们写的x ...

  3. java基础知识 + 常见面试题

    准备校招面试之Java篇 一. Java SE 部分 1.1 Java基础 1. 请你解释Object若不重写hashCode()的话,hashCode()如何计算出来的? Object 的 hash ...

  4. php 基础知识 常见面试题

    1.echo.print_r.print.var_dump之间的区别 * echo.print是php语句,var_dump和print_r是函数 * echo 输出一个或多个字符串,中间以逗号隔开, ...

  5. 夯实Java基础系列16:一文读懂Java IO流和常见面试题

    本系列文章将整理到我在GitHub上的<Java面试指南>仓库,更多精彩内容请到我的仓库里查看 https://github.com/h2pl/Java-Tutorial 喜欢的话麻烦点下 ...

  6. Java面试炼金系列 (1) | 关于String类的常见面试题剖析

    Java面试炼金系列 (1) | 关于String类的常见面试题剖析 文章以及源代码已被收录到:https://github.com/mio4/Java-Gold 0x0 基础知识 1. '==' 运 ...

  7. JAVA面试题集之基础知识

                           JAVA面试题集之基础知识 基础知识:  1.C 或Java中的异常处理机制的简单原理和应用. 当JAVA程序违反了JAVA的语义规则时,JAVA虚拟机就 ...

  8. 【搞定 Java 并发面试】面试最常问的 Java 并发基础常见面试题总结!

    本文为 SnailClimb 的原创,目前已经收录自我开源的 JavaGuide 中(61.5 k Star![Java学习+面试指南] 一份涵盖大部分Java程序员所需要掌握的核心知识.欢迎 Sta ...

  9. java常见面试题及答案 1-10(基础篇)

    java常见面试题及答案 1.什么是Java虚拟机?为什么Java被称作是"平台无关的编程语言"? Java 虚拟机是一个可以执行 Java 字节码的虚拟机进程.Java 源文件被 ...

随机推荐

  1. FI模块与SD、MM的接口配置方法

    [转自 http://blog.itpub.net/195776/viewspace-1023910/] 1 FI/SD 借口配置FI/SD通过tcode VKOA为billing设置过帐科目,用户可 ...

  2. re.sub用法

    re.sub功能是对于一个输入的字符串,利用正则表达式,来实现字符串替换处理的功能返回处理后的字符串 re.sub共有五个参数 三个必选参数pattern,repl,string 两个可选参数coun ...

  3. Zookeeper四字命令

    ZooKeeper 支持某些特定的四字命令(The Four Letter Words)与其进行交互.它们大多是查询命令,用来获取 ZooKeeper 服务的当前状态及相关信息.用户在客户端可以通过 ...

  4. swift中反向循环

    First of all, protocol extensions change how reverse is used: for i in (1...5).reverse() { print(i) ...

  5. matlab之text()函数

    text()函数用来给图加上说明性文字. 格式:text(x,y,'txt') 或者:text(x,y,'txt','color','k') x,y是位置 txt是说明文字,如果说明性文字是数字,则这 ...

  6. 草原psd素材

    草原PSD素材,草原,风景,蓝天白云,飞鸟,阳光,绿色,草地. http://www.huiyi8.com/caoyuan/psd/

  7. Elasticsearch: Five Things I was Doing Wrong

    Elasticsearch: Five Things I was Doing Wrong Update: Also check out my series on scaling Elasticsear ...

  8. Eclipse_插件_02_jd-eclipse插件的安装

    1.去官网下载 jd-eclipse插件 2.解压后的文件夹A放到eclipse的drops文件夹下 3.删掉多余文件,确保文件夹A下只有plugin 和 freature 两个文件夹 4.清空osg ...

  9. 【LeetCode】027. Remove Element

    题目: Given an array and a value, remove all instances of that value in place and return the new lengt ...

  10. 解决mongodb查询慢的问题

    最近项目上一直在用mongodb作为数据库,mongodb有他的优势,文档型类json格式存储数据,修改起来比传统的关系型数据库更方便,但是最近在用mongodb出现了查询缓慢的问题,我用命令行查询, ...