这是在金融领域(主要是大型投资银行)共享Java核心访谈问题和答案的新系列。 在JP Morgan,Morgan Stanley,Barclays或Goldman Sachs上会问许多这些Java面试问题。 银行主要从多线程 , 集合 ,序列化,编码和OOPS设计原理询问Java核心访谈问题。
我已经从我的朋友那里收集了这些问题,并希望与大家分享。 我希望这对我们双方都有帮助。 练习一些编程面试问题也是有益的,因为在几乎所有Java面试中,至少出现1或2个编码问题。 请分享未回答的Java面试问题的答案,并让我们知道这些Java面试问题有多好? 如果您正在认真准备Java面试并参加面试,那么我也建议您看看Markham公开的Java编程面试 。
这是一本专门针对Java的书籍,以其较早的畅销书为原型,也涉及编程访谈。 本书不仅包含来自Java的问题,还包含来自相关技术堆栈的问题,例如JUnit,Maven,设计模式,JVM内部原理,Android和最佳实践。 问题很好,答案也很明确,而且解释得很好,这也使它读起来很有趣。
20多个核心Java面试问题答案
这些Java面试问题是简单,棘手和棘手的Java问题的组合,例如,为什么Java中不支持多重继承是棘手的问题之一。 大多数问题是在高级和有经验的水平上提出的,即3、4、5或6年的Java经验,例如HashMap如何在Java中工作,这在有经验的Java访谈中最受欢迎。
顺便说一句,最近我正在查看关于本文中给出的Java面试问题的答案和评论,我发现其中的一些内容非常有用,可以包含在主要文章中,从而使所有人受益。 除了博客和文章之外,您还可以利用一些书籍,这些书籍是专门为清除编程访谈而写的,而另一些书籍则着重于Java编程,因此想到的两本书是《 破解编码面试》和《 公开的编程面试》:下一份工作的秘密和。 这两本书都侧重于一般的编程以及许多其他相关主题,例如数据结构,算法,数据库,SQL,网络和行为问题,但也包含Java概念。
问题1:在多线程环境中使用HashMap有什么问题? 什么时候get()
方法进入无限循环? ( 回答 )
好吧,没有错,这取决于您如何使用它。 例如,如果仅通过一个线程初始化HashMap,然后所有线程都仅从该线程读取,那么就很好了。 一个示例是包含配置属性的映射。 真正的问题开始于该线程中的至少一个线程正在更新HashMap,即添加,更改或删除任何键值对。 由于put()操作可能会导致重新调整大小,并且可能进一步导致无限循环,因此这就是为什么您以后应该使用Hashtable或ConcurrentHashMap的原因。
问题2.覆盖hashCode()方法是否对性能有影响? ( 回答 )
这是一个好问题,对所有人开放,因为据我所知,较差的哈希码功能将导致HashMap中频繁发生冲突,最终增加了将对象添加到Hash Map中的时间。 从Java 8开始,冲突不会像以前的版本那样影响性能,因为在阈值之后,链表将被二叉树取代,这在最坏的情况下会为您带来O(logN)性能,例如与链表的O(n)比较。
问题3:不可变对象的所有属性是否都必须是最终的? ( 回答 )
不一定,如上所述,您可以通过使成员成为非最终成员而不是私有成员并且不修改它们(除非在构造函数中)来实现相同的功能。 不要为它们提供setter方法,如果它是一个可变对象,则不要泄漏该成员的任何引用。 请记住将参考变量定为final,仅确保不会将其重新分配为其他值,但是您仍然可以更改该参考变量所指向的对象的各个属性。 这是关键点之一,面试官喜欢听候选人的话。
问题4:String中的substring()方法如何工作? ( 回答 )
另一个很好的Java面试问题,我认为答案还不够,但是这里是“子字符串通过从原始字符串中提取一部分来创建一个新对象”。 主要询问此问题,以查看开发人员是否熟悉内存泄漏的风险,子字符串可能会造成内存泄漏。 在Java 1.7之前,子字符串保留原始字符数组的引用,这意味着即使长5个字符的子字符串也可以通过保留强引用来防止1GB字符数组被垃圾回收。
此问题已在Java 1.7中修复,在Java 1.7中不再引用原始字符数组,但是这种更改也使创建子字符串的时间花费更高。 之前是在O(1)的范围内,在Java 7的最坏情况下可能是O(n)。
问题 5:您可以为单例编写关键的部分代码吗? ( 回答 )
这个核心Java问题是上一个问题的跟进,并期望候选人使用双重检查锁定来编写Java单例。 请记住使用volatile变量使Singleton线程安全。 这是使用双重检查的锁定习惯用法的线程安全的Singleton模式的关键部分的代码:
public class Singleton {
private static volatile Singleton _instance;
/**
* Double checked locking code on Singleton
* @return Singelton instance
*/
public static Singleton getInstance() {
if (_instance == null) {
synchronized (Singleton.class) {
if (_instance == null) {
_instance = new Singleton();
}
}
}
return _instance;
}
}
问题6:在编写存储过程或从Java访问存储过程时如何处理错误情况? ( 回答 )
这是Java面试中最棘手的问题之一,而且开放给所有人使用,我的朋友不知道答案,所以他不介意告诉我。 我的看法是,如果某些操作失败,则存储过程应该返回错误代码,但是如果存储过程本身失败,则捕获SQLException是唯一的选择。
问题7:Executor.submit()和Executer.execute()方法之间有什么区别? ( 回答 )
该问题来自我的15个Java多线程问题解答。 由于具有良好的并发技能的Java开发人员的巨大需求,它日趋流行。 答案是前者返回一个Future对象,该对象可用于从工作线程中查找结果。
查看异常处理时有所不同。 如果您的任务抛出异常,并且该异常是在执行时提交的,则该异常将进入未捕获的异常处理程序(当您未明确提供异常处理程序时,默认的异常处理程序只会将堆栈跟踪打印到System.err)。 如果在提交任务时提交了任何抛出的异常(是否经过检查的异常) ,则它将成为任务返回状态的一部分。 对于通过提交提交且以异常终止的任务, Future.get()
将重新引发此异常,并包装在ExecutionException
。
问题8:工厂模式和抽象工厂模式有什么区别? ( 回答 )
抽象工厂提供了另一层抽象。 考虑不同的工厂,每个工厂都从抽象工厂扩展而来,并负责根据工厂的类型创建不同的对象层次结构。 例如,由AutomobileFactory
, UserFactory
, RoleFactory
等扩展的RoleFactory
。每个工厂都将负责创建该类型的对象。 这是工厂和抽象工厂模式的UML图:
问题9:什么是单例? 使整个方法同步还是只同步关键部分更好? ( 回答 )
Java中的Singleton是一个类,在整个Java应用程序中只有一个实例,例如, java.lang.Runtime
是Singleton类。 在Java 4之前,创建Singleton非常棘手,但是一旦Java 5引入Enum,它就非常容易。
问题10:您可以在Java 4和Java 5中编写用于遍历HashMap的代码吗? ( 回答 )
棘手的但他设法使用while和for循环来编写。 实际上,有四种方法可以遍历Java中的任何Map,一种涉及使用keySet()
并遍历key,然后使用get()方法来检索值,这有点昂贵。 第二种方法涉及使用entrySet()
并通过使用每个循环或在Iterator.hasNext()方法中对其进行迭代。 这是一种更好的方法,因为在迭代过程中键和值对象都可以使用,并且您无需调用get()方法来检索值,这在出现巨大链接列表的情况下可以提高O(n)的性能。一桶。 有关详细说明和代码示例,请参见我的文章4在Java中迭代Map的方法 。
问题11:什么时候覆盖hashCode()和equals()? ( 回答 )
只要有必要,尤其是如果您要基于业务逻辑而不是对象相等性进行相等性检查,例如,两个雇员对象具有相同的emp_id,则它们相等,尽管它们是由代码的不同部分创建的两个不同的对象。 如果要将它们用作HashMap中的键,则必须重写这两种方法。 现在,作为Java中的equals-hashcode合同的一部分,当您覆盖equals时,还must
覆盖hashcode,否则您的对象将不会破坏类的不变性,例如Set,Map依靠equals()方法正常运行。 您还可以查看有关Java中的equals的文章5技巧,以了解在处理这两种方法时可能出现的细微问题。
问题12: 如果不重写hashCode()方法将是什么问题? ( 回答 )
如果不重写equals方法,则equals和hashcode之间的协定将不起作用,根据该契约,由equals()相等的两个对象必须具有相同的hashcode。 在这种情况下,另一个对象可能返回不同的hashCode并将被存储在该位置,这破坏了HashMap类的不变性,因为它们不应允许重复的键。 当您使用put()方法添加对象时,将迭代该存储桶位置中存在的所有Map.Entry对象,并更新先前映射的值(如果Map已经包含该键)。 如果未覆盖哈希码,则此方法将无效。
问题13:同步getInstance()方法的关键部分或整个getInstance()方法是否更好? ( 回答 )
答案只是关键部分,因为如果我们锁定整个方法,那么每次有人调用此方法时,即使我们没有创建任何对象,它都必须等待。 换句话说,仅在创建对象时才需要同步,同步仅发生一次。 创建对象后,无需进行任何同步。 实际上,就性能而言,这是非常差的编码,因为同步方法最多可将性能降低10到20倍。 这是Singleton模式的UML图:
顺便说一下,有几种方法可以在Java中创建线程安全的单例,您也可以在此问题或任何后续操作中提及这些方法。
问题14:在get()操作期间,图片中的equals()和hashCode()方法在哪里? ( 回答 )
这个核心Java面试问题是对先前Java问题的后续,应聘者应该知道,一旦您提到hashCode,人们很可能会问他们如何在HashMap中使用它们。 提供键对象时,首先调用其hashcode方法来计算存储桶位置。 由于存储桶可能包含多个条目作为链接列表,因此通过使用equals()
方法评估每个Map.Entry
对象,以查看它们是否包含实际的键对象。
问题15:如何避免Java陷入僵局? ( 回答 )
您可以通过打破循环等待条件来避免死锁。 为此,您可以在代码中进行安排以将顺序强加于锁的获取和释放。 如果将以一致的顺序获取锁并以相反的顺序释放锁,则不会出现一个线程持有由另一线程获取的锁的情况,反之亦然。 有关代码示例和详细说明,请参见详细答案。
问题16:将String创建为new()和立即数有什么区别? ( 回答 )
当我们使用new()
运算符创建字符串时,它是在堆中创建的,而不是添加到字符串池中,而使用文字创建的字符串是在存在于堆的PermGen区域的字符串池本身中创建的。
String str = new String("Test")
不会将对象str放入字符串池,我们需要调用String.intern()
方法,该方法用于显式将它们放入字符串池。 只有将String对象创建为String文字(例如String s = "Test"
,Java才会自动将其放入String池。 顺便说一句,这里有一个问题,因为我们将参数传递为“ Test”(这是一个字符串文字),所以它还将在字符串池上创建另一个对象作为“ Test”。 这是一个要点,直到Javarevisited博客的博学的读者提出来之前,这一点才被注意到。 要了解更多关于字符串文字和String对象之间的差异,看到这个文章。
这是一个很好的图像,很好地显示了这种差异:
问题17:什么是不可变对象? 你可以写一个不可变的类吗? ( 回答 )
不可变类是Java类,其对象一旦创建就无法修改。 不可变对象中的任何修改都会产生新对象。 例如, String在Java中是不可变的 。 大多数情况下,不可变类在Java中也是最终的,以防止子类覆盖会损害不可变性的方法。 您可以通过将成员设置为非最终成员但私有并将其修改(除非在构造函数中),来实现相同的功能。
除了显而易见的形式外,您还需要确保不公开Immutable对象的内部,尤其是当它包含可变成员时。 同样,当您从客户端(例如java.util.Date
)接受可变成员的值时,请使用clone()方法为您自己保留一个单独的副本,以防止恶意客户端在设置可变引用后对其进行修改的风险。
返回可变成员的值,将另一个单独的副本返回给客户端,从不返回Immutable类持有的原始引用时,需要采取相同的预防措施。 您可以查看我的文章如何在Java中创建不可变类,以获取逐步指南和代码示例。
问题18:提供一种最简单的方法来找出方法执行所需的时间而无需使用任何分析工具? ( 回答 )
在调用方法之前和thr方法返回之后立即读取系统时间。 采取时间差,这将为您提供执行方法所花费的时间。
放入代码中...
long start = System.currentTimeMillis ();
method ();
long end = System.currentTimeMillis ();
System.out.println (“Time taken for execution is ” + (end – start));
请记住,如果执行时间太短,则可能表明执行时间为零毫秒。 尝试使用一种足够大的方法,从某种意义上说,它正在执行大量处理
问题19:要在HashMap中使用对象作为键,需要实现以下两种方法? ( 回答 )
为了将任何对象用作HashMap或Hashtable中的Key,它必须在Java中实现equals和hash-code方法。 阅读HashMap如何在Java中工作,以获取有关如何使用等号和哈希码方法从HashMap放置和获取对象的详细说明。
问题20:如何防止客户直接实例化您的具体类? 例如,您有一个Cache接口以及两个实现类MemoryCache和DiskCache,如何确保客户端没有使用new()关键字创建这两个类的对象。
我将这个问题留给您练习和思考,然后再给出答案。 我相信您可以找到正确的方法来执行此操作,因为从维护的角度来看,这是保持对类的控制的重要决定之一。
我也非常感谢我的读者,他们慷慨地为初学者和经验丰富的开发人员提供了Java Interviews中的几个好问题。 我已经在此博客中回答了许多这些问题,您可以使用此页面右上角的搜索框轻松找到相关的帖子。