Thread 和 Runnable

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 接口创建线程类的方法,因为可以实现资源共享,比较符合企业需求

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏阿策小和尚

【Flutter 专题】12 ListView 用哪种方式绑定数据?

和尚昨天刚学习了一下底部状态栏 BottomNavigationBar 的基本使用方法,今天学习一下 ListView 的基本用法。 和...

14950
来自专栏阿策小和尚

【Flutter 专题】13 通过丑丑的【签到】页面学习以下【权重/比例】的重要性

和尚今天搭建了一个很丑的【签到】小页面,页面很简单,只有寥寥几个控件,但和尚想通过这个简单的小页面学习一下权重/比例的使用方式,顺便也学习了一下如何...

15130
来自专栏阿策小和尚

【Flutter 专题】22 易忽略的【小而巧】的技术点汇总 (二)

和继续整理一些个人不太注意但非常有用的小功能点,可能也是大家熟悉的,只是为了以后使用方便查找。

6520
来自专栏前端技术分享|前沿资讯|读书分享

vue全局拦截器配置

  项目中,我们经常遇到请求后台接口时要做后天返回的 code码验证判断。code码通常是一个特定的数值,比如一般返回 200作为正常请求,返回其他作为数据异常...

14520
来自专栏阿策小和尚

Android Java 动态修改 CheckBox 样式

和尚我一直在处理动态配置页面颜色方面的工作,包括各布局,各控件等,而和尚我却在最常用最基本的 CheckBox 选项框这个控件却栽了跟头,折腾了好久...

9410
来自专栏前端技术分享|前沿资讯|读书分享

JavaScript执行——Promise

  Promise构造函数接受一个函数作为参数,该函数的两个参数分别是 resolve和 reject。它们是两个函数,由JavaScript引擎提供,不用自己...

11520
来自专栏阿策小和尚

Android 实现圆角布局

和尚我最近在处理图片的圆角,不止是四个角全是圆角,还包括单左侧/单右侧/对角线方向的圆角。因为自己太菜只能寻求网上的大神,发现一个自定义圆角布局,这样可...

30810
来自专栏阿策小和尚

Android 极光推送与 WebView 日常问题小结

极光推送在日常的应用中应用广泛,集成简单送达率较高,和尚刚开始集成时很方便,但是随着项目的逐渐变大,分包和组件化等应用比较多,此时单独出 push ...

14830
来自专栏阿策小和尚

Android Color 色值小结

和尚我最近在处理主题色方面的问题,有个小需求是处理更改颜色,判断色值等,稍稍整理了一下。 Color 大家都很熟悉,其组成方式是 RGB...

13930
来自专栏阿策小和尚

Android 沉浸式状态栏的多种样式

和尚我最近正在处理客户端顶部沉浸式展示图片,借此整理了一下和尚自己研究测试的沉浸式状态栏。 沉浸式状态栏大家都很熟悉,即 APP 界面图...

26230

扫码关注云+社区

领取腾讯云代金券

年度创作总结 领取年终奖励