gball个人知识库
首页
基础组件
基础知识
算法&设计模式
  • 操作手册
  • 数据库
  • 极客时间
  • 每日随笔
  • 学习
  • 面试
  • 心情杂货
  • 实用技巧
  • 友情链接
  • 画图工具 (opens new window)
关于
  • 网盘 (opens new window)
  • 分类
  • 标签
  • 归档
项目
GitHub (opens new window)

ggball

后端界的小学生
首页
基础组件
基础知识
算法&设计模式
  • 操作手册
  • 数据库
  • 极客时间
  • 每日随笔
  • 学习
  • 面试
  • 心情杂货
  • 实用技巧
  • 友情链接
  • 画图工具 (opens new window)
关于
  • 网盘 (opens new window)
  • 分类
  • 标签
  • 归档
项目
GitHub (opens new window)
  • 面试

  • 数据库

  • linux

  • node

  • tensorFlow

  • 基础组件

  • 基础知识

    • java集合

    • jvm调优

    • java并发编程

      • 进程与线程
        • 进程与线程的含义
          • 进程:
          • 线程:
          • 个人理解的操作系统,cpu,进程,线程关系
          • 线程和进程的关系
        • 1.2 线程创建与运行
          • 1.继承Thead类创建线程
          • 2.实现Runnable接口创建线程
          • 3.使用Callable和Future创建线程
      • 多线程的方法介绍与使用
      • 线程死锁
      • 守护线程与用户线程
      • ThreadLocal了解
      • 什么是多线程并发编程
      • unsafe类了解
      • 伪共享
      • 锁的了解
      • ThreadLocalRandom原理解析
      • LongAdder,LongAccumulator类了解
      • 实践-创建多少线程合适
      • 线程通信——通知与等待
      • 缓存一致性问题
      • 利用Excutors异步执行任务
      • 线程池
      • 线程池操作数据库造成死锁
      • Java 浅拷贝和深拷贝的理解和实现方式
      • java内存模型JMM
      • 锁升级过程
      • io模型
      • 关键字介绍
      • AQS解析
    • java网络编程

    • java8新特性

    • javaAgent

    • java高级

  • 算法与设计模式

  • 分布式

  • 疑难杂症

  • go学习之旅

  • 极客时间

  • 知识库
  • 基础知识
  • java并发编程
ggball
2022-04-26

进程与线程

# 进程与线程的含义

# 进程:

进程是代码在数据集合上的一次运行活动 ,是系统进行资源分配和调度的基本单位,一个进程可以有多个线程。

# 线程:

进程中的一个执行任务(控制单元),负责当前进程中程序的执行。一个进程至少有一个线程,一个进程可以运行多个线程,多个线程可共享数据。

操作系统在分配资源时是把资源分配给进程 的,但是 CPU 资源 比较特殊 ,它是被分配到线程的 , 因为真正要占用 CPU 运行的是线程 , 所以也说线程是 CPU 分配的基本单位。

# 个人理解的操作系统,cpu,进程,线程关系

# 线程和进程的关系

image.png

# 程序计数器

是为了记录线程执行的指令地址,因为线程是被轮转分配时间片的,所以需要记录上一次执行的指令地址,方便下一次找到执行指令继续执行。

# 栈

用来保存方法的局部变量,该变量是私有的,除此之外,栈还用来存放线程的调用栈帧。

# 堆

堆是一个进程中最大的一块内存,堆是被进程中的所有线程共享的,是进程创建时分配的,堆里面主要存放使用 new 操作创建的对象实例 。

另外需要注意 的是,如果执行的是 native 方法,那么 pc 计数器记录 的是 undefined 地址,只有执行的是 Java 代码时 pc 计数器记录的才是下一条指令 的地址 。

# 1.2 线程创建与运行

使用java创建线程有三种方式 1.继承Thread类 2.实现Runnable接口 3.使用Callable和Future

# 1.继承Thead类创建线程

(1)继承Thread类并重写run方法 (2)创建线程对象 (3)调用该线程对象的start()方法来启动线程

public class CreateThreadTest {
public static void main(String[] args) {
    new ThreadTest().start();
    new ThreadTest().start();
    }
}
class ThreadTest extends Thread{
private int i = 0;
    @Override
    public void run() {
        for (; i < 100; i++) {
            System.out.println(Thread.currentThread().getName() + " is running: " + i);
        }
    }
 }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15

#

# 2.实现Runnable接口创建线程

(1)定义一个类实现Runnable接口,并重写该接口的run()方法 (2)创建 Runnable实现类的对象,作为创建Thread对象的target参数,此Thread对象才是真正的线程对象 (3)调用线程对象的start()方法来启动线程

public class CreateThreadTest {
public static void main(String[] args) {
    RunnableTest runnableTest = new RunnableTest();
    new Thread(runnableTest, "线程1").start();
    new Thread(runnableTest, "线程2").start();
	}
}
class RunnableTest implements Runnable{
private int i = 0;
@Override
public void run() {
    for (; i < 100; i++) {
        System.out.println(Thread.currentThread().getName()  + " is running: " + i);
    }
}
    }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16

#

# 3.使用Callable和Future创建线程

和Runnable接口不一样,Callable接口提供了一个call()方法作为线程执行体,call()方法比run()方法功能要强大:call()方法可以有返回值,可以声明抛出异常。

public interface Callable {
    V call() throws Exception;
}
1
2
3

Java5提供了Future接口来接收Callable接口中call()方法的返回值。 Callable接口是 Java5 新增的接口,不是Runnable接口的子接口,所以Callable对象不能直接作为Thread对象的target。针对这个问题,引入了RunnableFuture接口,RunnableFuture接口是Runnable接口和Future接口的子接口,可以作为Thread对象的target 。同时,Java5提供了一个RunnableFuture接口的实现类:FutureTask ,FutureTask可以作为Thread对象的target。

介绍了相关概念之后,使用Callable和Future创建线程的步骤如下: (1)定义一个类实现Callable接口,并重写call()方法,该call()方法将作为线程执行体,并且有返回值 (2)创建Callable实现类的实例,使用FutureTask类来包装Callable对象 (3)使用FutureTask对象作为Thread对象的target创建并启动线程 (4)调用FutureTask对象的get()方法来获得子线程执行结束后的返回值

import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.FutureTask;
public class CreateThreadTest {
public static void main(String[] args) {
	CallableTest callableTest = new CallableTest();
	FutureTask futureTask = new FutureTask<>(callableTest);
	new Thread(futureTask).start();
	try {
		System.out.println("子线程的返回值: " + futureTask.get());
	} catch (InterruptedException e) {
		e.printStackTrace();
	} catch (ExecutionException e) {
		e.printStackTrace();
	}
}
}
class CallableTest implements Callable{
@Override
public Integer call() throws Exception {
    int sum = 0;
    for (int i = 1; i < 101; i++) {
        sum += i;
    }
    System.out.println(Thread.currentThread().getName() + " is running: " + sum);
    return sum;
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28

创建线程的三种方式的对比 (opens new window)  可以看下

上次更新: 2025/06/04, 15:06:15
jvm垃圾收集器和内存分配策略
多线程的方法介绍与使用

← jvm垃圾收集器和内存分配策略 多线程的方法介绍与使用→

最近更新
01
AIIDE
03-07
02
githubActionCICD实战
03-07
03
windows安装Deep-Live-Cam教程
08-11
更多文章>
Theme by Vdoing
总访问量 次 | 总访客数 人
| Copyright © 2021-2025 ggball | 赣ICP备2021008769号-1
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式
×

评论

  • 评论 ssss
  • 回复
  • 评论 ssssssssssssssssssssssssssssssssssssssssssssssssssssssssssssssssss
  • 回复
  • 评论 ssssssssssssssssssssssssssssssssssssssssssssssssssssssssssssssssss
  • 回复
×