Java面试宝典2016版
//由于这里的 Thread1和下面的 Thread2内部 run 方法要用同一对象作为监视器,我们这里不能用 this,因为在 Thread2里面的 this 和这个 Thread1的 this 不是同一个对象。我们用MultiThread.class 这个字节码对象,当前虚拟机里引用这个变量时,指向的都是同一个对象。 synchronized
(MultiThread.class){System.out.println(\is waiting\
page39image7408page39image7568 try {
//释放锁有两种方式,第一种方式是程序自然离开监视器的范围,也就是离开了 synchronized 关键字管辖的代码范围,另一种方式就是在 synchronized 关键字管辖的代码内部调用监视器对象的 wait 方法。这里,使用 wait 方法释放锁。 MultiThread.class.wait(); } catch(InterruptedException e) { // TODO Auto-generatedcatch block e.printStackTrace();}
System.out.println(\System.out.println(\}}
private static classThread2implements Runnable{ @Override public void run() {
// TODO Auto-generated methodstubsynchronized (MultiThread.class){ System.out.println(\
System.out.println(\
//由于 notify 方法并不释放锁,即使 thread2调用下面的 sleep 方法休息了10毫秒,但 thread1仍然不会执行,因为 thread2没有释放锁,所以 Thread1无法得不到锁。
MultiThread.class.notify();
System.out.println(\Thread.sleep(10);
} catch (InterruptedExceptione) { // TODO Auto-generatedcatch block e.printStackTrace();}
36
Java面试宝典2016版
System.out.println(\over!\}} }}
going on...\being
48、同步和异步有何异同,在什么情况下分别使用他们?举例说明。
如果数据将在线程间共享。例如正在写的数据以后可能被另一个线程读到,或者正在读的数据可能已经被另一个线程写过了,那么这些数据就是共享数据,必须进行同步存取。当应用程序在对象上调用了一个需要花费很长时间来执行的方法,并且不希望让程序等待方法的返回时,就应该使用异步编程,在很多情况下采用异步途径往往更有效率。 49. 下面两个方法同步吗? (自己发明)class Test {
synchronizedstatic void say Hello3() { }
synchronizedvoid getX(){}}
50、多线程有几种实现方法?同步有几种实现方法?
多线程有两种实现方法,分别是继承 Thread 类与实现 Runnable 接口 同步的实现方面有两种,分别是 synchronized,wait 与 notify wait():使一个线程处于等待状态,并且释放所持有的对象的 lock。 sleep():使一个正在运行的线程处于睡眠状态,是一个静态方法,调用此方法要捕捉InterruptedException(中断异常)异常。
notify():唤醒一个处于等待状态的线程,注意的是在调用此方法的时候,并不能确切的唤醒某一个等待状态的线程,而是由 JVM 确定唤醒哪个线程,而且不是按优先级。
Allnotity():唤醒所有处入等待状态的线程,注意并不是给所有唤醒线程一个对象的锁,而是让它们竞争。
51、启动一个线程是用run()还是start()? .
启动一个线程是调用 start()方法,使线程就绪状态,以后可以被调度为运行状态,一个线程必须关联一些具体的执行代码,run()方法是该线程所关联的执行代码。
37
Java面试宝典2016版
52、当一个线程进入一个对象的一个 synchronized 方法后,其它线程是否可进入此对象的其它方法? 分几种情况:
1. 其他方法前是否加了 synchronized 关键字,如果没加,则能。 2. 如果这个方法内部调用了 wait,则可以进入其他 synchronized 方法。 3. 如果其他每个方法都加了 synchronized 关键字,并且内部没有调用 wait,则不能。
4. 如果其他方法是 static,它用的同步锁是当前类的字节码,与非静态的方法不能同步,因为非静态的方法用的是 this。
53、线程的基本概念、线程的基本状态以及状态之间的关系
一个程序中可以有多条执行线索同时执行,一个线程就是程序中的一条执行线索,每个线程上都关联有要执行的代码,即可以有多段程序代码同时运行,每个程序至少都有一个线程,即 main 方法执行的那个线程。如果只是一个 cpu,它怎么能够同时执行多段程序呢?这是从宏观上来看的,cpu 一会执行 a 线索,一会执行 b 线索,切换时间很快,给人的感觉是a,b 在同时执行,好比大家在同一个办公室上网,只有一条链接到外部网线,其实,这条网线一会为 a 传数据,一会为 b 传数据,由于切换时间很短暂,所以,大家感觉都在同时上网。 状态:就绪,运行,synchronize 阻塞,wait 和 sleep 挂起,结束。wait 必须在 synchronized内部调用。
调用线程的 start 方法后线程进入就绪状态,线程调度系统将就绪状态的线程转为运行状态,遇到 synchronized 语句时,由运行状态转为阻塞,当 synchronized 获得锁后,由阻塞转为运行,在这种情况可以调用 wait 方法转为挂起状态,当线程关联的代码执行完后,线程变为结束状态。
54、简述 synchronized 和 java.util.concurrent.locks.Lock 的异同?
主要相同点:Lock 能完成 synchronized 所实现的所有功能
主要不同点:Lock 有比 synchronized 更精确的线程语义和更好的性能。synchronized 会自动释放锁,而 Lock 一定要求程序员手工释放,并且必须在 finally 从句中释放。Lock 还有更强大的功能,例如,它的 tryLock 方法可以非阻塞方式去拿锁。
举例说明(对下面的题用 lock 进行了改写):package com.huawei.interview; import java.util.concurrent.locks.Lock; import java.util.concurrent.locks.ReentrantLock; publicclass ThreadTest {
38
Java面试宝典2016版
/**
* @paramargs*/ private int j;
private Lock lock =newReentrantLock();public static voidmain(String[] args) { // TODO Auto-generated method stubThreadTest tt = new ThreadTest();for(int i=0;i<2;i++) {
new Thread(tt.new Adder()).start(); new Thread(tt.new Subtractor()).start();} }
private class SubtractorimplementsRunnable{ @Override public void run() {
// TODO Auto-generated methodstubwhile(true) {
/*synchronized (ThreadTest.this) {System.out.println(\这里抛异常了,锁能释放吗? }*/
lock.lock();try {
System.out.println(\}finally{ lock.unlock();} }} }
private class AdderimplementsRunnable{ @Override public void run() {
// TODO Auto-generated methodstubwhile(true) {
/*synchronized (ThreadTest.this) {System.out.println(\lock.lock(); try {
System.out.println(\
39
Java面试宝典2016版
}finally{ lock.unlock();} } } }}
55、设计 4 个线程,其中两个线程每次对 j 增加 1,另外两个线程对 j 每次减少1。写出程序。
以下程序使用内部类实现线程,对 j 增减的时候没有考虑顺序问题。public class ThreadTest1 {
private int j;
public static void main(String args[]){ThreadTest1 tt=newThreadTest1();Inc inc=tt.new Inc();
Dec dec=tt.new Dec();for(inti=0;i<2;i++){ Thread t=newThread(inc);t.start(); t=new Thread(dec);t.start(); }}
private synchronized void inc(){j++;
System.out.println(Thread.currentThread().getName()+\:\private synchronized void dec(){j--;
System.out.println(Thread.currentThread().getName()+\:\class Inc implements Runnable{
public void run(){for(inti=0;i<100;i++){inc(); } }}
class Dec implements Runnable{public void run(){ for(inti=0;i<100;i++){dec(); } }} }
----------随手再写的一个-------------class A {
JManger j =new JManager();main() {
40