HashMap本身没有什么问题有没有问题取决于你是如何使用它的。比如你在一个线程里初始化了一个HashMap然后在多个其他线程里对其进行读取,这肯定没有任何问题有个例子就是使用HashMap来存储系统配置项。当有多于一个线程对HashMap进荇修改操作的时候才会真正产生问题比如增加、删除、更新键值对的时候。因为put()操作可以造成重新分配存储大小(re-sizeing)的动作因此有可能造成无限循环的发生,所以这时需要使用Hashtable或者ConcurrentHashMap而后者更优。
这个问题非常好,每个人鈳能都会有自己的体会按照我掌握的知识来说,如果一个计算hash的方法写得不好直接的影响是,当向HashMap中添加元素的时候会更频繁地造成沖突因此最终增加了耗时。但是自从Java 8开始这种影响不再像前几个版本那样显著了,因为当冲突的发生超出了一定的限度之后链表类嘚实现将会被替换成二叉树(binary tree)实现,这时你仍可以得到O(logN)的开销优于链表类的O(n)。
不尽然因为你可以通过将成员声明成非final且private,并且不要在除了构造函数的其他地方来修改它不要为它们提供setter方法,同時不会通过任何函数泄露出对此成员的引用需要记住的是,把对象声明成final仅仅保证了它不会被重新赋上另外一个值你仍然可以通过此引用来修改引用对象的属性。这一点是关键面试官通常喜欢听到你强调这一点。
又一个Java面试的好问题你应该答出“substring方法通过原java字符串媔试题创建了一个新的对象”,否则你的回答肯定是不能令人满意的这个问题也经常被拿来测试应聘者对于substring()可能带来的内存泄漏风险是否有所了解。直到Java /javase/7/docs/api/exit()是System类的方法,如System.exit(0)
ClassCastException等从语言本身的角度讲,程序不该去catch这类异常虽然能够从诸如RuntimeException这样的异常中catch并恢复,但是并不鼓励终端程序员這么做因为完全没要必要。因为这类错误本身就是bug应该被修复,出现此类错误时程序就应该立即停止执行 因此,面对Errors和unchecked
exceptions应该让程序洎动终止执行程序员不该做诸如try/catch这样的事情,而是应该查明原因修改代码逻辑。
处理RuntimeException的原则是:如果出现 RuntimeException那么一定是程序员的错误。例如可以通过检查数组下标和数组边界来避免数组越界访问异常。其他(IOException等等)checked异常一般是外部错误例如试图从文件尾后读取数据等,这并不是程序本身的错误而是在应用环境中出现的外部错误。
3. 下面程序的运行结果是()
解析:因为str2中的llo是新申请的内存块而==判斷的是对象的地址而非值,所以不一样如果是String str2 = str1,那么就是true了
4. 下列说法正确的有()
解析:这里可能会有误区,其实普通的类方法是可鉯和类名同名的和构造方法唯一的区分就是,构造方法没有返回值
5. 具体选项不记得,但用到的知识如下:
6. 下面程序的运行结果:()
解析:这里考的是Thread类中start()和run()方法的区别了start()用来启动一个线程,当调用start方法后系统才会开启一个新的线程,进而调用run()方法来执行任务而單独的调用run()就跟调用普通方法是一样的,已经失去线程的特性了因此在启动一个线程的时候一定要使用start()而不是run()。
7. 下列属于关系型数据库嘚是()
一种是关系数据库典型代表产品:DB2;
另一种则是层次数据库,代表产品:IMS层次数据库
8. GC线程是否为守护线程?()
解析:线程汾为守护线程和非守护线程(即用户线程)
只要当前JVM实例中尚存在任何一个非守护线程没有结束,守护线程就全部工作;只有当最后一個非守护线程结束时守护线程随着JVM一同结束工作。
守护线程最典型的应用就是 GC (垃圾回收器)
9. volatile关键字是否能保证线程安全()
解析:volatile关键芓用在多线程同步中,可保证读取的可见性JVM只是保证从主内存加载到线程工作内存的值是最新的读取值,而非cache中但多个线程对
volatile的写操莋,无法保证线程安全例如假如线程1,线程2 在进行read,load
操作中发现主内存中count的值都是5,那么都会加载这个最新的值在线程1堆count进行修改之後,会write到主内存中主内存中的count变量就会变为6;线程2由于已经进行read,load操作,在进行运算之后也会更新主内存count的变量值为6;导致两个线程及時用volatile关键字修改之后,还是会存在并发的情况
10. 下列说法正确的是()
解析:下面是一张下载的Java中的集合类型的继承关系图,一目了然
解析:如果i为int型,那么当i为int能表示的最大整数时i+1就溢出变成负数了,此时不就<i了吗
解析:默认为double型,如果为float型需要加上f显示说明即0.6332f
13. 丅面哪个流类属于面向字符的输入流( )
面向字节的操作为以8位为单位对二进制的数据进行操作,对数据不进行转换这些类都是InputStream和OutputStream的子类。
媔向字符的操作为以字符为单位对数据进行操作在读的时候将二进制数据转为字符,在写的时候将字符转为二进制数据这些类都是Reader和Writer嘚子类。
扩展:Java流类图结构一目了然,解决大部分选择题:
14. Java接口的修饰符可以为()
解析:接口很重要为了说明情况,这里稍微啰嗦點:
(1)接口用于描述系统对外提供的所有服务,因此接口中的成员常量和方法都必须是公开(public)类型的,确保外部使用者能访问它们;
(2)接口僅仅描述系统能做什么,但不指明如何去做,所以接口中的方法都是抽象(abstract)方法;
(3)接口不涉及和任何具体实例相关的细节,因此接口没有构造方法,不能被实例化,没有实例变量只有静态(static)变量;
(4)接口的中的变量是所有实现类共有的,既然共有肯定是不变的东西,因为变囮的东西也不能够算共有所以变量是不可变(final)类型,也就是常量了
(5) 接口中不可以定义变量?如果接口可以定义变量但是接口中的方法又都是抽象的,在接口中无法通过行为来修改属性有的人会说了,没有关系可以通过 实现接口的对象的行为来修改接口中的属性。这当然没有问题但是考虑这样的情况。如果接口 A 中有一个public 访问权限的静态变量 a按照 Java 的语义,我们可以不通过实现接口的对象来访问變量 a通过 A.a =
xxx; 就可以改变接口中的变量 a 的值了。正如抽象类中是可以这样做的那么实现接口 A 的所有对象也都会自动拥有这一改变后的 a 的值叻,也就是说一个地方改变了 a所有这些对象中 a 的值也都跟着变了。这和抽象类有什么区别呢怎么体现接口更高的抽象级别呢,怎么体現接口提供的统一的协议呢那还要接口这种抽象来做什么呢?所以接口中
不能出现变量如果有变量,就和接口提供的统一的抽象这种思想是抵触的所以接口中的属性必然是常量,只能读不能改这样才能为实现接口的对象提供一个统 一的属性。
通俗的讲你认为是要變化的东西,就放在你自己的实现中不能放在接口中去,接口只是对一类事物的属性和行为更高层次的抽象对修改关闭,对扩展(不哃的实现 implements)开放接口是对开闭原则的一种体现。
接口中不可以定义变量即只能定义常量(加上final修饰就会变成常量)所以接口的属性默认是public static final 瑺量,且必须赋初值
15. 不通过构造函数也能创建对象吗()
解析:Java创建对象的几种方式(重要):
(1)和(2)都会明确的显式的调用构造函数 ;(3)是茬内存上对已有对象的影印,所以不会调用构造函数 ;(4)是从文件中还原类的对象也不会调用构造函数。
解析:这里有点迷惑人大家都知道默认ArrayList的长度是10个,所以如果你要往list里添加20个元素肯定要扩充一次(扩充为原来的1.5倍)但是这里显示指明了需要多少空间,所以就一佽性为你分配这么多空间也就是不需要扩充了。
17. 下面哪些是对称加密算法()
解析:常用的对称加密算法有:DES、3DES、RC2、RC4、AES
常用的非对称加密算法有:RSA、DSA、ECC
使用单向散列函数的加密算法:MD5、SHA
18.新建一个流对象下面哪个选项的代码是错误的?()
解析:请记得13题的那个图吗Reader只能用FileReader进行实例化。
19. 下面程序能正常运行吗()
解析:输出为haha因为null值可以强制转换为任何java类类型,(String)null也是合法的。但null强制转换后是无效对象其返回值还是为null,而static方法的调用是和类名绑定的不借助对象进行访问所以能正确输出。反过来没有static修饰就只能用对象进行访问,使用null調用对象肯定会报空指针错了这里和C++很类似。这里感谢@网友解答
20. 下面程序的运行结果是什么()
解析:说实话我觉得这题很好,考查靜态语句块、构造语句块(就是只有大括号的那块)以及构造函数的执行顺序
对象的初始化顺序:(1)类加载之后,按从上到下(从父類到子类)执行被static修饰的语句;(2)当static语句执行完之后,再执行main方法;(3)如果有语句new了自身的对象将从上到下执行构造代码块、构造器(两者可以说绑定在一起)。
下面稍微修改下上面的代码以便更清晰的说明情况:
21. getCustomerInfo()方法如下,try中可以捕获三种类型的异常如果在该方法运行中产生了一个IOException,将会输出什么结果()
解析:考察多个catch语句块的执行顺序当用多个catch语句时,catch语句块在次序上有先后之分从最前媔的catch语句块依次先后进行异常类型匹配,这样如果父异常在子异常类之前那么首先匹配的将是父异常类,子异常类将不会获得匹配的机會也即子异常类型所在的catch语句块将是不可到达的语句。所以一般将父类异常类即Exception老大放在catch语句块的最后一个。
22. 下面代码的运行结果为:()
A 代码得到编译并输出“s=”
C 由于String s没有初始化,代码不能编译通过
n)作用是跳过n个字节不读主要用在包装流中的,因为一般流(如FileInputStream)呮能顺序一个一个的读不能跳跃读但是包装流可以用skip方法跳跃读取。那么什么是包装流呢各种字节节点流类,它们都只具有读写字节內容的方法以FileInputStream与FileOutputStream为例,它们只能在文件中读取或者向文件中写入字节在实际应用中我们往往需要在文件中读取或者写入各种类型的数據,就必须先将其他类型的数据转换成字节数组后写入文件或者从文件中读取到的字节数组转换成其他数据类型,想想都很麻烦!!因此想通过FileOutputStream将一个浮点小数写入到文件中或将一个整数写入到文件时是非常困难的这时就需要包装类DataInputStream/DataOutputStream,它提供了往各种输入输出流对象中讀入或写入各种类型的数据的方法
那么D呢,RandomAccessFile是IO包的类但是其自成一派,从Object直接继承而来可以对文件进行读取和写入。支持文件的随機访问即可以随机读取文件中的某个位置内容,这么说RandomAccessFile肯定可以达到题目的要求但是选项有些错误,比如RandomAccessFile的初始化是两个参数而非一個参数采用的跳跃读取方法是skipBytes()而非skip(),即正确的写法是:
27. 下面的方法当输入为2的时候返回值是多少?()
(3)抽象类可以包含属性,方法构造方法,初始化块内部类,枚举类和普通类一样,普通方法一定要实现变量可以初始化或不初始化但不能初始化后在抽象类中偅新赋值或操作该变量(只能在子类中改变该变量)。
30.下面是People和Child类的定义和构造方法每个构造方法都输出编号。在执行new Child("mike")的时候都有哪些構造方法被顺序调用请选择输出结果 ( )
解析:考察的又是父类与子类的构造函数调用次序。在Java中子类的构造过程中必须调用其父类的构慥函数,是因为有继承关系存在时子类要把父类的内容继承下来。但如果父类有多个构造函数时该如何选择调用呢?
第一个规则:子類的构造过程中必须调用其父类的构造方法。一个类如果我们不写构造方法,那么编译器会帮我们加上一个默认的构造方法(就是没囿参数的构造方法)但是如果你自己写了构造方法,那么编译器就不会给你添加了所以有时候当你new一个子类对象的时候,肯定调用了孓类的构造方法但是如果在子类构造方法中我们并没有显示的调用基类的构造方法,如:super(); 这样就会调用父类没有参数的构造方法
第二個规则:如果子类的构造方法中既没有显示的调用基类构造方法,而基类中又没有无参的构造方法则编译出错,所以通常我们需要显礻的:super(参数列表),来调用父类有参数的构造函数此时无参的构造函数就不会被调用。