Java 多线程 笔记

news2024/12/27 18:00:44

文章目录

  • 实现方式
    • 1. 通过Thread类
    • 2. 通过Runnable接口
    • 3. 通过Callable接口
  • 线程状态
    • 线程方法
      • 1. 线程停止
      • 2. 线程休眠`sleep`
      • 3. 线程礼让`yield`
      • 4. 线程强制执行`join`
      • 5. 线程状态观测`getState`
      • 6. 线程优先级
  • 守护线程(daemon
  • 线程同步 TODO
  • 线程死锁
  • `Lock`锁(可重入锁
  • 线程协作 TODO
  • 线程池

实现方式

1. 通过Thread类

  1. 继承Thread类,重写run方法。run是线程执行体
  2. 通过继承类对象,调用Threadstart方法

请添加图片描述

2. 通过Runnable接口

  1. 实现Runnable接口,重写run方法。run是线程执行体
  2. 将实现Runnable接口的类对象作为参数传入Thread,创建Thread对象
  3. 调用Thread对象的start方法

请添加图片描述

3. 通过Callable接口

线程状态

线程方法

  • setPriority(int newPriority):更改线程的优先级
  • static void sleep(long millis):指定毫秒数内让当前执行的线程休眠
  • void join:等待线程终止
  • static void yield():暂停当前执行的线程对象,执行其他进程
  • void interrupt():中断线程。(建议不用)
  • boolean isAlive():测试线程是否处于活动状态

1. 线程停止

  1. 建议线程正常停止。利用次数,不建议死循环
  2. 建议使用标志位
  3. 不要使用stopdestory等过时或者JDK不建议使用的方法

以下的写法并不安全,只是演示一下
请添加图片描述

2. 线程休眠sleep

  1. sleep(time):指定当前线程阻塞的毫秒数
  2. sleep存在异常InterruptedException
  3. sleep时间达到后,线程进入就绪状态
  4. sleep可以模拟网络延时,倒计时等
  5. 每个对象都有一个锁,sleep不会释放锁

3. 线程礼让yield

  1. 让线程从运行状态转为就绪状态
  2. 礼让不一定成功

4. 线程强制执行join

  1. 想象为插队
  2. join之后,其他线程被阻塞,直到该线程执行完
  3. 需要抛出InterruptedException异常

请添加图片描述

5. 线程状态观测getState

线程状态包括

  • NEW:尚未启动
  • RUNNABLE:在Java虚拟机中执行的线程处于该状态
  • BLOCKED:被阻塞等待监视器锁定的线程处于该状态
  • WAITING:正等待另一个线程执行特定动作的线程处于该状态
  • TIMED_WAITING:正等待另一个线程执行动作达到指定等待时间的线程处于此状态
  • TERMINATED:已退出的线程处于此状态

请添加图片描述

6. 线程优先级

  1. Java提供线程调度器来监控程序中启动后进入就绪状态的所有线程,线程调度器按照优先级决定应该调度哪个线程执行
  2. 线程优先级用数字表示,范围是110
  3. getPriority()获得优先级
  4. setPriority()设置优先级
  • MIN_PRIORITY = 1:最小优先级
  • MAX_PRIORITY = 10:最大优先级
  • NORM_PRIORITY = 5:默认优先级
  1. 优先级低只是意味着获得调度的概率低,并不是不会被调用,都要看CPU调度
  2. 优先级的设定建议在start之前

守护线程(daemon

  1. 线程分为用户线程守护线程,后者例如:后台记录操作日志、监控内存、垃圾回收等待
  2. 虚拟机必须确保用户线程执行完毕
  3. 虚拟机不用等待守护线程执行完毕
  4. 设置线程为守护线程setDaemon(true)

请添加图片描述

线程同步 TODO

解决多个线程操作同一个资源的问题

  1. 访问时加入锁机制synchronized,当一个线程获得对象的排它锁,独占资源,其他线程必须等待,使用后释放锁
  2. 一个线程持有锁会导致其他所有需要此锁的线程挂起
  3. 多线程竞争下,加锁、释放锁会导致较多上下文切换和调度延时,引起性能问题
  4. 如果一个优先级高的线程等待一个优先级低的线程释放锁,会导致优先级倒置,引起性能问题

在这里插入图片描述
⚠️ synchronized默认锁this。有些时候需要同步块

  • 同步块:synchronized(Obj) {}Obj是同步监视器,可以是任何对象,但是推荐使用共享资源作为同步监视器

线程死锁

  1. 多个线程各自占有一些共享资源,并且相互等待其他线程占有的资源才能运行,导致多个线程都在等待对方释放资源,都停止执行。
  2. 某一个同步块同时拥有“两个以上对象的锁”时,就可能发生死锁

死锁产生条件:

  1. 互斥条件:一个资源每次只能被一个进程使用
  2. 请求与保持条件:一个进程因为请求资源阻塞时,对已获得的资源保持不放
  3. 不剥夺条件:进程已获得的资源,在没有使用完之前,不能强行剥夺
  4. 循环等待条件:若干进程之间形成一种头尾相接的循环等待资源关系
public class TestDeadLock {
    public static void main(String[] args) {
        MakeUp g1 = new MakeUp(0, "syb");
        MakeUp g2 = new MakeUp(1, "yyj");
        g1.start();
        g2.start();
    }
}

class Mirror{}
class Table{}
class MakeUp extends Thread{
    static final Mirror mirror = new Mirror();
    static final Table table = new Table();
    int choice;
    String name;
    MakeUp(int choice, String name){
        this.choice = choice;
        this.name = name;
    }
    @Override
    public void run(){
        makeup();
    }
    public void makeup(){
        if(choice == 0){
            System.out.println("请求Mirror的锁");
            synchronized (mirror){
                System.out.println("获得Mirror的锁");
                System.out.println("请求Table的锁");
                synchronized (table){   // 有一把锁,还请求另一把,可能死锁
                    System.out.println("获得Table的锁");
                }
            }
        }else{
            System.out.println("请求Table的锁");
            synchronized (table){
                System.out.println("获得Table的锁");
                System.out.println("请求Mirror的锁");
                synchronized (mirror){  // 有一把锁,还请求另一把,可能死锁
                    System.out.println("获得Mirror的锁");
                }
            }
        }
    }
}

Lock锁(可重入锁

  1. 从JDK 5.0开始,Java提供更强大的线程同步机制,可以显式定义同步锁对对象实现同步
  2. Lock提供对共享资源的独占访问,每次只能有一个线程对Lock对象加锁,线程开始访问共享资源前需要先获得Lock对象
  3. ReentrantLock类实现了Lock,拥有与synchronized相同的并发性和内存语义,可以显式加锁lock()、释放锁unlock()

请添加图片描述

线程协作 TODO

Java提供以下方法解决进程之间的通信问题。均为Object类的方法,只能在同步方法或同步代码块中使用,否则会抛出异常IllegalMonitorStateException

  • wait():表示线程一直等待,直到其他线程通知。与sleep不同的是,会释放锁
  • wait(long timeout):指定等待的毫秒数
  • notify():唤醒一个等待的进程
  • notifyAll():唤醒同一个对象上所有调用wait()方法的线程,优先级高的线程优先调度

线程池

提前创建多个线程,放入线程池中,使用时直接获取,使用完放回池中,可以避免频繁创建销毁、实现重复利用

  1. JDK 5 提供线程池相关API:ExectuorServiceExecutors
  2. 真正的线程池接口:ExecutorService。常见子类ThreadPoolExecutor
  3. Executors:工具类、线程池的工厂类,用于创建并返回不同类型的线程池
  • void execute(Runnable command):执行命令,没有返回值
  • <T> Future<T> submit(Callable<T> task):执行任务,有返回值
  • void shutdown():关闭连接池

请添加图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/172885.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Typora+Gitee+PicGo搭建图床

引言 markdown原则上不建议使用base64内嵌图片&#xff0c;因为太麻烦。 如果只是在本机浏览、编辑的话&#xff0c;那引用相对路径或者绝对路径即可&#xff0c;但是考虑到要发布、分享的情况&#xff0c;使用图床是比较好的解决方案。 本教程可以快速得到一个相对稳定的免…

单片机数据、地址、指令、控制总线结构

数据、地址、指令&#xff1a;之所以将这三者放在一同&#xff0c;是因为这三者的实质都是相同的──数字&#xff0c;或者说都是一串‘0’和‘1’组成的序列。换言之&#xff0c;地址、指令也都是数据。 指令&#xff1a;具体可参考文章 由单片机芯片的设计者规则的一种数字…

c程序gcc编译常见报错及解决方法整理

目录一、简介二、常见报错及解决方法1、数组定义错误2、Not enough information to produce a SYMDEFs file3、文件乱码<U0000>4、未定义或未申明报错5、代码中误加入中文三、其他相关链接一、简介 本文主要是整理c程序编译过程的常见报错的解决方法&#xff0c;方便大家…

Leetcode.312 戳气球

题目链接 Leetcode.312 戳气球 题目描述 有 n个气球&#xff0c;编号为0到 n - 1&#xff0c;每个气球上都标有一个数字&#xff0c;这些数字存在数组 nums中。 现在要求你戳破所有的气球。戳破第 i 个气球&#xff0c;你可以获得 nums[i−1]∗nums[i]∗nums[i1]nums[i - 1]…

C++ 类 对象初学者学习笔记

C 类 & 对象 访问数据成员 #include <iostream>using namespace std;class Box {public:double length; // 长度double breadth; // 宽度double height; // 高度// 成员函数声明double get(void);void set( double len, double bre, double hei ); }; // 成员…

两种在CAD中加载在线卫星影像的方法

概述 经常使用CAD的朋友应该会有这样的一个烦恼&#xff0c;就是当加载卫星图到CAD中进行绘图的时候&#xff0c;由于CAD本身的限制和电脑性能等原因&#xff0c;往往不能加载太大的地图图片到CAD内&#xff0c;这里给大家介绍两种在CAD内加载在线卫星影像的方法&#xff0c;希…

用docker部署webstack导航网站-其一

序言 认识的好朋友斥资买了一个NAS&#xff0c;并搭建了webdev和其他一些web应用&#xff0c;用于存放电子书、电影&#xff0c;并用alist搭建了一个网盘。现在他还缺少一个导航页&#xff0c;于是拖我给他做一个导航页。我欣然接受了&#xff0c;他想做一个和TBox导航&#x…

Arduino开发:网页控制ESP8266三色LED灯闪烁

根据板卡原理RGB三色LED对应引脚&#xff1a;int LEDR12、int LEDG14、int LEDB13;设置串口波特率为115200Serial.begin(115200);源代码如下所示&#xff1a;#include <ESP8266WiFi.h> // 提供 Wi-Fi 功能的库#include <ESP8266WebServer.h> // 提供网站服务器功能…

分布式共识算法随笔 —— 从 Quorum 到 Paxos

分布式共识算法随笔 —— 从 Quorum 到 Paxos 概览: 为什么需要共识算法&#xff1f; 昨夜西风凋碧树&#xff0c;独上高楼&#xff0c;望尽天涯路 复制(Replication) 是一种通过将同一份数据在复制在多个服务器上来提高系统可用性和扩展写吞吐的策略, 。常见的复制策略有主从…

Flink CDC 原理

文章目录CDC&#xff0c;Change Data Capture 变更数据捕获 目前CDC有两种实现方式&#xff0c;一种是主动查询、一种是事件接收。 主动查询&#xff1a; 相关开源产品有Sqoop、Kafka JDBC Source等。 用户通常会在数据原表中的某个字段中&#xff0c;保存上次更新的时间戳或…

一篇博客教会你写序列化工具

文章目录什么是序列化&#xff1f;序列化格式JSON序列化精简序列化数据总结源码什么是序列化&#xff1f; 总所周知&#xff0c;在Java语言中&#xff0c;所有的数据都是以对象的形式存在Java堆中。 但是Java对象如果要存储在别的地方&#xff0c;那么单纯的Java对象就无法满…

我靠steam/csgo道具搬运实现财富自由

和莘莘学子一样&#xff0c;本着毕业对未来的憧憬&#xff0c;规划着漫漫人生&#xff0c;可是被残酷的事实打败。 直到一次偶然的同学聚会&#xff0c;谈及了现如今的生活才发现一片新大陆&#xff0c;通过信息差去赚取收益。 记得之前在校期间常常和哥几个通宵干CSGO,直到这…

Elasticsearch7.8.0版本高级查询—— 完全匹配查询文档

目录一、初始化文档数据二、完全匹配查询文档2.1、概述2.2、示例一、初始化文档数据 在 Postman 中&#xff0c;向 ES 服务器发 POST 请求 &#xff1a;http://localhost:9200/user/_doc/1&#xff0c;请求体内容为&#xff1a; { "name":"zhangsan", &quo…

【Nginx】使用Docker完成Nginx负载均衡+动静分离

前提是需要配置Nginx的反向代理&#xff0c;可以我看之前的文章 上篇Nginx配置动态代理的文章&#xff0c;我们在tomcat里写了两个简单html 这次我们依然采取同样的思路来演示负载均衡 一、负载均衡 1.在两个Tomcat容器&#xff08;我这里一个端口8081&#xff0c;一个8082…

Gradle vs Maven 基本对比(一)

Gradle 与Maven 的基本对比 对比目录&#xff1a; 1、工具包目录对比 2、创建项目结构对比 3、启动进程对比 4、性能对比 5、简洁性对比 什么是gradle: Gradle 是一个开源的运行在JVM上自动化构建工具&#xff0c;专注于灵活性和性能。Gradle 使用 Groovy 或 Kotlin DSL(领…

低代码平台飞速创软完成3000万元A+轮融资

疫情形势下&#xff0c;云原生全场景低代码及数字化基础设施提供商珠海飞速创软科技有限公司&#xff08;以下简称&#xff1a;飞速创软&#xff09;依然发展迅速&#xff0c;逆势而上。继2021年中获得珠海正菱创投、炼金术资本等机构A轮数千万融资之后&#xff0c;于2022年底&…

【手写 Vue2.x 源码】第三十一篇 - diff算法-比对优化(下)

一&#xff0c;前言 上篇&#xff0c;diff算法-比对优化&#xff08;上&#xff09;&#xff0c;主要涉及以下几个点&#xff1a; 介绍了如何进行儿子节点比对&#xff1b;新老儿子节点可能存在的3种情况及代码实现&#xff1b;新老节点都有儿子时的 diff 方案介绍与处理逻辑…

墙裂推荐,2023年最强、最实用的IDEA插件推荐合集

插件目录Alibaba Java Coding Guidelines(阿里巴巴java开发规范)Alibaba Cloud AI Coding Assistant(阿里云AI代码助理)Code Glance3(代码地图)Codota AI Autocomplete for Java and JavaScriptCSDN Tools(CSDN官方插件)FindBugsGenerateAllSetter Postfix Completion (自动生成…

小程序uni-app的api

小程序uni-app的apiuni api简介uni api使用uni-app自定义组件—传统方式核心步骤uni-app自定义组件—easycom简介核心步骤uni-app组件库uViewUIuview介绍关键步骤uni api简介 uni-api 指的是uni-app 针对一些 微信小程序api所做的封装它解决了两个问题 原生的小程序api不支持…

C/C++const关键字详解(全网最全)

目录 1、const修饰普通变量 2、const修饰指针 &#xff08;1&#xff09;const修饰p: &#xff08;2&#xff09;const修饰*p&#xff1a; &#xff08;3&#xff09;const修饰p和*p 4、const修饰数组 5、const修饰函数形参 &#xff08;1&#xff09;const修饰普通形参…