`
uule
  • 浏览: 6352064 次
  • 性别: Icon_minigender_1
  • 来自: 一片神奇的土地
社区版块
存档分类
最新评论

【线程的状态转换】

阅读更多
public Thread(String name)

	
Thread.currentThread().getName()

 

注意还有个调用 start()后的可运行(就绪 Runnable)状态!

sleep() ->  阻塞状态  -》 停止几秒

join()  -》 阻塞状态 -》等待前一线程执行完

wait() -》 释放锁和资源,进入等待队列  -》 notify()唤醒后进入锁池

synchronized  -》 获取不到锁,进入锁池

 

 

线程的状态转换:
     1、新建状态(New):新创建了一个线程对象。

  2、就绪状态(Runnable):线程对象创建后,其他线程调用了该对象的start()方法。该状态的线程位于可运行线程池中,变得可运行,等待获取CPU的使用权

  3、运行状态(Running):就绪状态的线程获取了CPU ,执行程序代码。

  4、阻塞状态(Blocked):阻塞状态是线程因为某种原因放弃CPU使用权,暂时停止运行 。直到线程进入就绪状态,才有机会转到运行状态。阻塞的情况分三种:
        (一)、等待阻塞:运行的线程执行wait()方法,JVM会把该线程放入等待池中。
      (二)、同步阻塞:运行的线程在获取对象的同步锁时,若该同步锁被别的线程占用,则JVM会把该线程放入锁池中。
      (三)、其他阻塞:运行的线程执行sleep()或join()方法,或者发出了I/O请求时,JVM会把该线程置为阻塞状态。当sleep()状态超时、join()等待线程终止或者超时、或者I/O处理完毕时,线程重新转入就绪状态。

  5、死亡状态(Dead):线程执行完了或者因异常退出了run()方法,该线程结束生命周期。

 



 

 解释:
   1、当线程调用了自身的sleep()方法或其他线程的join()方法,就会进入阻塞状态(该状态既停止当前线程,但并不释放所占有的资源) 。当sleep()结束或join()结束后,该线程进入可运行状态,继续等待OS分配时间片;
   2、线程调用了yield()方法,意思是放弃当前获得的CPU时间片,回到可运行状态 ,这时与其他进程处于同等竞争状态,OS有可能会接着又让这个进程进入运行状态;
   3、当线程刚进入可运行状态(即就绪状态),发现将要调用的资源被synchroniza(同步),获取不到锁标记,将会立即进入锁池状态,等待获取锁标记(这时的锁池里也许已经有了其他线程在等待获取锁标记,这时它们处于队列状态,既先到先得),一旦线程获得锁标记后,就转入可运行状态,等待 OS分配CPU时间片

        Wait()方法和notify()方法:当一个线程执行到wait()方法时,它就进入到一个和该对象相关的等待池中,同时失去了对象的锁。当它被一个notify()方法唤醒时,等待池中的线程就被放到了锁池中。该线程从锁池中获得锁,然后回到wait()前的中断现场
   4、当线程调用wait()方法后会进入等待队列(进入这个状态会释放所占有的所有资源,与阻塞状态不同),进入这个状态后,是不能自动唤醒的,必须依靠其他线程调用notify()或notifyAll()方法才能被唤醒 (
wait(1000)时可以自动唤醒 ) (由于notify()只是唤醒一个线程,但我们由不能确定具体唤醒的是哪一个线程,也许我们需要唤醒的线程不能够被唤醒,因此在实际使用时,一般都用notifyAll()方法,唤醒有所线程),线程被唤醒后会进入锁池,等待获取锁标记。

参考:http://hi.baidu.com/guessa/blog/item/bac50223a2657942925807ab.html

 

 

调用Sleep、join时,不会释放所占用的资源,所以会进入阻塞状态;

调用Wait时,会释放所占用的资源,所以会进入等待队列。

 

1、睡眠
   Thread.sleep(long millis)和Thread.sleep(long millis, int nanos)静态方法强制当前正在执行的线程休眠(暂停执行),以“减慢线程”。
 
   线程睡眠的原因:线程执行太快,或者需要强制进入下一轮,因为Java规范不保证合理的轮换。
   睡眠的实现:调用静态方法。
        try {
            Thread.sleep(123);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
   睡眠的位置:为了让其他线程有机会执行,可以将Thread.sleep()的调用放线程run()之内。这样才能保证该线程执行过程中会睡眠。

/**
* 一个计数器,计数到100,在每个数字之间暂停1秒,每隔10个数字输出一个字符串
*/
public class MyThread extends Thread {

    public void run() {
        for (int i = 0; i < 100; i++) {
            if ((i) % 10 == 0) {
                System.out.println("-------" + i);
            }
            System.out.print(i);
            try {
                Thread.sleep(1);
                System.out.print("    线程睡眠1毫秒!\n");
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

    public static void main(String[] args) {
        new MyThread().start();
    }
} 

  -------0
0    线程睡眠1毫秒!
1    线程睡眠1毫秒!
2    线程睡眠1毫秒!
3    线程睡眠1毫秒!
4    线程睡眠1毫秒!
5    线程睡眠1毫秒!
6    线程睡眠1毫秒!
7    线程睡眠1毫秒!
8    线程睡眠1毫秒!
9    线程睡眠1毫秒!
-------10
10    线程睡眠1毫秒!
11    线程睡眠1毫秒!
12    线程睡眠1毫秒!
13    线程睡眠1毫秒!
14    线程睡眠1毫秒!
15    线程睡眠1毫秒!
16    线程睡眠1毫秒!
17    线程睡眠1毫秒!
18    线程睡眠1毫秒!
19    线程睡眠1毫秒!
-------20
20    线程睡眠1毫秒!
21    线程睡眠1毫秒!
22    线程睡眠1毫秒!
23    线程睡眠1毫秒!
24    线程睡眠1毫秒!
25    线程睡眠1毫秒!
26    线程睡眠1毫秒!
27    线程睡眠1毫秒!
28    线程睡眠1毫秒!
29    线程睡眠1毫秒!
-------30

。。。

 

注意:
    1、线程睡眠是帮助所有线程获得运行机会 的最好方法。
    2、线程睡眠到期自动苏醒,并返回到可运行状态,不是运行状态 。sleep()中指定的时间是线程不会运行的最短时间。因此,sleep()方法不能保证该线程睡眠到期后就开始执行
    3、sleep()是静态方法,只能控制当前正在运行的线程。


2、join()方法    
   Thread的非静态方法join()让一个线程B“加入”到另外一个线程A的尾部。在A执行完毕之前,B不能工作。例如:
        Thread t = new MyThread();
        t.start();
        t.join();
另外,join()方法还有带超时限制的重载版本。 例如t.join(5000);则让线程等待5000毫秒,如果超过这个时间,则停止等待,变为可运行状态。
 
join()导致线程栈发生了变化,当然这些变化都是瞬时的。



 3、Thread.yield()方法
   yield()是让当前运行线程回到可运行状态,以允许具有相同优先级的其他线程获得运行机会。因此,使用yield()的目的是让相同优先级的线程之间能适当的轮转执行。但是,实际中无法保证yield()达到让步目的,因为让步的线程还有可能被线程调度程序再次选中。
   在大多数情况下,yield()将导致线程从运行状态转到可运行状态,但有可能没有效果。

 

 Yield()方法是停止当前线程,让同等优先权的线程运行。如果没有同等优先权的线程,那么Yield()方法将不会起作用。

线程的让步是通过Thread.yield()来实现的。yield()方法的作用是:暂停当前正在执行的线程对象,并执行其他线程。
 
要理解yield(),必须了解线程的优先级的概念。线程总是存在优先级,优先级范围在1~10之间。JVM线程调度程序是基于优先级的抢先调度机制。在大多数情况下,当前运行的线程优先级将大于或等于线程池中任何线程的优先级。但这仅仅是大多数情况。当线程池中线程都具有相同的优先级,调度程序的JVM实现自由选择它喜欢的线程。这时候调度程序的操作有两种可能:一是选择一个线程运行,直到它阻塞或者运行完成为止。二是时间分片,为池内的每个线程提供均等的运行机会。
 
注意:当设计多线程应用程序的时候,一定不要依赖于线程的优先级。因为线程调度优先级操作是没有保障的,只能把线程优先级作用作为一种提高程序效率的方法,但是要保证程序不依赖这种操作。
 
设置线程的优先级:线程默认的优先级是创建它的执行线程的优先级。可以通过setPriority(int newPriority)更改线程的优先级。例如:
        Thread t = new MyThread();
        t.setPriority(8);
        t.start();
线程优先级为1~10之间的正整数,JVM从不会改变一个线程的优先级。然而,1~10之间的值是没有保证的。一些JVM可能不能识别10个不同的值,而将这些优先级进行每两个或多个合并,变成少于10个的优先级,则两个或多个优先级的线程可能被映射为一个优先级。
 
线程默认优先级是5,Thread类中有三个常量,定义线程优先级范围:
    static int MAX_PRIORITY
          线程可以具有的最高优先级。
    static int MIN_PRIORITY
          线程可以具有的最低优先级。
    static int NORM_PRIORITY
          分配给线程的默认优先级。

参考:http://lavasoft.blog.51cto.com/62575/99153

 

 

 

stop()suspend()方法为何不推荐使用?

反对使用stop(),是因为它不安全。它会解除由线程获取的所有锁定,而且如果对象处于一种不连贯状态,那么其他线程能在那种状态下检查和修改它们。结果很难检查出真正的问题所在。suspend()方法容易发生死锁。调用suspend()的时候,目标线程会停下来,但却仍然持有在这之前获得的锁定。此时,其他任何线程都不能访问锁定的资源,除非被"挂起"的线程恢复运行。对任何线程来说,如果它们想恢复目标线程,同时又试图使用任何一个锁定的资源,就会造成死锁。所以不应该使用suspend(),而应在自己的Thread类中置入一个标志,指出线程应该活动还是挂起。若标志指出线程应该挂起,便用 wait()命其进入等待状态。若标志指出线程应当恢复,则用一个notify()重新启动线程。

 

 

 

 suspend() 和 resume() 方法:

两个方法配套使用,suspend()使得线程进入阻塞状态,并且不会自动恢复,必须其对应的 resume() 被调用,才能使得线程重新进入可执行状态。典型地,suspend() 和 resume() 被用在等待另一个线程产生的结果的情形:测试发现结果还没有产生后,让线程阻塞,另一个线程产生了结果后,调用 resume() 使其恢复。但suspend()方法很容易引起死锁问题,已经不推荐使用了。 

 

 

 

  • 大小: 63.5 KB
  • 大小: 8 KB
  • 大小: 39.8 KB
分享到:
评论
2 楼 jieke456 2013-07-14  
看了楼主的这边文章受益匪浅,非常感谢!
1 楼 zhongliangjun1 2012-12-10  
博主这篇帖子是目前看到的讲述线程状态转换最清楚的一篇了

相关推荐

    java线程状态转换图

    Java 线程状态转换图 Java 线程状态转换图是 Java 编程中非常重要的一个概念,它描述了线程在不同的状态之间的转换关系。了解线程状态转换图对 Java 编程的理解和应用非常重要。本文将详细介绍 Java 线程状态转换图...

    Java线程:线程状态的转换

    ### Java线程:线程状态的转换 #### 一、线程状态及其转换 ...通过以上分析,我们可以看到Java线程状态转换的基本原理以及如何利用线程状态来控制线程的行为,这对于开发高性能、高可靠性的并发程序至关重要。

    Java线程状态转换.pdf

    Java线程状态转换是Java多线程编程中的关键概念,对于理解和优化并发程序至关重要。Java线程在其生命周期中经历多种状态,这些状态之间的转换是由线程调度器根据特定的策略来决定的。以下是对Java线程状态转换的详细...

    Java-多线程线程状态转换图

    多线程线程状态转换图

    Java线程状态流转图

    Java线程状态流转图中,以下方法与线程状态转换相关: * Object.wait():使线程从RUNNABLE状态转换到WAITING状态。 * Object.notify():使线程从WAITING状态转换到RUNNABLE状态。 * Object.notifyAll():使多个线程...

    Java线程状态转换关系实例解析

    Java线程状态转换关系实例解析 Java线程状态转换关系实例解析是Java多线程编程中的一种重要概念,它指的是Java线程在不同的状态之间的转换关系。了解这些状态转换关系可以帮助开发者更好地编写多线程程序,避免线程...

    Java线程:线程状态的转换.pdf

    在多线程环境中,合理地控制线程状态转换可以提高程序的效率和正确性。 例如,下面的代码展示了如何使用`sleep()`方法来实现线程间的交替执行: ```java public class MyThread extends Thread { private static ...

    Java线程:线程状态的转换[参考].pdf

    线程状态转换是一个复杂的过程,涉及到线程调度、锁的管理等多个方面。了解这些状态和转换机制,有助于编写高效、可控的多线程程序。在实际编程中,应根据需求合理利用这些状态控制,确保线程间的协作和资源利用达到...

    在java中的线程的转换图

    Java中的线程状态转换是Java多线程编程中一个至关重要的概念。理解这些状态转换有助于开发者更好地管理和控制并发执行的任务,避免线程安全问题。线程在生命周期中会经历多种状态,包括新建(New)、可运行...

    Java多线程教程资料(强烈推荐).docx

    本文档提供了 Java 多线程编程的详细教程,涵盖了多线程的概念、原理、创建、启动、同步、锁、线程池、线程状态转换、线程交互、线程调度、volatile 关键字、信号量、阻塞队列、条件变量等知识点。 一、Java 多线程...

    JAVA多线程(精典总结)

    本文将深入探讨Java多线程的核心概念、创建方式、线程状态转换、线程调度以及线程优先级调整。 首先,理解线程的基本概念至关重要。线程是进程中的一个执行单元,是操作系统调度的基本单位。与进程相比,线程更轻...

    操作系统实验41

    线程状态转换是操作系统内核调度的基础,尤其在多线程环境中,处理器资源的高效利用和任务并行处理能力的提升都依赖于这种转换。 实验原理中提到,EOS操作系统将线程作为处理器调度的基本单元。进程创建时,系统会...

    多线程基础知识

    多线程思维导图和XMind文件可能包含了以下内容:线程的创建与销毁、线程状态转换、同步机制详解、线程安全的数据结构、线程池的使用、线程间通信实例、并发设计模式以及解决并发问题的最佳实践等。 通过学习和掌握...

    VC++挂起、启动和恢复线程的方法实例

    在VC++编程中,线程是并发执行的基本单元,它使得程序可以同时处理多个任务。...理解线程状态转换和使用适当的同步机制是多线程编程的关键。通过熟练掌握这些概念和API,开发者可以编写出高效、可靠的多线程应用程序。

    第多线程PPT学习教案.pptx

    【线程状态转换】 线程状态之间可以进行如下转换: 1. 创建状态 -&gt; 就绪状态(调用`start()`) 2. 就绪状态 -&gt; 运行状态(CPU分配) 3. 运行状态 -&gt; 就绪状态(被其他线程抢占或`yield()`) 4. 运行状态 -&gt; 阻塞状态...

    Java多线程编程深入详解

    - 分析了线程的生命周期,包括初始化、运行、冻结和死亡状态,并深入探讨了线程状态转换的内部机制。 4. 线程的同步机制 - 介绍了同步代码块和同步方法的用法,以及this锁和static锁的区别和应用场景。 - 讨论了...

    Java 多线程学习详细总结

    本文将深入探讨Java中的多线程概念、实现方式、线程状态转换、线程调度、线程同步以及数据传递等相关知识。 1. **扩展`java.lang.Thread`类** 创建线程的一种方式是通过扩展`Thread`类。在`Thread`类的子类中,...

    玩转多线程编程.pptx

    线程状态转换图可以help我们更好地理解线程的生命周期。 在多线程编程中,经常会遇到的问题包括死锁、程序性能低下、对象难以被删除等问题。为了解决这些问题,我们需要了解多线程编程的要点,包括对象管理、原子...

Global site tag (gtag.js) - Google Analytics