当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()方法结束。所谓死亡就是线程释放了实体,即释放分配给线程对象的内存。

代码展示如下所示:

public class Test05 {

    public static void main(String[] args) {
// TODO Auto-generated method stub
SpeakElephant speakElephant;
SpeakCar speakCar;
speakElephant = new SpeakElephant();
speakCar = new SpeakCar();
speakElephant.start();
speakCar.start();
for(int i = 1; i <= 15; i++)
{
System.out.print("主人" + i + " ");
}
} }

SpeakCar.java

public class SpeakCar extends Thread{
public void run()
{
for(int i = 0; i <= 20; i++)
{
System.out.print("轿车"+ i + " ");
}
}
}

SpeakElephant.java

public class SpeakElephant extends Thread{
public void run()
{
for(int i = 0; i <= 20; i++)
{
System.out.print("大象" + i + " ");
}
}
}

运行结果如下所示:

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

线程调度与优先级

处于就绪的线程首先进入就绪队列等候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. maven仓库添加本地jar

    一.将jar添加到本地仓库的做法: 以下面pom.xml依赖的jar包为例: 实际项目中pom.xml依赖写法: <dependency> <groupId>org.sprin ...

  2. Introduction to Mathematical Thinking - Week 7

    Q: Why did nineteenth century mathematicians devote time to the proof of self-evident results? Selec ...

  3. VS2015 如何打印出 类的内存布局?

    项目->属性->C/C++->命令行 添加如下内容: /d1 reportAllClassLayout [XXX] XXX 可选,代表源文件名,例如 main.cpp 或 /d1 r ...

  4. Python代码实现删除一个list里面的重复元素

    lst=[11,22,33,44,22,11,22,44] print(list(set(lst))) # 打印结果:[33, 11, 44, 22] d = {} for index,item in ...

  5. 外部导入js(javascript)文件方法

    <script src="myScript.js"></script>

  6. 提交cookie登录

    # coding:utf-8import requests # 先打开登录首页,获取部分cookieurl = "https://passport.cnblogs.com/user/sign ...

  7. 启动hive --service metastore &出现Missing Hive Execution Jar: /opt/apache-hive-1.2.0-bin//lib/hive-exec-*.jar

    原因:出现上述问题通常是运行hive 在bin/目录下的脚本所致. 解决办法:就是让HIVE_HOME变量指向hive-trunk/build/dist目录,将$HIVE_HOME/bin添加到PAT ...

  8. Armijo-Goldstein准则与Wolfe-Powell准则

    Armijo-Goldstein准则与Wolfe-Powell准则是不精确的一维搜索的两大准则. 之所以要遵循这些准则是为了能使算法收敛(求最优解).即要使我们的不精确的一维搜索的步长满足一定的规则, ...

  9. 剑指offer 面试21题

    面试21题: 题目:调整数组的顺序使奇数位于偶数前面 题一:输入一个整数数组,实现一个函数来调整该数组中数字的顺序,使得所有奇数位于数组的前半部分,所有偶数位于数组的后半部分. 解题思路:使用两个指针 ...

  10. win7 重启dns

    安装xshell.百度一搜就下载了. 修改hosts,hosts路径 C:\Windows\System32\drivers\etc\hosts 写法和linux一样. 重启dns命令 ipconfi ...