Java编程并发面试题及答案_极悦注册
专注Java教育14年 全国咨询/投诉热线:444-1124-454
极悦LOGO图
始于2009,口口相传的Java黄埔军校
首页 学习攻略 Java学习 Java编程并发面试题及答案

Java编程并发面试题及答案

更新时间:2019-12-24 13:47:03 来源:极悦 浏览1944次

1、在java中守护线程和本地线程区别?

  java中的线程分为两种:守护线程(Daemon)和用户线程(User)。

  任何线程都可以设置为守护线程和用户线程,通过方法Thread.setDaemon(bool on);true则把该线程设置为守护线程,反之则为用户线程。Thread.setDaemon()必须在Thread.start()之前调用,否则运行时会抛出异常。

  两者的区别:

  唯一的区别是判断虚拟机(JVM)何时离开,Daemon是为其他线程提供服务,如果全部的User Thread已经撤离,Daemon 没有可服务的线程,JVM撤离。也可以理解为守护线程是JVM自动创建的线程(但不一定),用户线程是程序创建的线程;比如JVM的垃圾回收线程是一个守护线程,当所有线程已经撤离,不再产生垃圾,守护线程自然就没事可干了,当垃圾回收线程是Java虚拟机上仅剩的线程时,Java虚拟机会自动离开。

  扩展:Thread Dump打印出来的线程信息,含有daemon字样的线程即为守护进程,可能会有:服务守护进程、编译守护进程、windows下的监听Ctrl+break的守护进程、Finalizer守护进程、引用处理守护进程、GC守护进程。

  2、线程与进程的区别?

  进程是操作系统分配资源的最小单元,线程是操作系统调度的最小单元。

  一个程序至少有一个进程,一个进程至少有一个线程。

  3、什么是多线程中的上下文切换?

  多线程会共同使用一组计算机上的CPU,而线程数大于给程序分配的CPU数量时,为了让各个线程都有执行的机会,就需要轮转使用CPU。不同的线程切换使用CPU发生的切换数据等就是上下文切换。

  4、死锁与活锁的区别,死锁与饥饿的区别?

  死锁:是指两个或两个以上的进程(或线程)在执行过程中,因争夺资源而造成的一种互相等待的现象,若无外力作用,它们都将无法推进下去。

  产生死锁的必要条件:

  互斥条件:所谓互斥就是进程在某一时间内独占资源。

  请求与保持条件:一个进程因请求资源而阻塞时,对已获得的资源保持不放。

  不剥夺条件:进程已获得资源,在末使用完之前,不能强行剥夺。

  循环等待条件:若干进程之间形成一种头尾相接的循环等待资源关系。

  活锁:任务或者执行者没有被阻塞,由于某些条件没有满足,导致一直重复尝试,失败,尝试,失败。

  活锁和死锁的区别在于,处于活锁的实体是在不断的改变状态,所谓的“活”, 而处于死锁的实体表现为等待;活锁有可能自行解开,死锁则不能。

  饥饿:一个或者多个线程因为种种原因无法获得所需要的资源,导致一直无法执行的状态。

  Java中导致饥饿的原因:

  高优先级线程吞噬所有的低优先级线程的CPU时间。

  线程被永久堵塞在一个等待进入同步块的状态,因为其他线程总是能在它之前持续地对该同步块进行访问。

  线程在等待一个本身也处于永久等待完成的对象(比如调用这个对象的wait方法),因为其他线程总是被持续地获得唤醒。

  5、Java中用到的线程调度算法是什么?

  采用时间片轮转的方式。可以设置线程的优先级,会映射到下层的系统上面的优先级上,如非特别需要,尽量不要用,防止线程饥饿。

  6、什么是线程组,为什么在Java中不推荐使用?

  ThreadGroup类,可以把线程归属到某一个线程组中,线程组中可以有线程对象,也可以有线程组,组中还可以有线程,这样的组织结构有点类似于树的形式。

  为什么不推荐使用?因为使用有很多的安全隐患吧,没有具体追究,如果需要使用,推荐使用线程池。

  7、为什么使用Executor框架?

  每次执行任务创建线程 new Thread()比较消耗性能,创建一个线程是比较耗时、耗资源的。

  调用 new Thread()创建的线程缺乏管理,被称为野线程,而且可以无限制的创建,线程之间的相互竞争会导致过多占用系统资源而导致系统瘫痪,还有线程之间的频繁交替也会消耗很多系统资源。

  接使用new Thread() 启动的线程不利于扩展,比如定时执行、定期执行、定时定期执行、线程中断等都不便实现。

  8、在Java中Executor和Executors的区别?

  Executors 工具类的不同方法按照我们的需求创建了不同的线程池,来满足业务的需求。

  Executor 接口对象能执行我们的线程任务。

  ExecutorService接口继承了Executor接口并进行了扩展,提供了更多的方法我们能获得任务执行的状态并且可以获取任务的返回值。

  使用ThreadPoolExecutor 可以创建自定义线程池。

  Future 表示异步计算的结果,他提供了检查计算是否完成的方法,以等待计算的完成,并可以使用get()方法获取计算的结果。

  9、什么是原子操作?在Java Concurrency API中有哪些原子类(atomic classes)?

  原子操作(atomic operation)意为”不可被中断的一个或一系列操作” 。

  处理器使用基于对缓存加锁或总线加锁的方式来实现多处理器之间的原子操作。

  在Java中可以通过锁和循环CAS的方式来实现原子操作。 CAS操作——Compare & Set,或是 Compare & Swap,现在几乎所有的CPU指令都支持CAS的原子操作。

  原子操作是指一个不受其他操作影响的操作任务单元。原子操作是在多线程环境下避免数据不一致必须的手段。

  int++并不是一个原子操作,所以当一个线程读取它的值并加1时,另外一个线程有可能会读到之前的值,这就会引发错误。

  为了解决这个问题,必须保证增加操作是原子的,在JDK1.5之前我们可以使用同步技术来做到这一点。到JDK1.5,java.util.concurrent.atomic包提供了int和long类型的原子包装类,它们可以自动的保证对于他们的操作是原子的并且不需要使用同步。

  java.util.concurrent这个包里面提供了一组原子类。其基本的特性就是在多线程环境下,当有多个线程同时执行这些类的实例包含的方法时,具有排他性,即当某个线程进入方法,执行其中的指令时,不会被其他线程打断,而别的线程就像自旋锁一样,一直等到该方法执行完成,才由JVM从等待队列中选择一个另一个线程进入,这只是一种逻辑上的理解。

  原子类:AtomicBoolean,AtomicInteger,AtomicLong,AtomicReference

  原子数组:AtomicIntegerArray,AtomicLongArray,AtomicReferenceArray

  原子属性更新器:AtomicLongFieldUpdater,AtomicIntegerFieldUpdater,AtomicReferenceFieldUpdater

  解决ABA问题的原子类:AtomicMarkableReference(通过引入一个boolean来反映中间有没有变过),AtomicStampedReference(通过引入一个int来累加来反映中间有没有变过)

  10、Java Concurrency API中的Lock接口(Lock interface)是什么?对比同步它有什么优势?

  Lock接口比同步方法和同步块提供了更具扩展性的锁操作。

  他们允许更灵活的结构,可以具有完全不同的性质,并且可以支持多个相关类的条件对象。

  它的优势有:

  可以使锁更公平

  可以使线程在等待锁的时候响应中断

  可以让线程尝试获取锁,并在无法获取锁的时候立即返回或者等待一段时间

  可以在不同的范围,以不同的顺序获取和释放锁

  整体上来说Lock是synchronized的扩展版,Lock提供了无条件的、可轮询的(tryLock方法)、定时的(tryLock带参方法)、可中断的(lockInterruptibly)、可多条件队列的(newCondition方法)锁操作。另外Lock的实现类基本都支持非公平锁(默认)和公平锁,synchronized只支持非公平锁,当然,在大部分情况下,非公平锁是高效的选择。

Java编程并发面试题及答案

      以上就是极悦注册机构小编介绍的“Java编程并发面试题及答案”的内容,希望对大家有帮助,如有疑问,请在线咨询,有专业老师随时为你服务。

相关推荐

(初级到高级)

提交申请后,顾问老师会电话与您沟通安排学习

免费课程推荐 >>
技术文档推荐 >>