Thread 和 Runnable

1. 简介

Java 主要是通过 java.lang.Thread 类以及 java.lang.Runnable 接口实现线程机制的。

  1. Thread 类为底层操作系统的线程体系架构提供一套统一接口
  2. Runnable 接口为关联 Thread 对象的线程提供执行代码

2. 创建 Thread 和 Runnable 对象

2.1 创建 Runnable 对象

创建 Runnable 有两种方式:

创建一个实现了 Runnable 接口的匿名类
Runnable r = new Runnable() {
@Override
public void run() {
System.out.println("Hello from thread");
}
};
或使用 lambda 表达式
Runnable r = () -> System.out.println("Hello from thread");

2.2 创建 Thread 对象

通过两种方式创建:

将 Runnable 对象作为 Thread 类的构造函数的参数
Thread t = new Thread(r);
继承 Thread 类继而重写它的 run() 方法
class MyThread extends Thread {
@Override
public void run() {
System.out.println("Hello from thread");
}
}
// ...
MyThread mt = new MyThread();

3. 获取和设置线程状态

几个方法:

  • getName() : 获取线程名称

  • setName(): 设置线程名称

  • isAlive(): 判断线程存活状态,存活返回 true,不存活返回 false(只有执行 start() 方法,线程才启动)

  • getState(): 获取线程执行状态

    线程的执行状态由 Thread.State 枚举常量标识:

    • NEW:线程还没有开始执行
    • RUNNABLE:线程正在 JVM 中执行
    • BLOCKED:线程被阻塞并等待一个监听锁
    • WAITING:线程无限期地等待另外一条线程执行特定地操作
    • TIMED_WAITING:线程在特定地时间内等待另外一条线程执行某种操作
    • TERMINATED:线程已经退出
  • setPriority(): 设置线程优先级

    传递给优先级的值介于 Thread.MIN_PRIORITYThread.MAX_PRIORITY 之间,而 Thread.NORMAL_PRIORITY 则确定了默认的优先级

  • isDaemon(): 判断线程是否为守护进程。守护进程返回 true,不是返回 false

  • start(): 启动与对象关联的线程。

    如果线程之前已经启动且处于运行状态,又或者线程已经死亡,这个方法就会抛出 java.lang.IllegalThreadStateException


4. 操作更高级的线程任务

中断线程

当一个线程被中断时,它会抛出 java.lang.InterruptedException,这一机制由下面的 3 种方法构成:

  • void interrupt(): 中断调用此方法的 Thread 对象所关联的线程。该线程的中断状态会被清除
  • static boolean interrupted(): 验证当前线程是否已经中断。该线程的中断状态会被这个方法清除掉
  • boolean isInterrupted(): 验证线程是否已经中断。该线程的中断状态不受此方法的影响
等待线程

Thread 类提供了 3 种 join() 方法,允许调用线程等待执行此方法的线程对象所关联的线程执行完毕。

  • void join(): 无限期地等待直至该线程死亡
  • void join(long millis):该线程死亡之前最多等待 millis 毫秒
  • void join(long millis, int nanos):该线程死亡之前最多等待 millis 毫秒加 nanos 纳秒
线程睡眠

Thread 类声明了一对静态方法致使线程睡眠(暂时性地停止执行)

  • void sleep(long millis):睡眠 millis 毫秒数
  • void sleep(long millis, int nanos):睡眠 millis 毫秒数和 nanos 纳秒数

5. Thread 和 Runnable 区别(重要)

首先讲一下多线程的实现思路,主要有两种方法:

  1. 通过继承 Thread 类,重写 run() 方法
class MyThread extends Thread{
private int ticket = 5;
@Override
public void run(){
for (int i=0;i<10;i++)
{
if(ticket > 0){
System.out.println("ticket = " + ticket--);
}
}
}
} class ThreadDemo{
public static void main(String[] args){
new MyThread().start();
new MyThread().start();
new MyThread().start();
}
}
  1. 通过实现 Runnable 接口,实现多线程
class MyThread implements Runnable{
private int ticket = 5;
@Override
public void run(){
for (int i=0;i<10;i++)
{
if(ticket > 0){
System.out.println("ticket = " + ticket--);
}
}
}
} class RunnableDemo{
public static void main(String[] args){
MyThread my = new MyThread();
new Thread(my).start();
new Thread(my).start();
new Thread(my).start();
}
}

这两种方法一样的,只有执行了 start() 命令,才会开始执行线程。

其中继承 Thread 生成的线程每一个都是独立的,实现 Runnable 生成的线程是共享资源的,也就是我们上边的例子,最后输出的结果不一样:

  • 第一种方式每一个线程都独立执行了 for 循环操作(资源不共享),所以最后返回的结果将轮番打印 3 次 54321 的结果。
ticket = 5
ticket = 4
ticket = 3
ticket = 2
ticket = 1
ticket = 5
ticket = 4
ticket = 3
ticket = 2
ticket = 1
ticket = 5
ticket = 4
ticket = 3
ticket = 2
  • 而第二种方式中,由于 3 个 Thread 对象共同执行一个 Runnable 对象中的代码,所以实现了资源共享,最后打印出来的结果只有一次 54321,但这种方式容易造成线程的不安全
ticket = 5
ticket = 4
ticket = 3
ticket = 2
ticket = 1
总结:
  1. 继承 Thread 类的方法生成的线程每一个都是独立的,资源不能共享
  2. 实现 Runnable 接口生成的线程由于共用 Runnable 方法,彼此之间能实现资源共享,但是是线程不安全的,有必要执行加锁操作
  3. 只有执行 start() 操作,线程才会被创建执行
  4. 一般开发过程中我们都习惯使用实现 Runnable 接口创建线程类的方法,因为可以实现资源共享,比较符合企业需求

Thread 和 Runnable的更多相关文章

  1. java: Thread 和 runnable线程类

    java: Thread 和 runnable线程类 Java有2种实现线程的方法:Thread类,Runnable接口.(其实Thread本身就是Runnable的子类) Thread类,默认有ru ...

  2. Java 使用线程方式Thread和Runnable,以及Thread与Runnable的区别

    一. java中实现线程的方式有Thread和Runnable Thread: public class Thread1 extends Thread{ @Override public void r ...

  3. Android中Handler 、Thread和Runnable之间的关系

    在多线程编程的时候,我们经常会用到Handler,Thread和Runnable这三个类,我们来看看这三个类之间是怎么样的关系? 首先说明Android的CPU分配的最小单元是线程,Handler一般 ...

  4. Java 多线程(1)-Thread和Runnable

    一提到Java多线程,首先想到的是Thread继承和Runnable的接口实现 Thread继承 public class MyThread extends Thread { public void ...

  5. Java Thread and runnable

    java中可有两种方式实现多线程, 一种是继承Thread类,(Thread本身实现了Runnable接口,就是说需要写void run 方法,来执行相关操作) 一种是实现Runnable接口 sta ...

  6. Thread和Runnable、run和start的区别

    多线程可以通过两种方式来创建: 一.通过继承Thread类. 二.通过实现Runnable接口. 那么中两种方式到底有什么区别呢?那种方式更好些呢? 先看看几个简单的Demo: Demo1 publi ...

  7. 关于Thread的Runnable和Callable接口

    其实非常简单:其实他们的区别就是Callable有返回值并且可以抛出异常. /** * Represents a command that can be executed. Often used to ...

  8. Thread 与 Runnable 混合使用测试

    package com.dava; public class TesThread extends Thread implements Runnable { public void run() { Sy ...

  9. java多线程机制中的Thread和Runnable()区别

    1.java语言使用Thread类及其子类对象来表示线程,新建的一个线程声明周期中经历 新建.(声明一个线程,此时他已经有了相应的内存空间和其他资源),运行(线程创建之久就据用了运行的条件,一旦轮到使 ...

  10. JAVA多线程Thread VS Runnable详解

    要求 必备知识 本文要求基本了解JAVA编程知识. 开发环境 windows 7/EditPlus 演示地址 源文件   进程与线程 进程是程序在处理机中的一次运行.一个进程既包括其所要执行的指令,也 ...

随机推荐

  1. ICEM—非结构化周期网格

    原视频下载地址:https://yunpan.cn/cPBnmsNheJ46q  访问密码 3441

  2. super关键字和调用父类构造方法

    表示父类对象的默认引用 如果子类要调用父类被覆盖的实例方法,可用super作为调用者调用父类被覆盖的实例方法. 使用super调用父类方法 使用super调用父类的构造方法 调用构造方法 本类中调用另 ...

  3. ArcGIS超级工具SPTOOLS-数据处理篇

    1. 数据处理 1.1  两个图层按重叠度赋属性 两个面层按重合度赋属性,下图把依据赋数据属性图层,按重合度,赋值给目标.,重合度设置为负值,取面积最大的. 1.2  分区域消除 按区域字段值相同的, ...

  4. 错误注入 异常行为 环境变量或代码动态激活来触发这些异常行为 模拟错误 容错性 正确性 稳定性 宏 本质 macro

    小结: 1. 微服务中某个服务出现随机延迟.某个服务不可用. 存储系统磁盘 IO 延迟增加.IO 吞吐量过低.落盘时间长. 调度系统中出现热点,某个调度指令失败. 充值系统中模拟第三方重复请求充值成功 ...

  5. SQL-W3School-高级:SQL 通配符

    ylbtech-SQL-W3School-高级:SQL 通配符 1.返回顶部 1. 在搜索数据库中的数据时,您可以使用 SQL 通配符. SQL 通配符 在搜索数据库中的数据时,SQL 通配符可以替代 ...

  6. mongodb 报错 not authorized on admin to execute command【 version 3.2.18 】

    mongodb version 3.2.18 测试问题: 分析: 从报错内容上看是权限不够,但不明了为什么,因为已经使用的超级用户权限: { "_id" : "admin ...

  7. OpenNESS,开源的边缘网络服务平台

    目录 文章目录 目录 参考文章 OpenNESS 的电梯间演讲 OpenNESS 与 ETSI MEC Edge Controller Software 的功能清单 Edge Platform Sof ...

  8. 阶段5 3.微服务项目【学成在线】_day02 CMS前端开发_12-webpack研究-webpack安装

    npm默认安装配置的路径配置在nodejs的node_modules目录 j加上 -g 就是全局安装 后面只写webpack默认安装的是最新版本 指定版本号 视频中建议指定版本号进行安装

  9. python3 __mian和__name__的区别

    1.新建 test.py 模块: def GetModuleName(): print('__name__ = ', __name__) def PrintName(): print('PrintNa ...

  10. Linux nginx 会话保持(session)

    nginx 会话保持(session)有2种算法,一种是自带IP HASH 算法,一种是基于第三方模块sticky模块来实现会话保持 1)ip_hash 简单易用,但是有如下缺点 后端服务器宕机后,s ...