夯实Java基础,一篇文章全解析线程问题
1. 线程是什么
操作系统支持多个应用程序并发执行,每个应用程序至少对应一个进程 ,彼此之间的操作和数据不受干扰,彼此通信一般采用管道通信、消息队列、共享内存等方式。当一个进程需要磁盘IO的时候,CPU就切换到另外的进程,提高了CPU利用率。
有了进程,为什么还要线程?因为进程的成本太高了。
启动新的进程必须分配独立的内存空间,建立数据表维护它的代码段、堆栈段和数据段,这是昂贵的多任务工作方式。线程可以看作轻量化的进程。线程之间使用相同的地址空间,切换线程的时间远小于切换进程的时间。
进程是资源分配的最小单位,而线程是CPU调度的最小单位。每一个进程中至少有一个线程,同一进程的所有线程共享该进程的所有资源,多个线程可以完成多个不同的任务,也就是我们常说的并发多线程。
2. 怎样创建线程
创建线程常用的有四种方式,分别是:
- 继承Thread类
- 实现Runnable接口
- 实现Callable接口
- 使用线程池创建
分别看一下怎么具体怎么使用代码创建的?
2.1 继承Thread类
public class ThreadDemo {
public static void main(String[] args) {
Thread thread = new MyThread();
thread.start(); // 启动线程
}
}
class MyThread extends Thread {
@Override
public void run() {
System.out.println("关注公众号:一灯架构");
}
}
输出结果:
关注公众号:一灯架构
start方法用来启动线程,只能被调用一次。
run方法是线程的核心方法,业务逻辑都写在run方法中。
2.2 实现Runnable接口
public class ThreadDemo {
public static void main(String[] args) {
MyRunnable myRunnable = new MyRunnable();
Thread thread1 = new Thread(myRunnable, "线程1");
Thread thread2 = new Thread(myRunnable, "线程2");
thread1.start(); // 启动线程1
thread2.start(); // 启动线程2
}
}
class MyRunnable implements Runnable {
private int count = 5;
@Override
public void run() {
while (count > 0) {
System.out.println(Thread.currentThread().getName()
+ ",关注公众号:一灯架构," + count--);
}
}
}
输出结果:
线程2,关注公众号:一灯架构,4
线程1,关注公众号:一灯架构,5
线程1,关注公众号:一灯架构,2
线程1,关注公众号:一灯架构,1
线程2,关注公众号:一灯架构,3
需要把Runnable实例放到Thread类中,才能执行,Thread对象才是真正的线程对象。
使用实现Runnable接口创建线程方式,相比继承Thread类创建线程,优点是:
- 实现的方式没有类的单继承性的局限性
- 实现的方式更适合来处理多个线程有共享数据的情况
2.3 实现Callable接口
public class ThreadTest {
public static void main(String[] args) throws ExecutionException, InterruptedException {
MyCallable myCallable = new MyCallable();
FutureTask<String> futureTask = new FutureTask<String>(myCallable);
Thread thread = new Thread(futureTask);
thread.start();
System.out.println(futureTask.get());
}
}
class MyCallable implements Callable {
@Override
public String call() throws Exception {
return "关注公众号:一灯架构";
}
}
输出结果:
关注公众号:一灯架构
实现Callable接口的线程实例对象,配合FutureTask使用,可以接收返回值。
2.4 使用线程池创建
public class ThreadDemo {
public static void main(String[] args) {
ExecutorService executorService = Executors.newFixedThreadPool(10);
executorService.execute(() -> System.out.println("关注公众号:一灯架构"));
}
}
输出结果:
关注公众号:一灯架构
使用线程池创建线程是工作开发中最常用的方式,优点是:
- 线程池帮忙管理对象的创建与销毁,减轻开发者工作量
- 线程池帮忙管理任务的调用,资源的创建与分配
- 复用线程和对象,提高使用效率
3. 线程的状态
线程共有6种状态,分别是NEW(初始化)、RUNNABLE(可运行)、WAITING(等待)、TIMED_WAITING(超时等待)、BLOCKED(阻塞)、TERMINATED(终止)。
NEW(初始化)
表示创建线程对象之后,还没有调用start方法。
RUNNABLE(可运行)
表示调用start方法之后,等待CPU调度。为了便于理解,通常又把RUNNABLE分别RUNNING(运行中)和READY(就绪)。处在RUNNING(运行中)状态的线程可以调用yield方法,让出CPU时间片,然后跟其他处于READY(就绪)一起等待被调度。
WAITING(等待)
处于RUNNABLE状态的线程调用wait方法之后,就处于等待状态,需要其他线程显示地唤醒。
TIMED_WAITING(超时等待)
处于RUNNABLE状态的线程调用wait(long)方法之后,就处于等待状态,需要其他线程显示地唤醒。
BLOCKED(阻塞)
等待进入synchronized方法/代码块,处于阻塞状态。
TERMINATED(终止)
表示线程已经执行结束。

4. 线程常用方法
说一下线程有哪些常用的方法。
| 方法定义 | 含义 | 使用方式 |
|---|---|---|
| public synchronized void start() {……} | 启动线程 | MyThread myThread = new MyThread(); myThread.start(); |
| public static native Thread currentThread(); | 获取当前线程实例对象 | Thread thread = Thread.currentThread(); |
| public static native void yield(); | 让出CPU时间片 | Thread.yield(); |
| public static native void sleep(long millis); | 睡眠指定时间 | Thread.sleep(1L); |
| public void interrupt() {……} | 中断线程 | MyThread myThread = new MyThread(); myThread.interrupt(); |
| public static boolean interrupted() {……} | 判断线程是否已中断 | MyThread myThread = new MyThread(); boolean interrupted = myThread.isInterrupted(); |
| public final native boolean isAlive(); | 判断线程是否是存活状态 | MyThread myThread = new MyThread(); boolean alive = myThread.isAlive(); |
| public final String getName() {……} | 获取线程名称 | MyThread myThread = new MyThread(); String name = myThread.getName(); |
| public State getState() {……} | 获取线程状态 | MyThread myThread = new MyThread(); Thread.State state = myThread.getState(); |
| public long getId() {……} | 获取线程ID | MyThread myThread = new MyThread(); long id = myThread.getId(); |
| public final void join() {……} | 等待其他线程执行完再执行 | MyThread myThread = new MyThread(); myThread.join(); |
我是「一灯架构」,如果本文对你有帮助,欢迎各位小伙伴点赞、评论和关注,感谢各位老铁,我们下期见

夯实Java基础,一篇文章全解析线程问题的更多相关文章
- 夯实Java基础系列3:一文搞懂String常见面试题,从基础到实战,更有原理分析和源码解析!
目录 目录 string基础 Java String 类 创建字符串 StringDemo.java 文件代码: String基本用法 创建String对象的常用方法 String中常用的方法,用法如 ...
- “全栈2019”Java多线程第九章:判断线程是否存活isAlive()详解
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java多 ...
- 夯实Java基础系列4:一文了解final关键字的特性、使用方法,以及实现原理
目录 final使用 final变量 final修饰基本数据类型变量和引用 final类 final关键字的知识点 final关键字的最佳实践 final的用法 关于空白final final内存分配 ...
- 夯实Java基础系列7:一文读懂Java 代码块和执行顺序
目录 Java中的构造方法 构造方法简介 构造方法实例 例 1 例 2 Java中的几种构造方法详解 普通构造方法 默认构造方法 重载构造方法 java子类构造方法调用父类构造方法 Java中的代码块 ...
- 夯实Java基础系列9:深入理解Class类和Object类
目录 Java中Class类及用法 Class类原理 如何获得一个Class类对象 使用Class类的对象来生成目标类的实例 Object类 类构造器public Object(); register ...
- 夯实Java基础系列10:深入理解Java中的异常体系
目录 为什么要使用异常 异常基本定义 异常体系 初识异常 异常和错误 异常的处理方式 "不负责任"的throws 纠结的finally throw : JRE也使用的关键字 异常调 ...
- 夯实Java基础系列11:深入理解Java中的回调机制
目录 模块间的调用 多线程中的"回调" Java回调机制实战 实例一 : 同步调用 实例二:由浅入深 实例三:Tom做题 参考文章 微信公众号 Java技术江湖 个人公众号:黄小斜 ...
- 夯实Java基础系列13:深入理解Java中的泛型
目录 泛型概述 一个栗子 特性 泛型的使用方式 泛型类 泛型接口 泛型通配符 泛型方法 泛型方法的基本用法 类中的泛型方法 泛型方法与可变参数 静态方法与泛型 泛型方法总结 泛型上下边界 泛型常见面试 ...
- 夯实Java基础系列14:深入理解Java枚举类
目录 初探枚举类 枚举类-语法 枚举类的具体使用 使用枚举类的注意事项 枚举类的实现原理 枚举类实战 实战一无参 实战二有一参 实战三有两参 枚举类总结 枚举 API 总结 参考文章 微信公众号 Ja ...
随机推荐
- 痞子衡嵌入式:浅析IAR下调试信息输出机制之半主机(Semihosting)
大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家分享的是IAR下调试信息输出机制之半主机(Semihosting). 在嵌入式世界里,输出打印信息是一种非常常用的辅助调试手段,借助打印信息,我 ...
- OpenCV CMake VSCode Windows 平台下运行配置及其解决方案
前言 最近在搞 计算机图形学相关的东西,有个 demo 用到了 opencv,找了 google 一圈,发现国内都没有比较好的配置和解决的办法,要不就是几年前的教程,最近正好踩坑完,其中经历了自己编译 ...
- EPIC限免提示
通过云函数每周定时推送限免内容到手机 import datetime import requests requests.packages.urllib3.disable_warnings() # da ...
- KingbaseES R6 集群repmgr witness 手工配置案例
使用见证服务器: 见证服务器是一个正常的KingbaseES实例,不是流复制群集的一部分; 其目的是,如果发生故障转移情况,则提供证明它是主服务器本身不可用的证据,而不是例如在不同物理位置之间的网络分 ...
- Linux云主机安全入侵排查步骤
导语 经常有用户报障系统被植入恶意程序,如挖矿软件.ddos攻击病毒.syn映射攻击病毒等,可以按照以下流程为用户排查入侵病毒类型: 一.定位病毒进程 对于用户反馈云主机性能卡顿,CPU和内存占用较高 ...
- 电商平台物流模块自建OR对接第三方物流平台
前沿 近几年来,电商行业竞争变得愈加激烈,公域流量获客成本越来越高,电商平台规则也越来越严格,数据无法出塔,商家无法自主运营用户群等等原因,很多大品牌纷纷开始搭建自有商城,运营私域流量,以此来降低 ...
- 自定义View5 -塔防小游戏:第二篇防御塔随意放置
第一篇:一个防御塔+多个野怪(简易版) 第二篇:防御塔随意放置 自定义View,处理事件分发,up,move,down. 第三篇:防御塔随意放置+多组野怪 第四篇:多波野怪 第五篇:杀死野怪获得金币 ...
- ConcurrentDictionary<T,V> 的这两个操作不是原子性的
好久不见,马甲哥封闭居家半个月,记录之前遇到的一件小事. ConcurrentDictionary<TKey,TValue>绝大部分api都是线程安全且原子性的, 唯二的例外是接收工厂委托 ...
- 痞子衡嵌入式:理解i.MXRT中FlexSPI外设lookupTable里配置访问行列混合寻址Memory的参数值
大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家介绍的是i.MXRT中FlexSPI外设lookupTable里配置访问行列混合寻址Memory的参数值. 关于 FlexSPI 外设的 loo ...
- 高性能 Java 计算服务的性能调优实战
作者:vivo 互联网服务器团队- Chen Dongxing.Li Haoxuan.Chen Jinxia 随着业务的日渐复杂,性能优化俨然成为了每一位技术人的必修课.性能优化从何着手?如何从问题表 ...