java面试总结报告.docx

上传人:b****8 文档编号:9894550 上传时间:2023-02-07 格式:DOCX 页数:23 大小:26.28KB
下载 相关 举报
java面试总结报告.docx_第1页
第1页 / 共23页
java面试总结报告.docx_第2页
第2页 / 共23页
java面试总结报告.docx_第3页
第3页 / 共23页
java面试总结报告.docx_第4页
第4页 / 共23页
java面试总结报告.docx_第5页
第5页 / 共23页
点击查看更多>>
下载资源
资源描述

java面试总结报告.docx

《java面试总结报告.docx》由会员分享,可在线阅读,更多相关《java面试总结报告.docx(23页珍藏版)》请在冰豆网上搜索。

java面试总结报告.docx

java面试总结报告

java面试总结报告

java面试总结报告

java面试总结报告怎么写?

下面是带来的java面试总结报告,欢迎阅读!

多线程、并发及线程的基础问题:

1)Java中能创建volatile数组吗?

能,Java中可以创建volatile类型数组,不过只是一个指向数组的引用,而不是整个数组。

我的意思是,如果改变引用指向的数组,将会受到volatile的保护,但是如果多个线程同时改变数组的元素,volatile标示符就不能起到之前的保护作用了。

2)volatile能使得一个非原子操作变成原子操作吗?

一个典型的例子是在类中有一个long类型的成员变量。

如果你知道该成员变量会被多个线程访问,如计数器、价格等,你最好是将其设置为volatile。

为什么?

因为Java中读取long类型变量不是原子的,需要分成两步,如果一个线程正在修改该long变量的值,另一个线程可能只能看到该值的一半(前32位)。

但是对一个volatile型的long或double变量的读写是原子。

3)volatile修饰符的有过什么实践?

一种实践是用volatile修饰long和double变量,使其能按原子类型来读写。

double和long都是64位宽,因此对这两种类型的读是分为两部分的,第一次读取第一个32位,然后再读剩下的32位,这个过程不是原子的,但Java中volatile型的long或double变量的读写是原子的。

volatile修复符的另一个作用是提供内存屏障(memorybarrier),例如在分布式框架中的应用。

简单的说,就是当你写一个volatile变量之前,Java内存模型会插入一个写屏障(writebarrier),读一个volatile变量之前,会插入一个读屏障(readbarrier)。

意思就是说,在你写一个volatile域时,能保证任何线程都能看到你写的值,同时,在写之前,也能保证任何数值的更新对所有线程是可见的,因为内存屏障会将其他所有写的值更新到缓存。

4)volatile类型变量提供什么保证?

volatile变量提供顺序和可见性保证,例如,JVM或者JIT为了获得更好的性能会对语句重排序,但是volatile类型变量即使在没有同步块的情况下赋值也不会与其他语句重排序。

volatile提供happens-before的保证,确保一个线程的修改能对其他线程是可见的。

某些情况下,volatile还能提供原子性,如读64位数据类型,像long和double都不是原子的,但volatile类型的double和long就是原子的。

5)你是如何调用wait()方法的?

使用if块还是循环?

为什么?

wait()方法应该在循环调用,因为当线程获取到CPU开始执行的时候,其他条件可能还没有满足,所以在处理前,循环检测条件是否满足会更好。

下面是一段标准的使用wait和notify方法的代码:

//Thestandardidiomforusingthewaitmethod

synchronized(obj){

while(conditiondoesnothold)

obj.wait();//(Releaseslock,andreacquiresonwakeup)

...//Performactionappropriatetocondition}

6)什么是多线程环境下的伪共享(falsesharing)?

伪共享是多线程系统(每个处理器有自己的局部缓存)中一个众所周知的性能问题。

伪共享发生在不同处理器的上的线程对变量的修改依赖于相同的缓存行,如下图所示:

7)什么是Busyspin?

我们为什么要使用它?

Busyspin是一种在不释放CPU的基础上等待事件的技术。

它经常用于避免丢失CPU缓存中的数据(如果线程先暂停,之后在其他CPU上运行就会丢失)。

所以,如果你的工作要求低延迟,并且你的线程目前没有任何顺序,这样你就可以通过循环检测队列中的新消息来代替调用sleep()或wait()方法。

它唯一的好处就是你只需等待很短的时间,如几微秒或几纳秒。

LMAX分布式框架是一个高性能线程间通信的库,该库有一个BusySpinWaitStrategy类就是基于这个概念实现的,使用busyspin循环EventProcessors等待屏障。

8)什么是线程局部变量?

线程局部变量是局限于线程内部的变量,属于线程自身所有,不在多个线程间共享。

Java提供ThreadLocal类来支持线程局部变量,是一种实现线程安全的方式。

但是在管理环境下(如web服务器)使用线程局部变量的时候要特别小心,在这种情况下,工作线程的生命周期比任何应用变量的生命周期都要长。

任何线程局部变量一旦在工作完成后没有释放,Java应用就存在内存泄露的风险。

9)Java中sleep方法和wait方法的区别?

虽然两者都是用来暂停当前运行的线程,但是sleep()实际上只是短暂停顿,因为它不会释放锁,而wait()意味着条件等待,这就是为什么该方法要释放锁,因为只有这样,其他等待的线程才能在满足条件时获取到该锁。

10)什么是不可变对象(immutableobject)?

Java中怎么创建一个不可变对象?

不可变对象指对象一旦被创建,状态就不能再改变。

任何修改都会创建一个新的对象,如String、Integer及其它包装类。

11)我们能创建一个包含可变对象的不可变对象吗?

是的,我们是可以创建一个包含可变对象的不可变对象的,你只需要谨慎一点,不要共享可变对象的引用就可以了,如果需要变化时,就返回原对象的一个拷贝。

最常见的例子就是对象中包含一个日期对象的引用。

数据类型和Java基础面试问题

12)Java中应该使用什么数据类型来代表价格?

如果不是特别关心内存和性能的话,使用BigDecimal,否则使用预定义精度的double类型。

13)怎么将byte转换为String?

可以使用String接收byte[]参数的构造器来进行转换,需要注意的点是要使用的正确的编码,否则会使用平台默认编码,这个编码可能跟原来的编码相同,也可能不同。

14)我们能将int强制转换为byte类型的变量吗?

如果该值大于byte类型的范围,将会出现什么现象?

是的,我们可以做强制转换,但是Java中int是32位的,而byte是8位的,所以,如果强制转化是,int类型的高24位将会被丢弃,byte类型的范围是从-128到128。

15)Java中++操作符是线程安全的吗?

不是线程安全的操作。

它涉及到多个指令,如读取变量值,增加,然后存储回内存,这个过程可能会出现多个线程交差。

16)a=a+b与a+=b的区别?

+=隐式的将加操作的结果类型强制转换为持有结果的类型。

如果两这个整型相加,如byte、short或者int,首先会将它们提升到int类型,然后在执行加法操作。

如果加法操作的结果比a的最大值要大,则a+b会出现编译错误,但是a+=b没问题,如下:

bytea=127;

byteb=127;

b=a+b;//error:

cannotconvertfrominttobyte

b+=a;//ok

注:

其实无论a+b的值为多少,编译器都会报错,因为a+b操作会将a、b提升为int类型,所以将int类型赋值给byte就会编译出错

17)我能在不进行强制转换的情况下将一个double值赋值给long类型的变量吗?

不行,你不能在没有强制类型转换的前提下将一个double值赋值给long类型的变量,因为double类型的范围比long类型更广,所以必须要进行强制转换。

18)3*0.1==0.3将会返回什么?

true还是false?

false,因为有些浮点数不能完全精确的表示出来。

19)int和Integer哪个会占用更多的内存?

Integer对象会占用更多的内存。

Integer是一个对象,需要存储对象的元数据。

但是int是一个原始类型的数据,所以占用的空间更少。

20)为什么Java中的String是不可变的(Immutable)?

Java中的String不可变是因为Java的设计者认为字符串使用非常频繁,将字符串设置为不可变可以允许多个客户端之间共享相同的字符串。

21)Java中的构造器链是什么?

当你从一个构造器中调用另一个构造器,就是Java中的构造器链。

这种情况只在重载了类的构造器的时候才会出现。

JVM底层与GC(GarbageCollection)的面试问题

22)64位JVM中,int的长度是多数?

Java中,int类型变量的长度是一个固定值,与平台无关,都是32位。

意思就是说,在32位和64位的Java虚拟机中,int类型的长度是相同的。

23)Serial与ParallelGC之间的不同之处?

Serial与Parallel在GC执行的时候都会引起stop-the-world。

它们之间主要不同serial收集器是默认的复制收集器,执行GC的时候只有一个线程,而parallel收集器使用多个GC线程来执行。

24)32位和64位的JVM,int类型变量的长度是多数?

32位和64位的JVM中,int类型变量的长度是相同的,都是32位或者4个字节。

25)Java中WeakReference与SoftReference的区别?

虽然WeakReference与SoftReference都有利于提高GC和内存的效率,但是WeakReference,一旦失去最后一个强引用,就会被GC回收,而软引用虽然不能阻止被回收,但是可以延迟到JVM内存不足的时候。

26)WeakHashMap是怎么工作的?

WeakHashMap的工作与正常的HashMap类似,但是使用弱引用作为key,意思就是当key对象没有任何引用时,key/value将会被回收。

27)JVM选项-XX:

+UseCompressedOops有什么作用?

为什么要使用?

当你将你的应用从32位的JVM迁移到64位的JVM时,由于对象的指针从32位增加到了64位,因此堆内存会突然增加,差不多要翻倍。

这也会对CPU缓存(容量比内存小很多)的数据产生不利___。

因为,迁移到64位的JVM主要动机在于可以指定最大堆大小,通过压缩OOP可以节省一定的内存。

通过-XX:

+UseCompressedOops选项,JVM会使用32位的OOP,而不是64位的OOP。

28)怎样通过Java程序来判断JVM是32位还是64位?

你可以检查某些系统属性如sun.arch.data.model或os.arch来获取该信息。

29)32位JVM和64位JVM的最大堆内存分别是多数?

理论上说上32位的JVM堆内存可以到达2^32,即4GB,但实际上会比这个小很多。

不同操作系统之间不同,如Windows系统大约1.5GB,Solaris大约3GB。

64位JVM允许指定最大的堆内存,理论上可以达到2^64,这是一个非常大的数字,实际上你可以指定堆内存大小到100GB。

甚至有的JVM,如Azul,堆内存到1000G都是可能的。

30)JRE、JDK、JVM及JIT之间有什么不同?

JRE代表Java运行时(Javarun-time),是运行Java引用所必须的。

JDK代表Java开发工具(Javadevelopmentkit),是Java程序的开发工具,如Java编译器,它也包含JRE。

JVM代表Java虚拟机(Javavirtualmachine),它的责任是运行Java应用。

JIT代表即时编译(JustInTimepilation),当代码执行的次数超过一定的阈值时,会将Java字节码转换为本地代码,如,主要的热点代码会被准换为本地代码,这样有利大幅度提高Java应用的性能。

31)解释Java堆空间及GC?

当通过Java命令启动Java进程的时候,会为它分配内存。

内存的一部分用于创建堆空间,当程序中创建对象的时候,就从对空间中分配内存。

GC是JVM内部的一个进程,回收无效对象的内存用于将来的分配。

32)你能保证GC执行吗?

不能,虽然你可以调用System.gc()或者Runtime.gc(),但是没有办法保证GC的执行。

33)怎么获取Java程序使用的内存?

堆使用的百分比?

可以通过java.lang.Runtime类中与内存相关方法来获取剩余的内存,总内存及最大堆内存。

通过这些方法你也可以获取到堆使用的百分比及堆内存的剩余空间。

Runtime.freeMemory()方法返回剩余空间的字节数,Runtime.totalMemory()方法总内存的字节数,Runtime.maxMemory()返回最大内存的字节数。

34)Java中堆和栈有什么区别?

JVM中堆和栈属于不同的内存区域,使用目的也不同。

栈常用于保存方法帧和局部变量,而对象总是在堆上分配。

栈通常都比堆小,也不会在多个线程之间共享,而堆被整个JVM的所有线程共享。

Java基本概念面试题

35)“a==b”和”a.equals(b)”有什么区别?

如果a和b都是对象,则a==b是比较两个对象的引用,只有当a和b指向的是堆中的同一个对象才会返回true,而a.equals(b)是进行逻辑比较,所以通常需要重写该方法来提供逻辑一致性的比较。

例如,String类重写equals()方法,所以可以用于两个不同对象,但是包含的字母相同的比较。

36)a.hashCode()有什么用?

与a.equals(b)有什么关系?

hashCode()方法是相应对象整型的hash值。

它常用于基于hash的集合类,如Hashtable、HashMap、LinkedHashMap等等。

它与equals()方法关系特别紧密。

根据Java规范,两个使用equal()方法来判断相等的对象,必须具有相同的hashcode。

37)final、finalize和finally的不同之处?

final是一个修饰符,可以修饰变量、方法和类。

如果final修饰变量,意味着该变量的值在初始化后不能被改变。

finalize方法是在对象被回收之前调用的方法,给对象自己最后一个复活的机会,但是什么时候调用finalize没有保证。

finally是一个关键字,与try和catch一起用于异常的处理。

finally块一定会被执行,无论在try块中是否有发生异常。

38)Java中的编译期常量是什么?

使用它又什么风险?

公共静态不可变(publicstaticfinal)变量也就是我们所说的编译期常量,这里的public可选的。

实际上这些变量在编译时会被替换掉,因为编译器知道这些变量的值,并且知道这些变量在运行时不能改变。

这种方式存在的一个问题是你使用了一个内部的或第三方库中的公有编译时常量,但是这个值后面被其他人改变了,但是你的客户端仍然在使用老的值,甚至你已经部署了一个新的jar。

为了避免这种情况,当你在更新依赖JAR文件时,确保重新编译你的程序。

Java集合框架的面试题

39)List、Set、Map和Queue之间的区别?

List是一个有序集合,允许元素重复。

它的某些实现可以提供基于下标值的常量访问时间,但是这不是List接口保证的。

Set是一个无序集合。

40)poll()方法和remove()方法的区别?

poll()和remove()都是从队列中取出一个元素,但是poll()在获取元素失败的时候会返回空,但是remove()失败的时候会抛出异常。

41)Java中LinkedHashMap和PriorityQueue的区别是什么?

PriorityQueue保证最高或者最低优先级的的元素总是在队列头部,但是LinkedHashMap维持的顺序是元素插入的顺序。

当遍历一个PriorityQueue时,没有任何顺序保证,但是LinkedHashMap课保证遍历顺序是元素插入的顺序。

42)ArrayList与LinkedList的不区别?

最明显的区别是ArrrayList底层的'数据结构是数组,支持随机访问,而LinkedList的底层数据结构书链表,不支持随机访问。

使用下标访问一个元素,ArrayList的时间复杂度是O

(1),而LinkedList是O(n)。

43)用哪两种方式来实现集合的排序?

你可以使用有序集合,如TreeSet或TreeMap,你也可以使用有顺序的的集合,如list,然后通过Collections.sort()来排序。

44)Java中怎么打印数组?

你可以使用Arrays.toString()和Arrays.deepToString()方法来打印数组。

由于数组没有实现toString()方法,所以如果将数组传递给System.out.println()方法,将无法打印出数组的内容,但是Arrays.toString()可以打印每个元素。

45)Hashtable与HashMap有什么不同之处?

这两个类有许多不同的地方,下面列出了一部分:

a)Hashtable是JDK1遗留下来的类,而HashMap是后来增加的。

b)Hashtable是同步的,比较慢,但HashMap没有同步策略,所以会更快。

c)Hashtable不允许有个空的key,但是HashMap允许出现一个nullkey。

46)Java中的HashSet,内部是如何工作的?

HashSet的内部采用HashMap来实现。

由于Map需要key和value,所以所有key的都有一个默认value。

类似于HashMap,HashSet不允许重复的key,只允许有一个nullkey,意思就是HashSet中只允许存储一个null对象。

47)写一段代码在遍历ArrayList时移除一个元素?

该问题的关键在于面试者使用的是ArrayList的remove()还是Iterator的remove()方法。

这有一段示例代码,是使用正确的方式来实现在遍历的过程中移除元素,而不会出现ConcurrentModificationException异常的示例代码。

48)我们能自己写一个容器类,然后使用for-each循环码?

可以,你可以写一个自己的容器类。

如果你想使用Java中增强的循环来遍历,你只需要实现Iterable接口。

如果你实现Collection接口,默认就具有该属性。

49)ArrayList和HashMap的默认大小是多数?

在Java7中,ArrayList的默认大小是10个元素,HashMap的默认大小是16个元素(必须是2的幂)。

这就是Java7中ArrayList和HashMap类的代码片段:

//fromArrayList.javaJDK1.7

privatestaticfinalintDEFAULT_CAPACITY=10;

//fromHashMap.javaJDK7

staticfinalintDEFAULT_INITIAL_CAPACITY=1<<4;//aka16

50)有没有可能两个不相等的对象有有相同的hashcode?

有可能,两个不相等的对象可能会有相同的hashcode值,这就是为什么在hashmap中会有冲突。

相等hashcode值的规定只是说如果两个对象相等,必须有相同的hashcode值,但是没有关于不相等对象的任何规定。

51)两个相同的对象会有不同的的hashcode吗?

不能,根据hashcode的规定,这是不可能的。

52)我们可以在hashcode()中使用随机数字吗?

不行,因为对象的hashcode值必须是相同的。

参见答案获取更多关于Java中重写hashCode()方法的知识。

53)Java中,Comparator与Comparable有什么不同?

Comparable接口用于定义对象的自然顺序,而parator通常用于定义用户定制的顺序。

Comparable总是只有一个,但是可以有多个parator来定义对象的顺序。

54)为什么在重写equals方法的时候需要重写hashCode方法?

因为有强制的规范指定需要同时重写hashcode与equal是方法,许多容器类,如HashMap、HashSet都依赖于hashcode与equals的规定。

Java最佳实践的面试问题

55)Java中,编写多线程程序的时候你会遵循哪些最佳实践?

a)给线程命名,这样可以帮助调试

展开阅读全文
相关资源
猜你喜欢
相关搜索

当前位置:首页 > 求职职场 > 简历

copyright@ 2008-2022 冰豆网网站版权所有

经营许可证编号:鄂ICP备2022015515号-1