当JVM加载代码,发现main方法之后,就会启动一个线程,这个线程称为“主线程”(main线程),该线程负责执行main方法。那么在main方法执行中再创建的其他线程就称为程序中的其他线程。

如果main方法中又创建了其他线程,那么JVM就要在主线程和其他线程之间轮流切换,保证每个线程都有机会使用CPU资源,main方法即使执行完最后的语句(主线程结束),JVM也不会结束java应用程序,JVM一定要等到Java应用程序中的所有线程都结束之后,才结束java应用程序。

java语言使用Thread类及其子类的对象来表示线程,新建的线程在它的一个完整的生命周期中通常要经历如下的四种状态。

1.新建

当一个Thread类或其子类的对象被声明并创建时,新生的线程对象处于新建状态。此时它已经有了相应的内存空间和其他资源。

2.运行

线程创建之后就具备了运行的条件,一旦轮到它来享用CPU资源时,即JVM将CPU使用权切换给该线程时,此线程就可以脱离创建它的主线程独立开始自己的生命周期。

线程创建后仅仅是占有了内存资源,在JVM管理的线程中还没有这个线程,此线程必须调用start()方法通知JVM,这样JVM就会知道又有一个新线程排队等候切换了。

当JVM将CPU使用权切换给线程时,如果线程是Thread的子类创建的,该类中的run()方法就立即执行,run()方法规定了该线程的具体使命。所以程序必须在子类中重写父类的run()方法,Thread类中的run()方法没有具体内容。

在没有结束run()方法前,不要让线程再调用start()方法,否则会发生IIlegalThreadStateException异常。

3.中断

有四种原因的中断。

  • JVM将CPU资源从当前线程切换给其他线程,使本线程让出CPU的使用权处于中断状态。
  • 线程使用CPU资源期间,执行了sleep(int millsecond)方法,使当前线程进入休眠状态。sleep(int millsecond)方法是Thread类的一个类方法,线程一旦执行了sleep(int millsecond)方法,就立刻让出CPU的使用权,使当前线程处于中断状态。经过参数millsecond指定的毫秒之后,该线程就重新进到线程队列中排队等待CPU资源,以便从中断出继续运行。
  • 线程使用CPU资源期间,执行了wait()方法,使得当前线程进入等待状态。
  • 线程使用CPU资源期间,执行某个操作进入阻塞状态,比如执行读/写操作引起阻塞。进入阻塞状态时线程不能进入排队队列,只有当引起阻塞的原因消除时,线程才重新进到线程队列中排队等待CPU资源。

4.死亡

处于死亡状态的线程不具有继续运行的能力。

线程死亡的原因有二,一个是正常运行的线程完成了它的全部工作,即执行完run()方法中的全部语句,结束了run()方法;另一个原因是线程被强制性地终止,即强制run()方法结束。所谓死亡就是线程释放了实体,即释放分配给线程对象的内存。

代码展示如下所示:

  1. public class Test05 {
  2.  
  3. public static void main(String[] args) {
  4. // TODO Auto-generated method stub
  5. SpeakElephant speakElephant;
  6. SpeakCar speakCar;
  7. speakElephant = new SpeakElephant();
  8. speakCar = new SpeakCar();
  9. speakElephant.start();
  10. speakCar.start();
  11. for(int i = 1; i <= 15; i++)
  12. {
  13. System.out.print("主人" + i + " ");
  14. }
  15. }
  16.  
  17. }

SpeakCar.java

  1. public class SpeakCar extends Thread{
  2. public void run()
  3. {
  4. for(int i = 0; i <= 20; i++)
  5. {
  6. System.out.print("轿车"+ i + " ");
  7. }
  8. }
  9. }

SpeakElephant.java

  1. public class SpeakElephant extends Thread{
  2. public void run()
  3. {
  4. for(int i = 0; i <= 20; i++)
  5. {
  6. System.out.print("大象" + i + " ");
  7. }
  8. }
  9. }

运行结果如下所示:

而且每次运行的结果都不一样,

线程调度与优先级

处于就绪的线程首先进入就绪队列等候CPU资源,同一时刻在就绪队列中的线程可能有很多个。Java虚拟机中的线程调度器负责管理线程,调度器把线程的优先级分为10个级别,分别用Thread类中的类常量表示。每个Java线程的优先级都在常数1和10之间。如果没有明确地设置好线程的优先级别,每个线程的优先级别都为常数5.

线程的优先级通过setPriority(int grade)方法调整,如果参数不在1和10之间,那么会产生异常。

review33的更多相关文章

  1. win10+Anaconda3+CUDA9.0+CUDNN7.1+TensorFlow-gpu1.9+Pycharm

    想在win10上运行下YOLO的例子,要先配置环境,折腾了两天,终于好了,整理下自己觉得有用且正确的流程. win10+Anaconda3+CUDA9.0+CUDNN7.1+TensorFlow1.9 ...

随机推荐

  1. 截取字符(pos,copy,Leftstr,MidStr,RightStr)以逗号为准把字符串拆分,判断字符串是否有数字、字母(大小写), 去掉字符串空格

    1.copy(a,b,c) 举个例子: str := “123456”;str1 := Copy(Str,2,3);结果就是 str1 等于 234.Copy有3个参数,第一个是你要处理的字符串,第二 ...

  2. influxDB概念

    一.基本概念 1)database--数据库,这个同传统数据库的数据库概念. 2)measurement--数据表,在InfluxDB中,measurement即为表的作用,同传统数据库中的table ...

  3. Linux 装JDK

    1.查看当前系统有没有装jdk java -version 2.看看有没有安装包 rpm -qa | grep java 3.卸载OpenJDK $>rpm -e --nodeps tzdata ...

  4. Java内存模型之happens-before

    转载自    http://www.cnblogs.com/chenssy/p/6393321.html 无论处理器.JVM.编译器都会都保证程序正确的前提下尽可能的对指令执行效率进行优化,进行指令重 ...

  5. python并发编程&多线程(二)

    前导理论知识见:python并发编程&多线程(一) 一 threading模块介绍 multiprocess模块的完全模仿了threading模块的接口,二者在使用层面,有很大的相似性 官网链 ...

  6. servlet3.0 的新特性之二注解代替了web.xml配置文件

    servlet3.0 的新特性: 注解代替了 web.xml 文件 支持了对异步的处理 对上传文件的支持 1.注解代替了配置文件 1.删除了web.xml 文件 2. 在Servlet类上添加@Web ...

  7. http请求设置setConnectTimeout()方法超时无响应

    相信非常多小伙伴在写 HttpURLConnection中都设置了setConnectTimeout()方法.目的就是在网络不好的情况下超时返回 然而我们设置的setConnectTimeout()并 ...

  8. Ubuntu14.04下Nginx反向代理Odoo域名

    安装nginx sudo apt-get install -y nginx 修改配置文件 vi /etc/nginx/nginx.conf #注释掉下面这行代码 #include /etc/nginx ...

  9. 0404-服务注册与发现-客户端负载均衡-两种自定义方式-Ribbon通过代码自定义配置、使用配置文件自定义Ribbon Client

    一.官方文档解读 官方地址:https://cloud.spring.io/spring-cloud-static/Edgware.SR3/single/spring-cloud.html#_cust ...

  10. 深度学习:Keras入门(一)之基础篇(转)

    转自http://www.cnblogs.com/lc1217/p/7132364.html 1.关于Keras 1)简介 Keras是由纯python编写的基于theano/tensorflow的深 ...