logo资料库

2019最全的java面试题.pdf

第1页 / 共146页
第2页 / 共146页
第3页 / 共146页
第4页 / 共146页
第5页 / 共146页
第6页 / 共146页
第7页 / 共146页
第8页 / 共146页
资料共146页,剩余部分请下载后查看
————————————————————————————— 教主的 Java 面试题汇总 有问题的小姐姐可以加 QQ:609947712 咨询 版本:V1.0 一 Java 基础部分 1、面向对象的特征有哪些方面? 答:面向对象的特征主要有以下几个方面: 1)抽象:抽象是将一类对象的共同特征总结出来构造类的过程,包括数据抽象和行为 抽象两方面。抽象只关注对象有哪些属性和行为,并不关注这些行为的细节是什么。 2)继承:继承是从已有类得到继承信息创建新类的过程。提供继承信息的类被称为父 类(超类、基类);得到继承信息的类被称为子类(派生类)。继承让变化中的软件系统有 了一定的延续性,同时继承也是封装程序中可变因素的重要手段(如果不能理解请阅读阎宏 博士的《Java 与模式》或《设计模式精解》中关于桥梁模式的部分)。 3)封装:通常认为封装是把数据和操作数据的方法绑定起来,对数据的访问只能通过 已定义的接口。面向对象的本质就是将现实世界描绘成一系列完全自治、封闭的对象。我们 在类中编写的方法就是对实现细节的一种封装;我们编写一个类就是对数据和数据操作的封 装。可以说,封装就是隐藏一切可隐藏的东西,只向外界提供最简单的编程接口(可以想想 普通洗衣机和全自动洗衣机的差别,明显全自动洗衣机封装更好因此操作起来更简单;我们 现在使用的智能手机也是封装得足够好的,因为几个按键就搞定了所有的事情)。 4)多态性:多态性是指允许不同子类型的对象对同一消息作出不同的响应。简单的说 就是用同样的对象引用调用同样的方法但是做了不同的事情。多态性分为编译时的多态性和 运行时的多态性。如果将对象的方法视为对象向外界提供的服务,那么运行时的多态性可以 解释为:当 A 系统访问 B 系统提供的服务时,B 系统有多种提供服务的方式,但一切对 A 系统来说都是透明的(就像电动剃须刀是 A 系统,它的供电系统是 B 系统,B 系统可以使 用电池供电或者用交流电,甚至还有可能是太阳能,A 系统只会通过 B 类对象调用供电的 方法,但并不知道供电系统的底层实现是什么,究竟通过何种方式获得了动力)。方法重载 教主出版
————————————————————————————— (overload)实现的是编译时的多态性(也称为前绑定),而方法重写(override)实现的 是运行时的多态性(也称为后绑定)。运行时的多态是面向对象最精髓的东西,要实现多态 需要做两件事:1. 方法重写(子类继承父类并重写父类中已有的或抽象的方法);2. 对象 造型(用父类型引用引用子类型对象,这样同样的引用调用同样的方法就会根据子类对象的 不同而表现出不同的行为)。 2、访问修饰符 public,private,protected,以及不写(默认)时的区 别? 答:区别如下: 作用域 当前类 同包 子类 其他 public √ √ √ √ protected √ √ √ × default √ √ × × private √ × × × 类的成员不写访问修饰时默认为 default。默认对于同一个包中的其他类相当于公开 (public),对于不是同一个包中的其他类相当于私有(private)。受保护(protected)对 子类相当于公开,对不是同一包中的没有父子关系的类相当于私有。 3、String 是最基本的数据类型吗? 答:不是。Java 中的基本数据类型只有 8 个:byte、short、int、long、float、double、 char、boolean;除了基本类型(primitive type)和枚举类型(enumeration type),剩下 的都是引用类型(reference type)。 4、float f=3.4;是否正确? 答:不正确。3.4 是双精度数,将双精度型(double)赋值给浮点型(float)属于下转型 (down-casting,也称为窄化)会造成精度损失,因此需要强制类型转换 float f =(float)3.4; 或者写成 float f =3.4F;。 5、short s1 = 1; s1 = s1 + 1;有错吗?short s1 = 1; s1 += 1;有错吗? 答:对于 short s1 = 1; s1 = s1 + 1;由于 1 是 int 类型,因此 s1+1 运算结果也是 int 型, 需要强制转换类型才能赋值给 short 型。而 short s1 = 1; s1 += 1;可以正确编译,因为 s1+= 1;相当于 s1 = (short)(s1 + 1);其中有隐含的强制类型转换。 教主出版
————————————————————————————— 6、Java 有没有 goto? 答:goto 是 Java 中的保留字,在目前版本的 Java 中没有使用。(根据 James Gosling (Java 之父)编写的《The Java Programming Language》一书的附录中给出了一个 Java 关键字列表,其中有 goto 和 const,但是这两个是目前无法使用的关键字,因此有些地方 将其称之为保留字,其实保留字这个词应该有更广泛的意义,因为熟悉 C 语言的程序员都 知道,在系统类库中使用过的有特殊意义的单词或单词的组合都被视为保留字) 7、int 和 Integer 有什么区别? 答:Java 是一个近乎纯洁的面向对象编程语言,但是为了编程的方便还是引入不是对 象的基本数据类型,但是为了能够将这些基本数据类型当成对象操作,Java 为每一个基本 数据类型都引入了对应的包装类型(wrapper class),int 的包装类就是 Integer,从 JDK 1.5 开始引入了自动装箱/拆箱机制,使得二者可以相互转换。 Java 为每个原始类型提供了包装类型: 原始类型: boolean,char,byte,short,int,long,float,double 包装类型:Boolean,Character,Byte,Short,Integer,Long,Float,Double public class AutoUnboxingTest { public static void main(String[] args) { Integer a = new Integer(3); Integer b = 3; // 将 3 自动装箱成 Integer 类型 int c = 3; System.out.println(a == b); // false 两个引用没有引用同一对象 System.out.println(a == c); // true a 自动拆箱成 int 类型再和 c 比较 } } 补充:最近还遇到一个面试题,也是和自动装箱和拆箱相关的,代码如下所示: public class Test03 { public static void main(String[] args) { Integer f1 = 100, f2 = 100, f3 = 150, f4 = 150; System.out.println(f1 == f2); System.out.println(f3 == f4); } } 如果不明就里很容易认为两个输出要么都是 true 要么都是 false。首先需要注意的是 f1、 教主出版
————————————————————————————— f2、f3、f4 四个变量都是 Integer 对象,所以下面的==运算比较的不是值而是引用。装箱的 本质是什么呢?当我们给一个 Integer 对象赋一个 int 值的时候,会调用 Integer 类的静态方 法 valueOf,如果看看 valueOf 的源代码就知道发生了什么。 public static Integer valueOf(int i) { if (i >= IntegerCache.low && i <= IntegerCache.high) return IntegerCache.cache[i + (-IntegerCache.low)]; return new Integer(i); } IntegerCache 是 Integer 的内部类,其代码如下所示: /** * Cache to support the object identity semantics of autoboxing for values between * -128 and 127 (inclusive) as required by JLS. * * The cache is initialized on first usage. The size of the cache * may be controlled by the {@code -XX:AutoBoxCacheMax=} option. * During VM initialization, java.lang.Integer.IntegerCache.high property * may be set and saved in the private system properties in the * sun.misc.VM class. */ 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() {} } 简单的说,如果字面量的值在-128 到 127 之间,那么不会 new 新的 Integer 对象,而 是直接引用常量池中的 Integer 对象,所以上面的面试题中 f1==f2 的结果是 true,而 f3==f4 的结果是 false。越是貌似简单的面试题其中的玄机就越多,需要面试者有相当深厚的功力。 8、&和&&的区别? 答:&运算符有两种用法:(1)按位与;(2)逻辑与。&&运算符是短路与运算。逻辑与跟 短路与的差别是非常巨大的,虽然二者都要求运算符左右两端的布尔值都是 true 整个表达 式的值才是 true。&&之所以称为短路运算是因为,如果&&左边的表达式的值是 false,右边 的表达式会被直接短路掉,不会进行运算。很多时候我们可能都需要用&&而不是&,例如在 验证用户登录时判定用户名不是 null 而且不是空字符串,应当写为:username != null &&!username.equals(“”),二者的顺序不能交换,更不能用&运算符,因为第一个条件如果 不成立,根本不能进行字符串的 equals 比较,否则会产生 NullPointerException 异常。注 意:逻辑或运算符(|)和短路或运算符(||)的差别也是如此。 9、解释内存中的栈(stack)、堆(heap)和静态存储区的用法。 答:通常我们定义一个基本数据类型的变量,一个对象的引用,还有就是函数调用的现 场保存都使用内存中的栈空间;而通过 new 关键字和构造器创建的对象放在堆空间;程序 中的字面量(literal)如直接书写的 100、“hello”和常量都是放在静态存储区中。栈空间操 作最快但是也很小,通常大量的对象都是放在堆空间,整个内存包括硬盘上的虚拟内存都可 以被当成堆空间来使用。 String str = new String(“hello”); 上面的语句中 str 放在栈上,用 new 创建出来的字符串对象放在堆上,而“hello”这个字 面量放在静态存储区。 教主出版
————————————————————————————— 补充:较新版本的 Java 中使用了一项叫“逃逸分析“的技术,可以将一些局部对象放在 栈上以提升对象的操作性能。 10、Math.round(11.5) 等于多少? Math.round(-11.5)等于多少? 答:Math.round(11.5)的返回值是 12,Math.round(-11.5)的返回值是-11。四舍五入的 原理是在参数上加 0.5 然后进行下取整。 11、swtich 是否能作用在 byte 上,是否能作用在 long 上,是否 能作用在 String 上? 答:早期的 JDK 中,switch(expr)中,expr 可以是 byte、short、char、int。从 1.5 版开始,Java 中引入了枚举类型(enum),expr 也可以是枚举,从 JDK 1.7 版开始,还 可以是字符串(String)。长整型(long)是不可以的。 12、用最有效率的方法计算 2 乘以 8? 答: 2 << 3(左移 3 位相当于乘以 2 的 3 次方,右移 3 位相当于除以 2 的 3 次方)。 补充:我们为编写的类重写 hashCode 方法时,可能会看到如下所示的代码,其实我 们不太理解为什么要使用这样的乘法运算来产生哈希码(散列码),而且为什么这个数是个 素数,为什么通常选择 31 这个数?前两个问题的答案你可以自己百度一下,选择 31 是因 为可以用移位和减法运算来代替乘法,从而得到更好的性能。说到这里你可能已经想到了: 31 * num <==> (num << 5) - num,左移 5 位相当于乘以 2 的 5 次方(32)再减去自身就相 当于乘以 31。现在的 VM 都能自动完成这个优化。 package com.loonstudio; public class PhoneNumber { private int areaCode; private String prefix; private String lineNumber; @Override public int hashCode() { final int prime = 31; int result = 1; result = prime * result + areaCode; result = prime * result result = prime * result + ((prefix == null) ? 0 : prefix.hashCode()); return result; 教主出版
————————————————————————————— } @Override public boolean equals(Object obj) { if (this == obj) return true; if (obj == null) return false; if (getClass() != obj.getClass()) return false; PhoneNumber other = (PhoneNumber) obj; if (areaCode != other.areaCode) return false; if (lineNumber == null) { if (other.lineNumber != null) return false; } else if (!lineNumber.equals(other.lineNumber)) return false; if (prefix == null) { if (other.prefix != null) return false; } else if (!prefix.equals(other.prefix)) return false; return true; } } 13、数组有没有 length()方法?String 有没有 length()方法? 答:数组没有 length()方法,有 length 的属性。String 有 length()方法。JavaScript 中, 获得字符串的长度是通过 length 属性得到的,这一点容易和 Java 混淆。 14、在 Java 中,如何跳出当前的多重嵌套循环? 答:在最外层循环前加一个标记如 A,然后用 break A;可以跳出多重循环。(Java 中 支持带标签的 break 和 continue 语句,作用有点类似于 C 和 C++中的 goto 语句,但是就 像要避免使用 goto 一样,应该避免使用带标签的 break 和 continue,因为它不会让你的程 序变得更优雅,很多时候甚至有相反的作用,所以这种语法其实不知道更好) 15、构造器(constructor)是否可被重写(override)? 答:构造器不能被继承,因此不能被重写,但可以被重载。 教主出版
分享到:
收藏