操作系统-多线程案例
一、单例模式(是一种设计模式)
设计模式有很多种,不同的语法中也有不同的设计模式
单例 = 单个实例(对象)
某个类,在一个进程中,只应该创建出一个实例,(原则上不该有多个)使用单例模式可以对咱们的代码进行个更严格的校验和检查
实现单例模式有多种实现方法,有两种最基础的实现方式
1、饿汉模式
饿表示非常迫切,因为实例在类加载的时候就创建了,创建时机非常早,程库一启动,实例就创建了
2、懒汉模式
创建实例的时机不大一样,创建实例的时机更晚,只到第一次使用才会创建实例
首次调用getlnstane,此时 instance 引用为 null,就会进入 if 条件,从而把实例创建出来,后续再调用 getInstance ,由于 instance 不再是null,此时不会进入 if ,直接返回之前创建好的引用
该操作的执行时机并不确定,甚至可能整个程序压根用不到这些方法,就把创建的操作省下来了
饿汉模式: getInstance 直接返回 Instance 实例,该操作的本质就是读操作,多个线程同时读是线程安全的
懒汉模式:有读有写,是线程不安全的
1、该线程安全问题,本质上是读,比较,写这三个操作不是原子的,导致后一个线程读的值,前一个线程还没写(脏读),把锁加在外面才能保证读和修改是一个整体
2、代码每次 getInstance 都要加锁,但加锁只是在 new 出对象前加上很必要,对象new完以后,后续调用getinstance是读操作,不用加锁
可以给上述代码一个判定:对象未创建才加锁,创建了就不加锁了
3、指令重排序引起的线程问题,指令重排序也是编译器优化的一种方式
调整原有代码的执行顺序,保证逻辑不变的情况下提高程序的效率,解决该问题核心思路还是volatile
二、阻塞队列(基于普通队列做出的扩展,先进先出,线程安全的)
1、带有阻塞性质
若是对一个已经满了的队列进行入队列,此时入队列就会阻塞,一直到队列不满结束
若是对一个已空的队列进行出队列,此时出队列就会阻塞,一直到队列不空结束
2、基于阻塞队列,就可以实现“生产者消费者模型”
引入生产者消费者模型是为了更好做到“解耦合“
实际开发中,经常会涉及到分布式系统(服务器的功能不是由一个服务器来完成的,而是每个服务器负责一部分功能,最后通过服务器之间的网络通信,完成整个功能)
3、阻塞队列代价:需加机器,引入更多的硬件资源
1、上述阻塞队列并非是简单的数据结构,而是基于这个数据结构实现的服务器程序,又被布署到单独的主机上了(被称为消息队列message queue)
2、整个系统的结构更复杂了,要维护的服务器更多了
3、效率会降低,引入中间商还是有差价的,请求从A发送给B,这个过程要经历队列的转发,也是有一定开销的
4、削峰填谷
外界请求突发峰值,突发的峰值就会使抗压能力更弱的服务器挂掉,使用阻塞队列,即使外面的请求出现峰值,也是由队列来承受峰值请求,队列无业务逻辑,只是存储数据,抗压能力是比较强的
当请求出现谷值,队列也会照常提供给服务器请求,让服务器保证正常运转
5、Java中提供了现成的阻塞队列数据结构
ArrayBlockingQueue 基于数组实现
LinkedBlackingQueue 基于链表实现
PriorityBlackingQueue 带优先级的队列
put 方法是带阻塞的加入,offer 不带阻塞,take方法出队列也是带阻塞的
对程序的优化,归根结底增加机器、增加硬件资源是最立杆见影的做法,仅仅是软件层面的优化,投入产出比较低
实际开发中,生产者消费者模型,住往是多个生产者,消费者,这里的生产者和消费者不仅仅是一个线程,也可能是一个独立的服务器程序,其至是一组服务器程序
生产者消费者模型核心仍是阻塞队列,使用synchronized 和 wait/notify来达到线程安全和阻塞
三、实现定时器
设定一个时间,当时间到了的时候,定时器自动去执行某个逻辑
什么样的情况能的使用lambda,得是函数式接口(interface只能有这一个接口)
Timer里内置了线程(前台线程),timer 不知道代码是否会添加新的任务进来,处于严阵以待的状态,要使用 cncel 主动结束否则 timer 不知道是否其他地方要继续添加任务
定时器核心:
1、有个扫描线程,负责判定时间到/执行任务
2、还要有个数据结构保护所有被注册的任务
数据结构用优先级队列来表示(在多线程下使用,调用 schedule 是一个线程,扫描是另一个线程,要关注线程安全问题)
四、线程池
提前把要用的对象准备好,用完的对象也不要立刻释放,先留着下次备用
虽然线程成本低,但是频繁的创建和销毁开销还是很大的
1、优化方案
1、引入轻量级线程:也称为纤程/协程,协程本质是程序员在用户代码中进行调度,不是靠内核的调度器调度,节省了很多调度上的开销
2、线程池
在使用过程中并未真的频繁创建销毁,而只是从线程池里取线程,使用完了还给线程池,从线程池中取线程(纯用户代码,可控)
通过系统申清来创建线程,需要内核来完成,(不太可控)
2、标准库中的线程池 ThreadPoolExecutor
这个类,构造方法的参数,侧面映射出线程池的设计思路,需要咱们了解一下标准库提供的线程池,持有的线程个数并非是一成不变的,会根据当前任务量自适应线程数
int corePoolSize 核心线程数(线程池内最少有多少线程)
int maximumpoolSize 最大线程数(线程池内最多有多少线程)
kpepAliveTime 实习生线程空闲时间超过了这个时间阈值,就会被销毁
BlockingQueue<Runnabe> 使用Runnable作为描述任务的主体,若想任务带有优先级,则使用PriorityBlrkingQueue
ThreadFactory 线程工厂(工厂模式也是常见的设计模式)
通过静态方法封装 new 操作,在方法内部没定不同属性完成对象初始化,构造对象的过程就是工厂模式
线程工厂提供了方法,让方法封装 new Thread 操作并给 Thread 设置一些属性
3、RejectedExecutionHandler(拒绝策略)
线程池能容纳的元素是有限的,若继续添加任务会怎么样
AborPolicy:继续添加任务,直接抛出异常,新的任务未完成,旧的任务也没法做了
calerRunsPolicy: 新的任务由添加任务的线程执行
discardoldestPolicy:丢弃最老的任务
discardPolicy:丢弃最新任务(调用的线程不执行该任务,线程池也不执行)
创建线程池要没定线程池的线程数量,若一个进程中所有的线程都是cpu密集型,每个线程所有工作都是在cpu上执行的,此时线程数不该超过N(cpu核心数)
若一个进程的所有线程都是IO密集型,每个线程大部分工作都等待IO,cpu消耗很少,此时线程数可以很多,远远超过N
综上,由于程序的复杂性很难对于线程池的线程数进行估数,可通过测试的方式,尝试给线程池设定不同的线程数目分别进行能测试,衡量每种程数目下总的时间开销和系统资源占用的开销,找到二者的合适值
相关文章:
操作系统-多线程案例
一、单例模式(是一种设计模式) 设计模式有很多种,不同的语法中也有不同的设计模式 单例 单个实例(对象) 某个类,在一个进程中,只应该创建出一个实例,(原则上不该有多个ÿ…...

什么是FUSE用户态文件系统
零. 文件系统 1. 为什么要有文件系统 文件系统是操作系统中管理文件和目录的一种机制。它提供了组织、存储、检索和更新文件的方法,主要如下: 数据组织:文件系统将数据组织成文件和目录,使用户能够更方便地管理和查找文件。每个…...
[每日一练]销售分析(通过数据的0/1转换进行是否存在的查询)
#该题目来源于力扣: 1083. 销售分析 II - 力扣(LeetCode) 题目要求: 表:Product----------------------- | Column Name | Type | ----------------------- | product_id | int | | product_name | varch…...

.NET Core WebApi第7讲:项目的发布与部署
一、理解 前端跟后端拿数据,然后在前端页面中展示,就是我们要完成的事情。 把前端跟后端开发好之后,我们需要落地部署,这个时候就需要一个服务器。 服务器就是一台电脑,只要windows里面有一个叫IIS的管理器。 二、项目…...
【python 将数据写入csv文件】正确方式
data [{username: jack, password: 1234}, ……]# 保存为CSV文件 with open(IP_output.csv, w, newline, encodingutf-8) as file:fieldnames [username, password]writer csv.DictWriter(file, fieldnamesfieldnames, quotingcsv.QUOTE_NONE)writer.writeheader() # 写入列…...
OpenCV4.8 开发实战系列专栏之 10 - 像素值统计
大家好,欢迎大家学习OpenCV4.8 开发实战专栏,长期更新,不断分享源码。 专栏代码全部基于C++ 与Python双语演示,专栏答疑群 请联系微信 OpenCVXueTang_Asst 本文关键知识点:像素值统计 最小(min)最大(max)均值(mean)标准方差(standard deviation)API知识点 最大最小值min…...

pandas计算相关性并画热力图
实现这个功能有很多方法,但是下面的方法还是比较优雅的: cols ["ASSET", "HOUSES", "INCOME", "DEBT", "EDUC"] corr df[cols].corr() corr.style.background_gradient(axisNone)讲解: …...

初始Docker
概述: 容器,作为云原生技术的重要组成部分,与虚拟机一样,均属于虚拟化技术的范畴。然而,容器技术以其独特的优势,在虚拟化领域中脱颖而出。与虚拟机不同,容器能够摆脱操作系统的束缚࿰…...

Redis-概念、安装、基本配置
文章目录 一、Redis及Redis集群概念、分布式系统概念一-1 Redis是什么一-2 什么是分布式系统、分布式缓存一-3 什么是Redis集群、实现Redis集群的方法有哪些、这些跟Redis的sentinel和cluster有什么关系一-4 Redis的库一-5 Redis中的Key与Value是什么、如何进行操作使用它们添加…...

qt QPlainTextEdit详解
QPlainTextEdit是一个功能强大、易于使用的纯文本编辑器/查看器。它使用与QTextEdit相同的技术和概念,但是为纯文本的处理进行了优化,因此更适合处理大型纯文本文档。QPlainTextEdit不提供富文本编辑功能,如字体、颜色、大小等的格式化&#…...

【机器学习】23. 聚类-GMM: Gaussian Mixture Model
1. 定义和假设 定义:probabilistic clustering(model-base) 假设:数据服从正态分布 2. 算法内容 我们假设数据是由k个高斯(正态)分布混合生成的。每个分布有2个参数:μ和σ。 一个分布对应一…...

深度探索C++对象模型
文章目录 前言一、关于对象C对象模型 二、构造函数实例分析 拷贝构造函数程序转化语意学(Program Transformation Semantics)成员初始化列表 三、数据语义学(The Semantics of Data)数据存取多种继承情况讨论仅单一继承加上虚函数多重继承虚拟继承 Pointer to Data Members 四、…...

电脑怎么设置开机密码:保障个人信息安全的第一步
在数字化时代,个人信息的安全至关重要。电脑作为我们日常工作和生活中不可或缺的设备,存储了大量的私人数据和敏感信息。为了防止未经授权的访问,设置开机密码是保护个人隐私和信息安全的基本措施之一。本文将详细介绍如何在不同操作系统下为…...

MybatisPlus入门(六)MybatisPlus-null值处理
一、MybatisPlus-null值处理 1.1)问题引入: 在查询中遇到如下情况,有部分筛选条件没有值,如商品价格有最大值和最小值,商品价格部分时候没有值。 1.2)解决办法: 步骤一:新建查询实…...

红帽认证有必要考吗?这四大人群推荐考取!
红帽认证(Red Hat Certification)作为全球公认的Linux技能认证,对于某些特定人群来说,考取这一认证无疑是一个明智的选择。本文将探讨红帽认证的必要性,并为四类人群提供考取红帽认证的建议。 1. IT专业人士 对于IT专业人士来说࿰…...

基于SSM+微信小程序的社团登录管理系统(社团1)
👉文末查看项目功能视频演示获取源码sql脚本视频导入教程视频 1 、功能描述 2、项目技术 3、开发环境 4、功能介绍 1、项目介绍 基于SSM微信小程序的社团登录管理系统实现了管理员及社团、用户。 1、管理员实现了首页、用户管理、社团管理、社团信息管理、社…...
html中cookie如何存储
在HTML中,可以使用JavaScript来创建、读取和删除cookie。以下是创建和读取cookie的基本示例: 创建cookie: function setCookie(name, value, daysToLive) { var cookie name "" encodeURIComponent(value); if (typeof daysToLive …...

C++基础三(构造函数,形参默认值,函数重载,单例模式,析构函数,内联函数,拷贝构造函数)
C有六个默认函数,分别是: 1、默认构造函数; 2、默认拷贝构造函数; 3、默认析构函数; 4、赋值运算符; 5、取址运算符; 6、取址运算符const; 构造函数 构造函数(初始化类成员变量): 1、属于类的成员函数之一 …...

Flutter Color 大调整,需适配迁移,颜色不再是 0-255,而是 0-1.0,支持更大色域
在之前的 3.10 里, Flutter 的 Impeller 在 iOS 上支持了 P3 广色域图像渲染,但是当时也仅仅是当具有广色域图像或渐变时,Impeller 才会在 iOS 上显示 P3 的广色域的颜色,而如果你使用的是 Color API,会发现使用的还是…...

如何使用VBA识别Excel中的“单元格中的图片”(2/2)
Excel 365升级了新功能,支持两种不同的插入图片方式: 放置在单元格中(Place in cell),新功能,此操作插入的图片下文中简称为单元格中的图片。放置在单元格上(Place over cell)&…...

AI-调查研究-01-正念冥想有用吗?对健康的影响及科学指南
点一下关注吧!!!非常感谢!!持续更新!!! 🚀 AI篇持续更新中!(长期更新) 目前2025年06月05日更新到: AI炼丹日志-28 - Aud…...

基于FPGA的PID算法学习———实现PID比例控制算法
基于FPGA的PID算法学习 前言一、PID算法分析二、PID仿真分析1. PID代码2.PI代码3.P代码4.顶层5.测试文件6.仿真波形 总结 前言 学习内容:参考网站: PID算法控制 PID即:Proportional(比例)、Integral(积分&…...
日语学习-日语知识点小记-构建基础-JLPT-N4阶段(33):にする
日语学习-日语知识点小记-构建基础-JLPT-N4阶段(33):にする 1、前言(1)情况说明(2)工程师的信仰2、知识点(1) にする1,接续:名词+にする2,接续:疑问词+にする3,(A)は(B)にする。(2)復習:(1)复习句子(2)ために & ように(3)そう(4)にする3、…...
.Net Framework 4/C# 关键字(非常用,持续更新...)
一、is 关键字 is 关键字用于检查对象是否于给定类型兼容,如果兼容将返回 true,如果不兼容则返回 false,在进行类型转换前,可以先使用 is 关键字判断对象是否与指定类型兼容,如果兼容才进行转换,这样的转换是安全的。 例如有:首先创建一个字符串对象,然后将字符串对象隐…...

用机器学习破解新能源领域的“弃风”难题
音乐发烧友深有体会,玩音乐的本质就是玩电网。火电声音偏暖,水电偏冷,风电偏空旷。至于太阳能发的电,则略显朦胧和单薄。 不知你是否有感觉,近两年家里的音响声音越来越冷,听起来越来越单薄? —…...
LRU 缓存机制详解与实现(Java版) + 力扣解决
📌 LRU 缓存机制详解与实现(Java版) 一、📖 问题背景 在日常开发中,我们经常会使用 缓存(Cache) 来提升性能。但由于内存有限,缓存不可能无限增长,于是需要策略决定&am…...

[ACTF2020 新生赛]Include 1(php://filter伪协议)
题目 做法 启动靶机,点进去 点进去 查看URL,有 ?fileflag.php说明存在文件包含,原理是php://filter 协议 当它与包含函数结合时,php://filter流会被当作php文件执行。 用php://filter加编码,能让PHP把文件内容…...
MinIO Docker 部署:仅开放一个端口
MinIO Docker 部署:仅开放一个端口 在实际的服务器部署中,出于安全和管理的考虑,我们可能只能开放一个端口。MinIO 是一个高性能的对象存储服务,支持 Docker 部署,但默认情况下它需要两个端口:一个是 API 端口(用于存储和访问数据),另一个是控制台端口(用于管理界面…...
Easy Excel
Easy Excel 一、依赖引入二、基本使用1. 定义实体类(导入/导出共用)2. 写 Excel3. 读 Excel 三、常用注解说明(完整列表)四、进阶:自定义转换器(Converter) 其它自定义转换器没生效 Easy Excel在…...
【Java】Ajax 技术详解
文章目录 1. Filter 过滤器1.1 Filter 概述1.2 Filter 快速入门开发步骤:1.3 Filter 执行流程1.4 Filter 拦截路径配置1.5 过滤器链2. Listener 监听器2.1 Listener 概述2.2 ServletContextListener3. Ajax 技术3.1 Ajax 概述3.2 Ajax 快速入门服务端实现:客户端实现:4. Axi…...