多线程(一):线程的基本特点线程安全问题ThreadRunnable

news2025/1/13 9:27:36

目录

1、线程的引入

2、什么是线程

3、线程的基本特点

4、线程安全问题 

5、创建线程

5.1 继承Thread类,重写run

5.1.1 创建Thread类对象

5.1.2 重写run方法

5.1.3 start方法创建线程

5.1.4 抢占式执行

5.2 实现Runnable,重写run【解耦合】★★★

6、知识拓展

6.1 拓展一:名词解释——api

6.2 拓展二:异常处理方式

6.3 拓展三:名词解释——客户端&服务器

 6.4 拓展四:高内聚,低耦合

6.4.1 耦合

6.4.2 内聚


1、线程的引入

大家都知道,当代CPU为多任务处理器,具备多个核心,而为了充分发挥CPU多核心的性能,避免出现“一核有难,多核围观”的情况,“并发编程”就成为刚需。

而通过多进程的方式,可以实现“并发编程”的效果。

虽然说多进程的方式可以实现“并发编程”,但是要知道,进程整体是一个比较“重量级”的概念,如果频繁的创建与销毁,开销是很大的。

尤其是对于服务器来说,一个服务器会为多个客户端提供服务,服务的客户多了,进程的创建与销毁操作自然也会增多。

举个例子:此时,我们打开了百度的网页,但是,这时全国甚至全球会有大量的用户与我们进行相同的操作,会有大量用户对服务器发送大量的请求,大量的进行创建与销毁操作,如果采用多进程方式的话,开销是很大的。

为了解决上述问题,引入一个轻量级的概念——线程(thread)。

2、什么是线程

线程(thread)又称为轻量级进程。

也就是说,线程是一个轻量级的东西,它的创建与销毁的开销要比进程小得多。

因此,可以通过多线程的方式,来实现“并发编程”。

3、线程的基本特点

上篇博客说到,一个进程,相当于一个要执行的任务。

而,一个线程,也相当于一个要执行的任务。

线程与进程的区别如下:

  • 进程包含线程:每个进程中,都会有一个或者多个线程。且至少有一个线程,这个线程在进程创建时随进程一起创建,称为主线程。
  • 进程是操作系统资源分配的基本单位,每个进程都会分配一定的CPU资源、内存资源、硬盘(文件描述符表)资源、网络带宽资源.....。也就是说,在进程创建时,需要申请资源;在进程销毁时,需要释放资源。(会增大系统开销)
  • 而对于线程来说,在进程内部管辖的多个线程之间,会共享进程分配到的资源。对于线程,只是在第一个线程创建时(随进程创建时创建的主线程)需要申请资源,后续再创建的线程,不需要进行资源申请操作。且只有所有的线程都销毁(进程销毁)时,才会释放资源,运行过程中销毁某个进程,也不会释放资源。(系统开销低)
  • 进程和进程间,每个进程分配到的资源都是各自独立的,彼此之间互不干扰,具有稳定性。
  • 进程内部的线程间,会出现相互影响的情况,具有“线程安全问题”
  • 上文所讲的“进程调度”,准确的来说,其实是“线程调度”(当一个进程中只有一个线程时,可以称为“进程调度”)。也就是说,线程是CPU上调度执行的基本单位。如果一个进程中有多个线程,那么这些线程是各自去CPU上调度执行的(可能多个线程由1个核心执行(并发),也可能多个线程由多个核心同时执行(并行),也可能在不同的CPU上来回切换),具体线程是怎么调度执行的,由操作系统内部“调度器”自行完成,程序猿感知不到也干预不了。
  • 每个线程,都会有属于自己单独的调度相关的信息:线程状态、线程上下文、线程优先级、线程记账信息。(也就是说,如果一个进程中有10个线程,就会有10份这样的信息)。但是,一个进程中的线程,共用一个文件描述符表和内存指针。

4、线程安全问题 

对于线程安全问题,先举个例子:

一个房间的桌子上放着100只烧鸡,把小明同学叫来,让他把这100只烧鸡全部吃完(小明相当于一个线程),但是一个人吃100只鸡,显然效率很低。于是,再把小刚同学叫来(再创建一个线程),让小刚和小明共同把这100只鸡吃完。此时,两个人吃100只鸡,显然比一个人吃100只鸡的效率要高的多。如果再叫来两三个其他同学,这时的效率就会更高。但是如果一直再叫来其他同学,比如叫到了50名同学,50名同学共同吃这100只鸡,其中两人都想吃同一只鸡,这两人间就会发生冲突(即线程安全问题),甚至冲突过大会把桌子掀翻,这时所有的人都吃不了鸡了(直接带走进程,所有线程无法继续工作)。

综上,总结如下:

  • 虽然多线程的方式能够提高工作效率,但是也并非“线性增长”,当一个进程中的线程过多时,线程与线程间就会出现互相影响的情况,会拖慢效率,甚至会抛出异常使整个进程终止(如果及时捕捉到异常,也是不会终止的)。
  • 线程数目如果太多,线程的调度开销也会非常明显,会因为调度开销拖慢程序性能。

5、创建线程

线程,是操作系统提供的概念,同时操作系统也提供了一些线程相关的api供程序员使用。

操作系统提供的原生api是C语言写的,并且不同操作系统所提供的线程api是不同的,是不是我们Java程序猿就得去学习C语言呢?

并不是的,Java对操作系统提供的线程api统一进行了封装,在标准库中提供了Thread类,我们可以通过Thread类来创建和使用多线程。

而创建线程的方式有两种:

  1. 继承Thread,重写run
  2. 实现Runnable,重写run

5.1 继承Thread类,重写run

5.1.1 创建Thread类对象

Thread类被封装在了java.lang包中,java.lang是Java的核心包,包含了String、Math、System、Thread、Runnable等等,这个包中的类被自动导入到每个Java程序中,无需显式导入。所以当我们使用Thread类时,不会自动显示导入包。

5.1.2 重写run方法

作为程序员,我们需要创建一个类继承于Thread并且重写其中的run方法,在run这个方法中,我们可以根据自己的思维将这个线程要做的任务写在这个run方法中。

这个run方法,就相当于线程的入口。

为后续观察多线程的状态,这里使用死循环的方式打印“hello thread”,再使用Thread中静态的sleep方法休眠1秒(防止CPU红温)。

使用sleep方法会抛出受查异常,解决方法有两个:

  1. throws:进行异常声明
  2. try-catch:进行异常捕获

但是由于run为重写方法,不能使用throws在函数头声明,只能使用try-catch捕获。

5.1.3 start方法创建线程

start方法的作用是真正创建一个新的进程,相当于多了一个执行流,多了一个干活的人,让代码能够“一心两用”,同时做两件事。

在main方法(主线程)中使用Thread对象调用start方法创建线程,并且在main方法中循环打印“hello main”,观察多线程现象。

注意,start的作用才是创建线程,run方法只是线程的入口,不是创建线程。

如果按照我们之前学习的程序运行逻辑,程序遇到死循环就会一直停留在那里(单线程模式),但是我们现在创建了多个线程,会发生什么样的情况呢?

多线程运行:

运行后,“hello main”和“hello thread”无规律交替打印,这就是多线程。

class MyThread extends Thread{
    @Override
    public void run() {
        //线程入口
        while (true) {
            System.out.println("hello thread");
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
        }
    }
}
public class Demo1 {
    public static void main(String[] args) throws InterruptedException {
        Thread thread = new MyThread();
        //start -> 真正的创建线程
        thread.start();
        while (true) {
            System.out.println("hello main");
            Thread.sleep(1000);
        }
    }
}

5.1.4 抢占式执行

观察到,“hello main”和“hello thread”的打印是随机的,也就是,这两个线程的调度是随机的,谁先执行,谁后执行,都是无法预测的,我们称这种情况为“抢占式执行”,通俗来说,就是谁先抢到谁就先执行。

我们唯一能做的就是给线程设置优先级,但是对于操作系统来说,也只是仅供参考,不会一定的按照优先级的顺序来调度执行。

5.2 实现Runnable,重写run【解耦合】★★★

我们可以把要重写的run方法抽象出来,使用自定义类实现Runnable接口,在类中重写run方法(即要完成的任务),将要完成的任务和线程分离开来,实现与线程Thread的解耦合。

就是仅仅把runnable当做一个任务,单纯的把任务抽象到runnable接口的run方法中,最后线程还是要靠Thread来创建。

这样解耦合的有以下优点:

  1. 将所要完成的任务和线程分类开来,而不是把任务直接写到线程当中
  2. 以后可以通过其他方式执行该任务(不一定是在线程中),使线程是线程,任务是任务。
  3. 方便以后修改任务时不会影响到线程,方便代码的维护(容易改,不会一改一大片)
class MyRunnable implements Runnable {

    @Override
    public void run() {//run --> 相当于线程的入口
        while (true) {
            System.out.println("hello thread");
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            }
        }
    }
}
public class Demo2 {
    public static void main(String[] args) throws InterruptedException {
        //任务
        Runnable runnable = new MyRunnable();
        //线程
        Thread thread = new Thread(runnable);
        //start --> 真正的创建线程
        thread.start();
        while (true) {
            System.out.println("hello main");
            Thread.sleep(1000);
        }
    }
}

6、知识拓展

6.1 拓展一:名词解释——api

上文很多地方都提到了api,但是大家都知道啥是api嘛?

api(application programming interface),应用程序编程接口。

  • 通俗来说,api就是别人写的一些函数/类,你直接拿过来就能用。

api是一个广义的概念,操作系统会提供api、标准库会提供api、第三方库会提供api、其他各种开源项目会提供api、甚至工作中项目组给你的代码中也会提供api。

  • api也可以理解为,别人给你提供的库/程序,你都能用来干啥。

举个例子:对于同班同学,你可以给他在微信上发消息、可以问他题、可以和他聊天、....,这是你同学向你提供的api;你谈了个对象,你可以和你的对象亲亲抱抱举高高....,这是你对象向你提供的api。

  • 而基于api,你可以用来编程(api的目的就是用于编程)。

比如接上例,基于你同学或者对象向你提供的api,你可以做出规划(编程):周末约同学打球;周末约对象看电影......

而对于Java程序猿的我们,我们可以使用标准库向我们提供的api去编程,比如ArrayList、StringBuffer、.......

在计算机界,Demo/Sample/quick start 的意思是示例、演示的意思,告诉我们如何使用。

test是更为详细的测试过程。

6.2 拓展二:异常处理方式

在上文中,我们提到对于受查异常有两种处理方式:

  1. throws
  2. try-catch

当我们使用IDEA进行自动的异常处理时,它是这样处理的:

它在catch中又重新拋了一个新的异常,只不过拋了个非受查的异常,所以没有再编译报错了,这种方法仅仅是满足了语法的要求,但是对于异常来说,就相当于没处理异常。在实际开发中,我们并不会这么干~

在实际工作中,通常会这样处理异常:

  1. 记录异常信息作为日志,后续根据日志调查问题。——使程序仍然正常执行,不会因为这个异常就终止。(不交给jvm处理)(服务器是7*24小时运行的,如果服务器因为异常导致崩溃,就无法给客户提供服务,这对于服务器来说非常关键)
  2. 进行重试。(有的异常是概率性发生的,如:网络抖动原因)
  3. 报警机制——如果是特别严重的问题,程序会立即通知程序猿处理(通过写代码来以短信、电话、微信等方式通知程序猿)。

6.3 拓展三:名词解释——客户端&服务器

客户端(client),服务器(server)指的两个程序(两个软件),这两个程序,通过配合完成一些工作。

客户端向服务器发送的数据,称为“请求”(request)。

服务器向客户端返回的数据,称为“响应”(response)。

客服端和服务器的主要区别如下:

  1. 主动发起请求的一方叫做客户端。被动接受请求,返回响应的一方叫做服务器。
  2. 通常一个服务器,给多个客户端提供服务。
  3. 服务器,不知道客户端来不来,啥时候来,所以只能将程序一直持续的运行下去,即7*24小时的跑(007)。(正因此,异常导致服务器崩溃的后果是非常严重的,必须将异常处理好)

 6.4 拓展四:高内聚,低耦合

6.4.1 耦合

耦合,指两个东西的关联程度。关联度越高,耦合就越大;关联度越低,耦合就越小。

在代码中,我们希望代码间是低耦合的(解耦),因为在开发中,代码是经常会修改的,低耦合的代码可维护性高(也就是好改),要修改代码的话,改一小部分就行,能够防止“改一个,改坏一片”的情况发生。

举个例子:

你结婚后,你媳妇生病住院了,你只能立刻放下手中的活,到医院来,照顾她、陪伴她,什么工作也干不了。因为你媳妇对你来说是很主要的人,你媳妇的生病对你的工作/生活影响很大,你必须放下你手头的事,哪怕再紧急的工作也得放下。

这就说明,你和你媳妇是高耦合,你媳妇出现了状况,对你的影响很大,你啥事也干不了。

而,如果你高中时的白月光发了个朋友圈说她生病住院了,对你来说呢,你只是点了个赞,评论了句“早日康复”,接着放下手机回头就忘了这件事,对你一点影响也没有。

这就说明,你和你高中的白月光是低耦合,她出现了啥状况,对你一定影响也没有。

6.4.2 内聚

内聚是指有相同的功能、逻辑关系的东西的集中程度。

代码中,我们希望高内聚,将相同逻辑、功能的或者有关联的代码放到一起,

而不是这放一块,那放一块的(低内聚)。

举个例子:

你结婚有了孩子后,你媳妇这个人她比较懒,总是把衣服这扔一件那扔一件的,有的衣服在沙发,有的衣服在床上,有的衣服在椅子上,有的还在沙发缝里。有一天,你媳妇让你给孩子拿一件衣服,由于衣服哪都有,你非常的痛苦,遍历了整个屋子都没找到孩子的衣服在哪。

这就反应的是低内聚。

后来,你媳妇变得贤惠了,把衣服都知道收拾整理好放到衣柜了,你再给孩子找衣服的时候,直接去衣柜里拿就行了。

这反应的就是高内聚。

综上:高内聚(一个模块内,有关联的东西放在一块),低耦合(模块之间,依赖尽量小,影响尽量小)。


END

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

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

相关文章

MySQL-数据库设计

1.范式 数据库的范式是⼀组规则。在设计关系数据库时,遵从不同的规范要求,设计出合理的关系型数 据库,这些不同的规范要求被称为不同的范式。 关系数据库有六种范式:第⼀范式(1NF)、第⼆范式(…

【Mysql】SQL语言基础

1、SQL的概述 SQL全称:Structured Query Language,是结构化查询语言,用于访问和处理数据库的标准的计算机语言。SQL语言1974年由Boyce和Chamberlin提出,并首先在IBM公司研制的关系数据库系统systemr上实现。 美国国家标准局&#x…

亚信安全发布第34期《勒索家族和勒索事件监控报告》

本周态势快速感知 本周全球共监测到勒索事件91起,近三周勒索事件数量较为稳定。从整体上看,Ransomhub是影响最严重的勒索家族;Play和ElDorado恶意家族也是两个活动频繁的恶意家族,需要注意防范。本周,土耳其公司巴克皮…

小红书2024秋招后端开发(Java工程师、C++工程师等)

前几天做了美团,OPPO的秋招笔试题,然后又做了一场小红书,总体难度我觉得都差不多,涉及到的知识点要么是语法模拟,或者就是一些基础算法,所以这样看秋招编程题还是很简单的,对于笔试我们还要把除…

深刻理解Redis集群(下):Redis 哨兵(Sentinel)模式

背景 现在对3个节点的sentinel进行配置。sentinel的配置文件在redis的安装目录中已经存在,只需要复制到指定的位置即可。 sentinel是独立进程,有对应的脚本来执行。 基于之前的redis 一主二从的架构,我们继续启动3个sentinel进程。 哨兵模式的…

使用微服务Spring Cloud集成Kafka实现异步通信

在微服务架构中,使用Spring Cloud集成Apache Kafka来实现异步通信是一种常见且高效的做法。Kafka作为一个分布式流处理平台,能够处理高吞吐量的数据,非常适合用于微服务之间的消息传递。 微服务之间的通信方式包括同步通信和异步通信。 1&a…

GPU参数指标

以英伟达的A800卡为例,简单聊聊GPU卡的核心参数指标,A800的核心指标主要有5个,为算力、显存大小、显存带宽、功耗情况和卡间互联速率。 性能:则可以理解为货车对不同货物类型的马力大小,决定能“拉动”多少重量的货&…

实用工具推荐---- PDF 转换

直接上链接:爱PDF |面向 PDF 爱好者的在线 PDF 工具 (ilovepdf.com) 主要功能如下: 全免费!!!!

什么是 Apache Ingress

Apache Ingress 主要用于管理来自外部的 HTTP 和 HTTPS 流量,并将其路由到合适的 Kubernetes 服务。 容器化与 Kubernetes 是现代云原生应用程序的基础。Kubernetes 的主要职责是管理容器集群,确保它们的高可用性和可扩展性,同时还提供自动化…

httpsok-v1.17.0-SSL通配符证书自动续签

🔥httpsok-v1.17.0-SSL通配符证书自动续签 介绍 httpsok 是一个便捷的 HTTPS 证书自动续签工具,基于全新的设计理念,专为 Nginx 、OpenResty 服务器设计。已服务众多中小企业,稳定、安全、可靠。 一行命令,一分钟轻…

Java中使用接口实现回调函数的详解与示例

💝💝💝欢迎莅临我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。 推荐:「storm…

【2025】springboot基于微信小程序记账本的设计与实现(源码+文档+调试+答疑)

文章目录 前言一、主要技术?二、项目内容1.整体介绍(示范)2.运行截图3.系统测试 总结更多项目 前言 时代在飞速进步,每个行业都在努力发展现在先进技术,通过这些先进的技术来提高自己的水平和优势,记账本小…

【游戏分组】

题目来源 from itertools import combinations def get_input(): """获取输入的整数列表。""" return list(map(int, input("请输入10个整数(用空格分隔): ").split())) def get_min_difference(arr): &q…

OpenCV C++霍夫圆查找

OpenCV 中的霍夫圆检测基于 霍夫变换 (Hough Transform),它是一种从边缘图像中识别几何形状的算法。霍夫圆检测是专门用于检测图像中的圆形形状的。它通过将图像中的每个像素映射到可能的圆参数空间,来确定哪些像素符合圆形状。 1. 霍夫变换的原理 霍夫…

【韩顺平Java笔记】第3章:变量

只记录我觉得重点的,自用,如果有漏的请自己看视频 文章目录 33. 内容梳理34. 变量原理34.1 为什么需要变量35. 变量概念35.1 概念35.2 变量使用的基本步骤36. 变量入门36.1 变量使用入门案例 37. 变量细节37.1 变量使用注意事项 38. 加号使用38.1 程序中…

身份证号、定位信息等个人信息敏感性判定解析

关于身份证号号码以及精确定位信息是否是敏感个人信息的疑问一直以来不少合规安全从业者有疑惑,本文来自于《数安标准强基助力计划 》作者为指南和标准的起草者,其观点具有一定的权威性,一下为内容摘要,以供大家学习和解惑&#x…

【sourceTree问题】拉取提交的时候需要频繁输入账号密码

用sourceTree进行代码管理的时候会出现一直让输入账号密码的问题,烦不胜烦,可以点击【设置】 → 【编辑配置文件...】打开配置文件: 在配置文件里找到url,把url里面的网址修改为: http://username:passwordxxxxx/xx…

LeetCode 热题 100 回顾7

干货分享,感谢您的阅读!原文见:LeetCode 热题 100 回顾_力code热题100-CSDN博客 一、哈希部分 1.两数之和 (简单) 题目描述 给定一个整数数组 nums 和一个整数目标值 target,请你在该数组中找出 和为目标…

阿里云 SAE Web:百毫秒高弹性的实时事件中心的架构和挑战

作者:胡志广(独鳌) 背景 Serverless 应用引擎 SAE 事件中心主要面向早期的 SAE 控制台只有针对于应用维度的事件,这个事件是 K8s 原生的事件,其实绝大多数的用户并不会关心,同时也可能看不懂。而事件中心,是希望能够…

实验3 使用Wiresharkl观察ping命令的工作过程

1、实验目的: 了解嗅探器工具Ethereal(Wireshark)的下载和安装方法; 掌握Ethereal(Wireshark)的简单使用方法; 了解抓包结果的分析方法(最好是把菜单中所有的菜单命令都尝试一下&…