16、多线程的单例设计模式:保证某个类中内存中只有一个对象 (1)饿汉式: class Single { private Single(){}//将构造函数私有化,不让别的类建立该类对象 private static final Single s=new Single();//自己建立一个对象 public static Single getInstance()//提供一个公共访问方式 { return s; } } (2)懒汉式: class Single { private Single(){} private static Single s; public static Single getInstance() { if(s==null) s=new Single(); return s; } } 饿汉式和懒汉式的区别: ** 饿汉式是类一加载进内存就创建好了对象; 懒汉式则是类加载进内存的时候,对象还没有存在,只有调用了getInstance()方法时,对象才开始创建。 ** 懒汉式是延迟加载,如果多个线程同时操作懒汉式时就有可能出现线程安全问题,解决线程安全问题 可以加同步来解决。但是加了同步之后,每一次都要比较锁,效率就变慢了, 所以可以加双重判断来提高程序效率。 如将上述懒汉式的Instance函数改成同步: public static Single getInstance() { if(s==null) { synchronized(Single.class) { if(s==null) s=new Single(); } } return s; } 17、死锁 两个线程对两个同步对象具有循环依赖时,就会发生死锁。即同步嵌套同步,而锁却不同。 18、wait()、sleep()、notify()、notifyAll() wait():使一个线程处于等待状态,并且释放所持有的对象的lock。 sleep():使一个正在运行的线程处于睡眠状态,是一个静态方法,调用此方法要捕捉InterruptedException异常。 notify():唤醒一个处于等待状态的线程,注意的是在调用此方法的时候,并不能确切的唤醒某一个等待状态的线程, 而是由JVM确定唤醒哪个线程(一般是最先开始等待的线程),而且不是按优先级。 Allnotity():唤醒所有处入等待状态的线程,注意并不是给所有唤醒线程一个对象的锁,而是让它们竞争。 18、为什么wait()、notify()、notifyAll()这些用来操作线程的方法定义在Object类中? (1)这些方法只存在于同步中; (2)使用这些方法时必须要指定所属的锁,即被哪个锁调用这些方法; (3)而锁可以是任意对象,所以任意对象调用的方法就定义在Object中。 19、多线程间通讯: 多线程间通讯就是多个线程在操作同一资源,但是操作的动作不同. (1)为什么要通信 多线程并发执行的时候, 如果需要指定线程等待或者唤醒指定线程, 那么就需要通信.比如生产者消费者的问题, 生产一个消费一个,生产的时候需要负责消费的进程等待,生产一个后完成后需要唤醒负责消费的线程, 同时让自己处于等待,消费的时候负责消费的线程被唤醒,消费完生产的产品后又将等待的生产线程唤醒, 然后使自己线程处于等待。这样来回通信,以达到生产一个消费一个的目的。 (2)怎么通信 在同步代码块中, 使用锁对象的wait()方法可以让当前线程等待, 直到有其他线程唤醒为止. 使用锁对象的notify()方法可以唤醒一个等待的线程,或者notifyAll唤醒所有等待的线程. 多线程间通信用sleep很难实现,睡眠时间很难把握。 20、Lock和Condition 实现提供比synchronized方法和语句可获得的更广泛的锁的操作,可支持多个相关的Condition对象 Lock是个接口 锁是控制多个线程对共享数据进行访问的工具。 JDK1.5中提供了多线程升级的解决方案: 将同步synchonized替换成了显示的Lock操作,将Object中的wait、notify、notifyAll替换成了Condition对象。 该对象可以Lock锁进行获取 Lock的方法摘要: void lock() 获取锁。 Condition newCondition() 返回绑定到此 Lock 实例的新 Condition 实例。 void unlock() 释放锁。 Condition方法摘要: void await() 造成当前线程在接到信号或被中断之前一直处于等待状态。 void signal() 唤醒一个等待线程。 void signalAll() 唤醒所有等待线程。 21、停止线程: stop方法已经过时,如何停止线程? 停止线程的方法只有一种,就是run方法结束。如何让run方法结束呢? 开启多线程运行,运行代码通常是循环体,只要控制住循环,就可以让run方法结束,也就是结束线程。 特殊情况:当线程属于冻结状态,就不会读取循环控制标记,则线程就不会结束。 为解决该特殊情况,可引入Thread类中的Interrupt方法结束线程的冻结状态; 当没有指定的方式让冻结线程恢复到运行状态时,需要对冻结进行清除,强制让线程恢复到运行状态
|