当前位置: 移动技术网 > IT编程>开发语言>Java > 常见的java面试题

常见的java面试题

2019年07月22日  | 移动技术网IT编程  | 我要评论
本文主要为大家整理java常见的面试题,供大家参考,具体内容如下 1.  java中sleep和wait的区别 ① 这两个方法来自不同的类分别是,slee

本文主要为大家整理java常见的面试题,供大家参考,具体内容如下

1.  java中sleep和wait的区别

① 这两个方法来自不同的类分别是,sleep来自thread类,和wait来自object类。
sleep是thread的静态类方法,谁调用的谁去睡觉,即使在a线程里调用b的sleep方法,实际上还是a去睡觉,要让b线程睡觉要在b的代码中调用sleep。

② 锁: 最主要是sleep方法没有释放锁,而wait方法释放了锁,使得其他线程可以使用同步控制块或者方法。

sleep不出让系统资源;wait是进入线程等待池等待,出让系统资源,其他线程可以占用cpu。一般wait不会加时间限制,因为如果wait线程的运行资源不够,再出来也没用,要等待其他线程调用notify/notifyall唤醒等待池中的所有线程,才会进入就绪队列等待os分配系统资源。sleep(milliseconds)可以用时间指定使它自动唤醒过来,如果时间不到只能调用interrupt()强行打断。

thread.sleep(0)的作用是“触发操作系统立刻重新进行一次cpu竞争”。

③ 使用范围:wait,notify和notifyall只能在同步控制方法或者同步控制块里面使用,而sleep可以在任何地方使用。

 synchronized(x){ 
  x.notify() 
  //或者wait() 
 }

2.  java中hashmap和hashtable的区别

① 历史原因: hashtable是给予陈旧的dictonary类的,  hashmap是java1.2引进的map接口的一个实现

② hashmap允许空的键值对, 而hashtable不允许

③ hashtable同步,而hashmap非同步,效率上比hashtable要高

3. 请简述在异常当中,throw和throws有什么区别

① throw代表动作,表示抛出一个异常的动作;throws代表一种状态,代表方法可能有异常抛出
② throw用在方法实现中,而throws用在方法声明中
③ throw只能用于抛出一种异常,而throws可以抛出多个异常

4. 内存溢出和内存泄露的区别

内存溢出 out of memory,是指程序在申请内存时,没有足够的内存空间供其使用,出现out of memory;比如申请了一个integer,但给它存了long才能存下的数,那就是内存溢出。

内存泄露 memory leak,是指程序在申请内存后,无法释放已申请的内存空间,一次内存泄露危害可以忽略,但内存泄露堆积后果很严重,无论多少内存,迟早会被占光。

memory leak会最终会导致out of memory!

内存溢出就是你要求分配的内存超出了系统能给你的,系统不能满足需求,于是产生溢出。

    内存泄漏是指你向系统申请分配内存进行使用(new),可是使用完了以后却不归还(delete),结果你申请到的那块内存你自己也不能再访问(也许你把它的地址给弄丢了),而系统也不能再次将它分配给需要的程序。一个盘子用尽各种方法只能装4个果子,你装了5个,结果掉倒地上不能吃了。这就是溢出!比方说栈,栈满时再做进栈必定产生空间溢出,叫上溢,栈空时再做退栈也产生空间溢出,称为下溢。就是分配的内存不足以放下数据项序列,称为内存溢出。

   以发生的方式来分类,内存泄漏可以分为4类:

① 常发性内存泄漏。发生内存泄漏的代码会被多次执行到,每次被执行的时候都会导致一块内存泄漏。
② 偶发性内存泄漏。发生内存泄漏的代码只有在某些特定环境或操作过程下才会发生。常发性和偶发性是相对的。对于特定的环境,偶发性的也许就变成了常发性的。所以测试环境和测试方法对检测内存泄漏至关重要。
③ 一次性内存泄漏。发生内存泄漏的代码只会被执行一次,或者由于算法上的缺陷,导致总会有一块仅且一块内存发生泄漏。比如,在类的构造函数中分配内存,在析构函数中却没有释放该内存,所以内存泄漏只会发生一次。
④ 隐式内存泄漏。程序在运行过程中不停的分配内存,但是直到结束的时候才释放内存。严格的说这里并没有发生内存泄漏,因为最终程序释放了所有申请的内存。但是对于一个服务器程序,需要运行几天,几周甚至几个月,不及时释放内存也可能导致最终耗尽系统的所有内存。所以,我们称这类内存泄漏为隐式内存泄漏。

从用户使用程序的角度来看,内存泄漏本身不会产生什么危害,作为一般的用户,根本感觉不到内存泄漏的存在。真正有危害的是内存泄漏的堆积,这会最终消耗尽系统所有的内存。从这个角度来说,一次性内存泄漏并没有什么危害,因为它不会堆积,而隐式内存泄漏危害性则非常大,因为较之于常发性和偶发性内存泄漏它更难被检测到。

5. string,stringbuffer 和 stringbuilder的区别

①可变与不可变
  string类中使用字符数组保存字符串,如下就是,因为有“final”修饰符,所以可以知道string对象是不可变的。

    private final char value[];

  stringbuilder与stringbuffer都继承自abstractstringbuilder类,在abstractstringbuilder中也是使用字符数组保存字符串,如下就是,可知这两种对象都是可变的。

    char[] value;

②是否多线程安全

  string中的对象是不可变的,也就可以理解为常量,显然线程安全。

  abstractstringbuilder是stringbuilder与stringbuffer的公共父类,定义了一些字符串的基本操作,如expandcapacity、append、insert、indexof等公共方法。

  stringbuffer对方法加了同步锁或者对调用的方法加了同步锁,所以是线程安全的。看如下源码:

public synchronized stringbuffer reverse() { 
  super.reverse(); 
  return this; 
 } 
 
 public int indexof(string str) { 
  return indexof(str, 0);  //存在 public synchronized int indexof(string str, int fromindex) 方法 
 } 

stringbuilder并没有对方法进行加同步锁,所以是非线程安全的。

③stringbuilder与stringbuffer共同点

  stringbuilder与stringbuffer有公共父类abstractstringbuilder(抽象类)。

  抽象类与接口的其中一个区别是:抽象类中可以定义一些子类的公共方法,子类只需要增加新的功能,不需要重复写已经存在的方法;而接口中只是对方法的申明和常量的定义。

  stringbuilder、stringbuffer的方法都会调用abstractstringbuilder中的公共方法,如super.append(...)。只是stringbuffer会在方法上加synchronized关键字,进行同步。

  最后,如果程序不是多线程的,那么使用stringbuilder效率高于stringbuffer。

6. 数组和链表的区别

二者都属于一种数据结构

从逻辑结构来看:
① 数组必须事先定义固定的长度(元素个数),不能适应数据动态地增减的情况。当数据增加时,可能超出原先定义的元素个数;当数据减少时,造成内存浪费;数组可以根据下标直接存取。
② 链表动态地进行存储分配,可以适应数据动态地增减的情况,且可以方便地插入、删除数据项。(数组中插入、删除数据项时,需要移动其它数据项,非常繁琐)链表必须根据next指针找到下一个元素。

从内存存储来看:
①(静态)数组从栈中分配空间,对于程序员方便快速,但是自由度小。
②链表从堆中分配空间,自由度大但是申请管理比较麻烦。
从上面的比较可以看出,如果需要快速访问数据,很少或不插入和删除元素,就应该用数组;相反,如果需要经常插入和删除元素就需要用链表数据结构了。

7. arraylist和linkedlist的区别

①arraylist是实现了基于动态数组的数据结构,linkedlist基于链表的数据结构。
②对于随机访问get和set,arraylist觉得优于linkedlist,因为linkedlist要移动指针。
③对于新增和删除操作add和remove,linedlist比较占优势,因为arraylist要移动数据。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持移动技术网。

如您对本文有疑问或者有任何想说的,请 点击进行留言回复,万千网友为您解惑!

相关文章:

验证码:
移动技术网