Java多线程——基本概念「建议收藏」

Java多线程——基本概念「建议收藏」线程和多线程程序:是一段静态的代码,是应用软件执行的蓝本进程:是程序的一次动态执行过程,它对应了从代码加载、执行至执行完毕的一个完整过程,这个过程也是进程本身从产生、发展至消亡的过程线程:是比进程更小的执行单位。进程在其执行过程中,可以产生多个线程,形成多条执行线索,每条线索,即每个线程也有它自身的产生、存在和消亡的过程,也是一个动态的概念主线程:(每个Java程序都有一个…

大家好,又见面了,我是你们的朋友全栈君。

线程和多线程

程序:是一段静态的代码,是应用软件执行的蓝本

进程:是程序的一次动态执行过程,它对应了从代码加载、执行至执行完毕的一个完整过程,这个过程也是进程本身从产生、发展至消亡的过程

Java多线程——基本概念「建议收藏」

线程:是比进程更小的执行单位。进程在其执行过程中,可以产生多个线程,形成多条执行线索,每条线索,即每个线程也有它自身的产生、存在和消亡的过程,也是一个动态的概念

Java多线程——基本概念「建议收藏」

主线程:(每个Java程序都有一个默认的主线程)

当JVM加载代码发现main方法之后,就会立即启动一个线程,这个线程称为主线程

注意:主线程不一定是最后完成执行的线程,各个线程运行时完全独立,争夺cpu,很可能主线程执行完了,子进程没有。

单线程:如果main方法中没有创建其他的线程,那么当main方法执行完最后一个语句,JVM就会结束Java应用程序

多线程:如果main方法中又创建了其他线程,那么JVM就要在主线程和其他线程之间轮流切换,JVM要等程序中所有线程都结束之后才结束程序。

多线程的优势:

减轻编写交互频繁、涉及面多的程序的困难

程序的吞吐量会得到改善

由多个处理器的系统,可以并发运行不同的线程

“同时”执行是人的感觉,在线程之间实际上轮换执行

线程生命周期(五个状态新建、就绪、运行、阻塞、死亡

新建状态:线程对象已经创建,还没有在其上调用start()方法

就绪状态:当线程调用start方法,但调度程序还没有把它选定为运行线程时线程

运行状态:线程调度程序从可运行池中选择一个线程作为当前线程时线程所处的状态。(是线程进入运行状态的唯一方式)

阻塞(等待/睡眠)状态:线程仍旧是活的,但是当前没有条件运行。当某件事件出现,他可能返回到可运行状态

死亡状态:当线程的run()方法完成时就认为它死去。线程一旦死亡,就不能复生。 一个死去的线程上调用start()方法,会抛出java.lang.IllegalThreadStateException异常

Java多线程——基本概念「建议收藏」

Java中两种创建线程的方式:

1.继承Thread

Java多线程——基本概念「建议收藏」

重写run() 方法

new一个线程对象

调用对象的 start() 启动线程

优点:编写简单,如果需要访问当前线程直接使用this即可获得当前线程.

缺点:因为线程类已经继承了Thread类,不能再继承其他的父类

 

2.实现Runnable接口

Java多线程——基本概念「建议收藏」

实现run() 方法

创建一个Runnable类的对象r,new MyRunnable()

创建Thread类对象并将Runnable对象作为参数,new Thread(r)

调用Thread对象的start()启动线程

优点:线程类只实现了Runable接口,还可以继承其他的类.

缺点:编程稍微复杂,需要访问当前线程,必须使用Thread.currentThread()方法

 

线程创建的问题:

线程的名字:JVM给的名字或者我们自定义的名字,通过setName方法设置

获取当前线程对象:Thread.currentThread()

在一个程序里多个线程只能保证其开始时间,而无法保证其结束时间,执行顺序也无法确定

一个线程的run方法执行结束后,该线程结束

一个线程只能被启动一次,一次只能运行一个线程

JVM线程调度程序决定实际运行哪个处于可运行状态的线程。采用队列形式

线程中的常用方法:

start():启动线程,让线程从新建状态进入就绪队列排队

run():线程对象被调度之后所执行的操作

sleep():暂停线程的执行,让当前线程休眠若干毫秒

currentThread():返回对当前正在执行的线程对象的引用

Java多线程——基本概念「建议收藏」

isAlive():测试线程的状态,新建、死亡状态的线程返回false

interrupt():“吵醒”休眠的线程,唤醒“自己”

yield():暂停正在执行的线程,让同等优先级的线程运行

join():当前线程等待调用该方法的线程结束后,再排队等待CPU资源

stop():终止线程

阻止线程执行的方法:

Java多线程——基本概念「建议收藏」Java多线程——基本概念「建议收藏」

线程睡眠:(当线程睡眠时,它暂停执行,当睡眠时间到期,则返回到可运行状态)

Thread.sleep()

使用场景:线程执行太快

需要强制设定为下一轮执行

线程睡眠是帮助其他线程获得运行机会的最好方法

线程睡眠到期自动苏醒,并返回到可运行状态(不是运行状态)

sleep()中指定的时间是线程不会运行的最短时间(sleep()方法不能保证该线程睡眠到期后就开始执行)

sleep()是静态方法,只能控制当前正在运行的线程

 

线程的优先级

设置线程优先级:

1.线程默认优先级是创建它的执行线程的优先级

2.通过Thread实例调用setPriority()方法设置线程优先级

Thread.MIN_PRIORITY       (1)

Thread.NORM_PRIORITY     (5)

Thread.MAX_PRIORITY      (10)

通过Thread示例调用getPriority()方法得到线程优先级

 

线程让步(yield方法 暂停当前正在执行的线程对象,并执行同等优先级的其他线程)

Thread.yieId();

yield()将导致线程从运行状态转到可运行状态,有可能没有效果无法保证yield()达到让步目的,因为让步的线程还有可能被线程调度程序再次选中

线程离开运行状态的方法:

1.调用Thread.sleep():使当前线程睡眠至少多少毫秒(尽管它可能在指定的时间之前被中断)

2.调用Thread.yield():不能保障太多事情,尽管通常它会让当前运行线程回到可运行性状态,使得有相同优先级的线程有机会执行

3.调用join()方法:保证当前线程停止执行,直到调用join方法的线程完成为止。然而,如果调用join的线程没有存活,则当前线程不需要停止

4.线程的run()方法完成

 

多线程问题——资源协调

两个线程A和B在同时使用Stack的同一个实例对象,A正在往堆栈里push一个数据,B则要从堆栈中pop一个数据

Java多线程——基本概念「建议收藏」

这时,假设idx=5

如果push执行了第一行,没执行第二行,也就是说idx并没有+1,这时开始执行pop,弹出的元素就不对了,并且落下了一个元素。

synchronized是Java中的关键字,是一种同步锁。它修饰的对象有以下几种:

1. 修饰一个代码块,被修饰的代码块称为同步语句块,其作用的范围是大括号{}括起来的代码,作用的对象是调用这个代码块的对象;

2. 修饰一个方法,被修饰的方法称为同步方法,其作用的范围是整个方法,作用的对象是调用这个方法的对象;

3. 修饰一个静态的方法,其作用的范围是整个静态方法,作用的对象是这个类的所有对象;

4. 修饰一个类,其作用的范围是synchronized后面括号括起来的部分,作用主的对象是这个类的所有对象。

Synchronized的作用主要有三个:(1)确保线程互斥的访问同步代码(2)保证共享变量的修改能够及时可见(3)有效解决重排序问题。

Java多线程——基本概念「建议收藏」

资源同步——对象互斥锁

关键字synchronized 与对象互斥锁联合起来使用保证对象在任意时刻只能由一个线程访问

Java多线程——基本概念「建议收藏」

synchronized可以修饰方法,表示这个方法在任意时刻只能由一个线程访问

Java多线程——基本概念「建议收藏」

synchronized可以修饰类,则表明该类的所有对象共用一把锁

Java多线程——基本概念「建议收藏」

多线程同步模型(生产者——消费者示例)

Java多线程——基本概念「建议收藏」Java多线程——基本概念「建议收藏」Java多线程——基本概念「建议收藏」

多线程问题——死锁

(两个或两个以上的线程在执行过程中,因争夺资源而造成了互相等待)

产生死锁的必要条件

互斥条件:指线程对所分配到的资源进行排它性使用

请求和保持条件:指线程已经保持至少一个资源,但又提出了新的资源请求

不可剥夺条件:进程已获得的资源,在未使用完之前,不能被剥夺,只能在使用完时由自己释放

环路等待条件:指在发生死锁时,必然存在一个线程—资源的环形链

出现死锁的情况

相互排斥:一个线程永远占用某一共享资源

循环等待:线程A在等待线程B,线程B在等待线程C,线程C在等待线程A

部分分配:线程A得到了资源1,线程B得到了资源2,两个线程都不能得到全部的资源

缺少优先权:一个线程访问了某资源,但一直不释放该资源,即使该线程处于阻塞状态

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/145920.html原文链接:https://javaforall.net

(0)
上一篇 2022年5月13日 下午6:00
下一篇 2022年5月13日 下午6:20


相关推荐

  • nginx配置ssl证书实现https访问_更换ssl证书

    nginx配置ssl证书实现https访问_更换ssl证书1,登录阿里云,工作台找SSL证书或者安全下找CA证书2,点击创建证书(或购买证书),创建好以后点击证书申请、3,设置配置以及域名信息,仅填写圈住内容,其他默认即可4,随后等待一会,查看状态,是否为 已签发5,为已签发时,点击下载选择下载类型6,下载后解压文件7,上传至服务器,存放位置,先找到nginx所在位置 “/nginx/conf/”找到该位置创建“cert”把刚才解压的两个文件存放至此。8,开始nginx配置内容`server { #SSL 访问端口号为 443 li

    2022年8月19日
    14
  • 查看局域网内的所有IP地址

    查看局域网内的所有IP地址打开电脑的 cmd 命令行界面 依次点击 开始 运行 输入 cmd 在命令行窗口输入 for L iIN 1 1 254 DOping w1 n1192 168 0 i 执行完毕之后 输入 arp a 就可以看到局域网内在线的 IP 了

    2026年3月17日
    1
  • 在黑群辉nas中搭建Emby Server的步骤

    在黑群辉nas中搭建Emby Server的步骤参考 ytb 的视频 https www youtube com watch v G7zRogHZSX81 下载 emby 在群辉中的各种版本 在黑裙 6 i5cpu7200U 的配置中使用的版本是 EmbyServer 3 5 3 0 18 broadwell spk 具体版本已存我的百度网盘 2 进入群辉 点击套件 手动添加 上传本地的 spk 文件 3 上传完成 打开 emby 完成基本的媒体库服务器配置

    2026年3月17日
    2
  • 开机出现DISK BOOT FAILURE,INSERT SYSTEM DISK AND PRESS ENTER「建议收藏」

    开机出现DISK BOOT FAILURE,INSERT SYSTEM DISK AND PRESS ENTER「建议收藏」开机就出现DISKBOOTFAILURE,INSERTSYSTEMDISKANDPRESSENTER我的第一引导是从光驱,第二是从硬盘。以前是可以正常从硬盘启动的,突然发现这种现象。光驱里面没有光盘,为什么不能从硬盘启动了呢?开机滴的一声,应该是自检正常啊。打开BIOS查看了一下,好像也没动什么数据,打开机箱,把几个插头插紧了一下,(不记得做了哪些操作,反正没动内存

    2022年7月13日
    43
  • WebJars介绍

    WebJars介绍参考文章 WebJars web 端静态资源的 jar 包 1 WebJars 介绍 Web 前端使用了越来越多的 JS 或 CSS 如 jQuery Backbone js 和 Bootstrap 一般情况下 我们是将这些 Web 资源拷贝到 JavaWeb 项目的 webapp 相应目录下进行管理 这种通过人工方式管理可能会产生版本误差 拷贝版本错误 漏拷等现象 导致前端页面无法正确展示 版本不一致 文件混乱等 导致出现一些莫

    2026年3月19日
    1
  • box-sizing:border-box的理解和作用

    box-sizing:border-box的理解和作用要想清楚这个属性的作用,首先要理解盒子模型盒子模型是指:外边距(margin)+border(边框)+内边距(padding)+content(内容)可以把每一个容器,比如div,都看做是一个盒子模型比如你给一个div设置宽高为500px,但实际你设置的只是content,之后你又设置了padding:10px;border:1pxsolidred;这时div的宽高就会变为544px(content500px+padding40px+border4px)相当于一个元素的实际宽高是由

    2025年7月17日
    7

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号