【JUC并发编程--java线程】

文章目录

  • 1. 线程
    • 1.1 线程的使用
    • 线程运行原理

1. 线程

1.1 线程的使用

方法一,直接使用 Thread
【JUC并发编程--java线程】_第1张图片
方法二,使用 Runnable 配合 Thread

把【线程】和【任务】(要执行的代码)分开

  • Thread 代表线程
  • Runnable 可运行的任务(线程要执行的代码)

【JUC并发编程--java线程】_第2张图片
【JUC并发编程--java线程】_第3张图片

Thread 与 Runnable 的关系

  • 方法1 是把线程和任务合并在了一起,方法2 是把线程和任务分开了
  • 用 Runnable 更容易与线程池等高级 API 配合
  • 用 Runnable 让任务类脱离了 Thread 继承体系,更灵活(聚合优于继承)

方法三,FutureTask 配合 Thread

FutureTask 能够接收 Callable 类型的参数,用来处理有返回结果的情况

FutureTask是Future和Runable的实现

这里的lambda表达式是对FutureTask的参数,它的参数是callable类型的
【JUC并发编程--java线程】_第4张图片
Future就是对于具体的Runnable或者Callable任务的执行结果进行取消、查询是否完成、获取结果。必要时可以通过get方法获取执行结果,该方法会阻塞直到任务返回结果。
【JUC并发编程--java线程】_第5张图片

线程运行原理

栈与栈帧
我们都知道 JVM 中由堆、栈、方法区所组成,其中栈内存给线程用,每个线程启动后,虚拟机就会为其分配一块栈内存。

  • 每个栈由多个栈帧(Frame)组成,对应着每次方法调用时所占用的内存
  • 每个线程只能有一个活动栈帧,对应着当前正在执行的那个方法
    【JUC并发编程--java线程】_第6张图片

线程上下文切换(Thread Context Switch)
因为以下一些原因导致 cpu 不再执行当前的线程,转而执行另一个线程的代码

  • 线程的 cpu 时间片用完
  • 垃圾回收
  • 有更高优先级的线程需要运行
  • 线程自己调用了 sleep、yield、wait、join、park、synchronized、lock 等方法

当 Context Switch 发生时,需要由操作系统保存当前线程的状态,并恢复另一个线程的状态,Java 中对应的概念就是程序计数器(Program Counter Register),它的作用是记住下一条 jvm 指令的执行地址,是线程私有的

  • 状态包括程序计数器、虚拟机栈中每个栈帧的信息,如局部变量、操作数栈、返回地址等
  • Context Switch 频繁发生会影响性能

你可能感兴趣的:(JUC并发编程,java)