思诚科技 seecen LOGO
咨询热线:0791-87557233
  首页 |   Java EE  
关于思诚
关注官方微信

浅析Java多线程编程中的高级技术

来源:网络    更新时间:2014-12-3


  线程组

  线程是被个别创建的,但可以将它们归类到线程组中,以便于调试和监视。只能在创建线程的同时将它与一个线程组相关联。在使用大量线程的程序中,使用线程组组织线程可能很有帮助。可以将它们看作是计算机上的目录和文件结构。

  线程间发信

  当线程在继续执行前需要等待一个条件时,仅有synchronized关键字是不够的。虽然synchronized关键字阻止并发更新一个对象,但它没有实现线程间发信。Object类为此提供了三个函数:wait()、notify()和notifyAll()。以全球气候预测程序为例。这些程序通过将地球分为许多单元,在每个循环中,每个单元的计算都是隔离进行的,直到这些值趋于稳定,然后相邻单元之间就会交换一些数据。所以,从本质上讲,在每个循环中各个线程都必须等待所有线程完成各自的任务以后才能进入下一个循环。这个模型称为屏蔽同步,下例说明了这个模型:

  屏蔽同步

publicclassBSync{
 inttotalThreads;
 intcurrentThreads;

 publicBSync(intx){
  totalThreads=x;
  currentThreads=0;
 }

 publicsynchronizedvoidwaitForAll(){
  currentThreads ;
  if(currentThreads<totalThreads){
   try{
    wait();
   }catch(Exceptione){}
  }
  else{
   currentThreads=0;
   notifyAll();
  }
 }
}
  当对一个线程调用wait()时,该线程就被有效阻塞,只到另一个线程对同一个对象调用notify()或notifyAll()为止。因此,在前一个示例中,不同的线程在完成它们的工作以后将调用waitForAll()函数,最后一个线程将触发notifyAll()函数,该函数将释放所有的线程。第三个函数notify()只通知一个正在等待的线程,当对每次只能由一个线程使用的资源进行访问限制时,这个函数很有用。但是,不可能预知哪个线程会获得这个通知,因为这取决于Java虚拟机(JVM)调度算法。

  将CPU让给另一个线程

  当线程放弃某个稀有的资源(如数据库连接或网络端口)时,它可能调用yield()函数临时降低自己的优先级,以便某个其他线程能够运行。

  守护线程

  有两类线程:用户线程和守护线程。用户线程是那些完成有用工作的线程。守护线程是那些仅提供辅助功能的线程。Thread类提供了setDaemon()函数。Java程序将运行到所有用户线程终止,然后它将破坏所有的守护线程。在Java虚拟机(JVM)中,即使在main结束以后,如果另一个用户线程仍在运行,则程序仍然可以继续运行。

  避免不提倡使用的方法

  不提倡使用的方法是为支持向后兼容性而保留的那些方法,它们在以后的版本中可能出现,也可能不出现。Java多线程支持在版本1.1和版本1.2中做了重大修订,stop()、suspend()和resume()函数已不提倡使用。这些函数在JVM中可能引入微妙的错误。虽然函数名可能听起来很诱人,但请抵制诱惑不要使用它们。

  调试线程化的程序

  在线程化的程序中,可能发生的某些常见而讨厌的情况是死锁、活锁、内存损坏和资源耗尽。

  死锁

  死锁可能是多线程程序最常见的问题。当一个线程需要一个资源而另一个线程持有该资源的锁时,就会发生死锁。这种情况通常很难检测。但是,解决方案却相当好:在所有的线程中按相同的次序获取所有资源锁。例如,如果有四个资源?A、B、C和D?并且一个线程可能要获取四个资源中任何一个资源的锁,则请确保在获取对B的锁之前首先获取对A的锁,依此类推。如果“线程1”希望获取对B和C的锁,而“线程2”获取了A、C和D的锁,则这一技术可能导致阻塞,但它永远不会在这四个锁上造成死锁。

  活锁

  当一个线程忙于接受新任务以致它永远没有机会完成任何任务时,就会发生活锁。这个线程最终将超出缓冲区并导致程序崩溃。试想一个秘书需要录入一封信,但她一直在忙于接电话,所以这封信永远不会被录入。内存损坏

  • 上一篇文章:

  • 下一篇文章:
  •  

    0791-87557233

    重视每个来电 珍惜您的时间
    思诚者开发沙龙
    江西思诚科技有限公司  赣ICP备17006097号  CopyRight©2014 - 2018