java中多线程的实现方式以及生命周期?
来源:互联网 发布:Python开发安卓 编辑:程序博客网 时间:2024/06/07 05:00
前言:
Java 给多线程编程提供了内置的支持。一个多线程程序包含两个或多个能并发运行的部分。程序的每一部分都称作一个线程,并且每个线程定义了一个独立的执行路径。
多线程是多任务的一种特别的形式,但多线程使用了更小的资源开销。这里就不得不说到一个和线程相关的另一个术语
- 进程:一个进程包括由操作系统分配的内存空间,包含一个或多个线程。但是一个线程不能独立的存在,它必须是进程的一部分。
一个进程一直运行,直到所有的非守候线程都结束运行后才能结束。
一、JAVA多线程的实现有哪几种方式(附带最简单的示例)?
1.继承Thread类
线程的具体执行体是在run()方法中,通过继承了Thread类的类的实例调用start()方法。从而执行run()方法
public class ExtendsThreadTest extends Thread{public static void main(String[] args) {
ExtendsThreadTest test = new ExtendsThreadTest();test.start();}@Overridepublic void run() {System.out.println("这里是执行主体内容");}}
2.实现Runnable接口
线程的具体执行体是在run()方法中,通过实现了Runnable接口的类的实例调用start()方法。从而执行run()方法
public class ExtendsThreadTest implements Runnable{private Thread thread;public static void main(String[] args) {ExtendsThreadTest test = new ExtendsThreadTest();ExtendsThreadTest test2 = new ExtendsThreadTest();test.start();test2.start();}public void start(){if(thread == null){thread = new Thread(this);thread.start();}}@Overridepublic void run() {System.out.println("这里是执行主体内容");}}
3.实现Callable<V>接口,并接收Future
. 创建 Callable接口的实现类,并实现 call()方法,该 call()方法将作为线程执行体,并且有返回值。
这个call方法中的执行,可以通过实现类的构造函数中所传递进来的服务类来调用本身所重写的业务方法
. 创建 Callable实现类的实例,使用 FutureTask类来包装 Callable对象,该 FutureTask对象封装了该 Callable对象的 call()方法的返回值。
. 使用 FutureTask对象作为 Thread对象的 target创建并启动新线程。
. 调用 FutureTask对象的 get()方法来获得子线程执行结束后的返回值。
public class ExtendsThreadTest implements Callable<String>{public static void main(String[] args) {ExtendsThreadTest test = new ExtendsThreadTest();FutureTask<String> task = new FutureTask<String>(test);new Thread(task).start();try {String result = task.get();System.out.println(result);} catch (InterruptedException e) {// TODO Auto-generated catch blocke.printStackTrace();} catch (ExecutionException e) {// TODO Auto-generated catch blocke.printStackTrace();}}@Overridepublic String call() throws Exception {//这里面执行主体内容代码return "hello Thread";}}
二、JAVA线程的生命周期?
1.新建状态:
使用 new关键字和 Thread类或其子类建立一个线程对象后,该线程对象就处于新建状态。它保持这个状态直到程序 start()这个线程。
2.就绪状态:
当线程对象调用了start()方法之后,该线程就进入就绪状态。就绪状态的线程处于就绪队列中,要等待JVM里线程调度器的调度。
3.运行状态:
如果就绪状态的线程获取 CPU资源,就可以执行 run(),此时线程便处于运行状态。处于运行状态的线程最为复杂,它可以变为阻塞状态、就绪状态和死亡状态。
4.阻塞状态:
如果一个线程执行了sleep(睡眠)、suspend(挂起)等方法,失去所占用资源之后,该线程就从运行状态进入阻塞状态。在睡眠时间已到或获得设备资源后可以重新进入就绪状态。可以分为三种:
等待阻塞:运行状态中的线程执行 wait()方法,使线程进入到等待阻塞状态。
同步阻塞:线程在获取 synchronized同步锁失败(因为同步锁被其他线程占用)。
其他阻塞:通过调用线程的 sleep()或 join()发出了 I/O请求时,线程就会进入到阻塞状态。当sleep()状态超时,join()等待线程终止或超时,或者 I/O处理完毕,线程重新转入就绪状态。
5.死亡状态:
一个运行状态的线程完成任务或者其他终止条件发生时,该线程就切换到终止状态。
三、Thread类的主要运用方法?
1 public void start() 使该线程开始执行;Java虚拟机调用该线程的 run方法。
2 public void run() 如果该线程是使用独立的 Runnable运行对象构造的,则调用该 Runnable对象的 run方法;否则,该方法不执行任何操作并返回。
3 public final void setName(String name) 改变线程名称,使之与参数 name相同。
4 public final void setPriority(int priority) 更改线程的优先级。
5 public final void setDaemon(boolean on) 将该线程标记为守护线程或用户线程。
6 public final void join(long millisec) 等待该线程终止的时间最长为 millis毫秒。
7 public void interrupt() 中断线程。该方法实际上只是设置了一个中断状态,当该线程由于下列原因而受阻时,这个中断状态就起作用了:
(1)如果线程在调用 Object类的 wait()、wait(long)或 wait(long, int)方法,或者该类的 join()、join(long)、join(long, int)、sleep(long)或 sleep(long, int)方法过程中受阻,则其中断状态将被清除,它还将收到一个InterruptedException异常。这个时候,我们可以通过捕获InterruptedException异常来终止线程的执行,具体可以通过return等退出或改变共享变量的值使其退出。
(2)如果该线程在可中断的通道上的 I/O操作中受阻,则该通道将被关闭,该线程的中断状态将被设置并且该线程将收到一个 ClosedByInterruptException。这时候处理方法一样,只是捕获的异常不一样而已
8 public final boolean isAlive() 测试线程是否处于活动状态。
9 public static void yield() 暂停当前正在执行的线程对象,并执行其他线程。
10 public static void sleep(long millisec) 在指定的毫秒数内让当前正在执行的线程休眠(暂停执行,时间到则自动唤醒),此操作受到系统计时器和调度程序精度和准确性的影响。
11 public static boolean holdsLock(Object x) 当且仅当当前线程在指定的对象上保持监视器锁时,才返回 true。
12 public static Thread currentThread() 返回对当前正在执行的线程对象的引用。
13 public static void dumpStack() 将当前线程的堆栈跟踪打印至标准错误流
14 public final void wait() 当前线程进行等待,需要通过手工调用notify()或者notifyAll()方法才能唤醒
四、interrupt、interrupted 、isInterrupted区别是什么?
1、interrupt方法用于中断线程。调用该方法的线程的状态为将被置为"中断"状态。
注意:线程中断仅仅是置线程的中断状态位,不会停止线程。需要用户自己去监视线程的状态为并做处理。支持线程中断的方法(也就是线程中断后会抛出interruptedException的方法)就是在监视线程的中断状态,一旦线程的中断状态被置为“中断状态”,就会抛出中断异常
2、interrupted是作用于当前线程,底层方法入参是true,说明返回线程的状态位后,要清掉原来的状态位(恢复成原来情况),也只有当前线程能够清楚自己的状态位
3、isInterrupted是作用于调用该方法的线程对象所对应的线程。(线程对象对应的线程不一定是当前运行的线程。例如我们可以在A线程中去调用B线程对象的isInterrupted方法。),底层方法入参是false,就是直接返回线程的状态位。- java中多线程的实现方式以及生命周期?
- java中多线程的实现方式以及区别(1)
- java中多线程的实现方式以及区别(2)
- java中多线程的实现方式有两种
- Java中多线程的实现方式
- java中实现多线程的方式
- JAVA单线程以及java多线程的实现方式
- Java实现多线程的两种方式以及概述
- java中多线程实现方式
- java实现多线程的方式
- java多线程的实现方式
- Java多线程的实现方式
- Java多线程的实现方式
- java 实现多线程的方式
- Java实现多线程的方式
- Java 实现多线程的方式
- Java中有两种实现多线程的方式以及两种方式之间的区别
- Java中有两种实现多线程的方式以及两种方式之间的区别
- JavaScript 继承---寄生式继承
- 一周IT歪评丨清北BAT落户雄安新区/刘强东穿9块钱大裤衩/青少年沉迷王者日本称不惧怕中国
- 大一实训 C语言笔记
- 设置宽高相等的背景图片
- CodeForce 899C
- java中多线程的实现方式以及生命周期?
- iOS申请证书教程
- JVM学习笔记(二)------Java代码编译和执行的整个过程
- 20条最最常用的Linux命令讲解
- webpack的常用命令
- 8.4输入某班学生某门课的成绩,用函数编程统计不及格人数
- vs qt 无法找到 ui...h
- (收藏)故障树常用符号及含义
- i++与++i的区别