JAVA线程及简单同步实现的原理解析

本文涉及的产品
全局流量管理 GTM,标准版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
云解析 DNS,旗舰版 1个月
简介: JAVA线程及简单同步实现的原理解析线程一、内容简介:  本文主要讲述计算机中有关线程的相关内容,以及JAVA中关于线程的基础知识点,为以后的深入学习做铺垫。如果你已经是高手了,那么这篇文章并不适合你。

JAVA线程及简单同步实现的原理解析
线程
一、内容简介:
  本文主要讲述计算机中有关线程的相关内容,以及JAVA中关于线程的基础知识点,为以后的深入学习做铺垫。如果你已经是高手了,那么这篇文章并不适合你。

二、随笔正文:
1、计算机系统组成
  计算机系统由计算机硬件系统和软件系统组成。我们今天要说的线程和硬件系统中的cpu中央处理器,及软件系统中的操作系统,进程等有比较紧密的联系。操作系统是软件中比较特殊的存在,与硬件系统直接交互,其他程序(软件)运行在操作系统之上。

2、cpu简单说明
  硬件系统中特别重要的一项就是处理器CPU,与我们所说的线程有非常紧密的联系。cpu中有几项参数,以及如何查看该信息,在下文逐一说明:

块数:民用pc机,基本都是一块物理cpu,每块主板上只能装一块cpu。

核心数:也就是单块物理cpu是由几组处理芯片组,组成的。4核心 8核心等。

线程数:老款cpu都是单线程的,及一组芯片组只能运行一个线程。现款因特尔cpu大多支持超线程技术可支持多个逻辑线程。但是需要操作系统及相关编程语言的支持,JAVA相较C++在多线程方面能表现的更出色。

主频:单位GHZ(hz赫兹 每秒的周期性变动重复次数)在计算机中即高低电平变化一次,可以产生两个不同的电信号0、1。以我的CPU I5-4200M 2.5GHZ 举例,及cpu可以每秒完成25亿次震荡! 也就是说主频越高理论上计算能力越强,处理计算机指令越快。但是并不代表计算机整体运算速度约高,这点通常满足水桶效应,而cpu一直稳居长板地位。

缓存:cpu内置缓存,很小通常为几Mb至十几Mb,和cpu交互更频繁,速度也远高于普通运行内存,提高cpu处理能力的有效手段。

查看cpu参数指令:

DOS命令
3、关系梳理
  操作系统,程序,进程,线程之间的关系梳理、

  程序:是计算机上的静态代码,指令文件集合,是静态的存在。比如:QQ,LOL等

  进程:程序的执行实体(过程),持有及分配资源的主体。chrome.ext,QQ.exe等执行进程。

  线程:是进程中的劳动力,由进程创建,完成指定任务后结束。

  关系:

    操作系统 1 ——> n 程序 1 ——> n 进程 1 ——> n 线程

     平台       集合     资源     干活的

  普通进程创建线程去完成指定的计算机指令,这个时候需要调用系统资源如cpu进行运算,但是用户线程并不能直接驱动硬件,而是通过操作系统去统一分配、控制硬件的使用。

4、进程、线程基本状态

  五个基本状态,创建和终止不说了。计算机中的线程创建之后会进入就绪状态,当cpu为此线程分配时间片时,线程由就绪转为执行状态,开始干活,当时间片结束时回到就绪状态等待下次获取时间片,循环直到任务完成,当任务完成时,线程终止(死亡)。若在执行过程中遇到耗时操作比如IO或者JAVA中的线程休眠等,会进入阻塞状态,阻塞结束会进入就绪状态继续排队等待被分配时间片。

5、JAVA中的线程
  java中提供了两种方式去创建线程,继承类和实现接口。由于java中单继承机制的限制,大多数情况下使用实现Runnable接口的形式创建线程。

1 、继承Thread类
继承Thread类
2、实现Runnable接口
实现Runnable接口
3、线程的常用方法
复制代码
1 start()
2 //启动线程,调用run方法
3 sleep()
4 //线程休眠,进入阻塞状态,让出时间片,但不会让出锁
5 Thread.currentThread()
6 //获取当前执行线程
7 wait/notify()
8 //仅能存在synchronized代码块中,wait线程休息,让出时间片,进入等待状态,notify()唤醒该线程;该方法存在重载
9 join()
10 //等待此线程执行完毕
11 setDaemon()
12 //设置守护线程,守护线程是服务线程,当用户线程结束,守护线程自动结束
13 yield()
14 //主动让出时间片给其他线程
15 interrupt()
16 //中断线程,不推荐
17 get/setId()
18 //设置获取线程id
19 get/setName()
20 //设置获取线程名
21 get/setPriority()
22 //设置获取线程优先级,理论上优先级越高,获取时间片的概率越大,默认是5最高10最小1
复制代码
4、JAVA中的线程状态图

5、线程练习
简单模拟多线程购票业务

售票业务
  在不做任何控制的情况下,出错的几率很小,我反复测试几次,结果基本都正确!分析原因:业务本身相对简单,没有耗时操作,每个时间片基本能保证线程将本次任务执行完毕!也就是将票数减一并打印内容。

  为了模拟在售票前双方的问询阶段及付款阶段的等待,在售票前(后)加入Thread.sleep(ms) 模拟耗时操作。

加入线程休眠
  改动后系统出现bug同一张票被售出了多次,并且可能将票卖出负数。究其原因:线程之间的数据争用问题,我们将引入JAVA内存模型进行分析(图片来自网络侵删)

  首先我们需要知道几个概念,如下:

  共享变量:主内存中存在被多个线程同时用到的变量,在多个线程中存在相应副本变量。

  可见性:线程对共享变量的值进行修改,能否被其他线程可见。

  变量访问规则:   1、线程对共享变量的操作只能在自己工作内存中的副本。

            2、工作内存中的变量变化需要通过主内存传递。

6、实现同步
  实现同步即实现共享变量可见性,工作内存1 ——> 主内存 ——>工作内存2,在数据传递的两个环节中出现问题都会对同步造成影响,从而影响执行结果。

  synchronized实现同步

  synchronized 可以实现指令的原子性,及共享变量的可见性。

  原子性:synchronize修饰的方法或者代码块会获取互斥锁,保证同一时间只能有一个对象访问该方法或代码块,将其作为一个整体,保证了原子性。

​   可见性:加锁后,先清空工作内存,同步主内存中的共享变量。解锁前,先将工作内存中的变量同步到主内存,再释放锁。

  所以结合上述例子,为sellTickets()方法加锁即可实现同步;或者对核心代码片段加锁;

复制代码
private synchronized boolean sellTickets() { //… 省略中间代码
}
  //或者如下实现
  
synchronized(this){

   //this指的是调用sellTickets()的对象

}
复制代码
7、volatile关键字
  共享变量(类的成员变量、类的静态成员变量)被volatile修饰之后,那么就具备了两层语义:

  原理:被volatile修饰的变量所生成汇编代码时有lock前缀,生成"内存屏障"。

    1)保证了不同线程对这个变量进行操作时的可见性,即工作内存中的变量值在修改后会被立即同步到主内存中;

    2)并且使其他线程中的缓存无效,这样当其他线程在访问共享变量时就必须取主内存中获取;

    3)禁止进行指令重排序;

  综上所述,volatile可以保证可见性,但不能保证原子性;

  举例分析:

复制代码
volatile int number = 0 ;
number ++ ;
/*

操作可以解析成三步:  1.获取number中的值 
                  2.计算加1操作 
                  3.number = 0 + 1; 
在一个时间片中,虽然volatile修饰的number一定会被立即同步到主内存中,但不能保证完整执行这三步,所以不能保证++操作的原子性 。

*/
复制代码
三、总结说明:
  首先感谢各位能看到最后,对本文内容如果存在疑问,欢迎留言交流,若存在错误,也还望斧正!我将不定期对文章进行修改和调整,如发现错误一定及时改正以免误人子弟!
原文地址https://www.cnblogs.com/lijizhi/p/10775748.html

相关文章
|
1天前
|
存储 监控 Java
【Java并发】【线程池】带你从0-1入门线程池
欢迎来到我的技术博客!我是一名热爱编程的开发者,梦想是编写高端CRUD应用。2025年我正在沉淀中,博客更新速度加快,期待与你一起成长。 线程池是一种复用线程资源的机制,通过预先创建一定数量的线程并管理其生命周期,避免频繁创建/销毁线程带来的性能开销。它解决了线程创建成本高、资源耗尽风险、响应速度慢和任务执行缺乏管理等问题。
88 60
【Java并发】【线程池】带你从0-1入门线程池
|
17天前
|
机器学习/深度学习 算法 数据挖掘
解析静态代理IP改善游戏体验的原理
静态代理IP通过提高网络稳定性和降低延迟,优化游戏体验。具体表现在加快游戏网络速度、实时玩家数据分析、优化游戏设计、简化更新流程、维护网络稳定性、提高连接可靠性、支持地区特性及提升访问速度等方面,确保更流畅、高效的游戏体验。
64 22
解析静态代理IP改善游戏体验的原理
|
14天前
|
编解码 缓存 Prometheus
「ximagine」业余爱好者的非专业显示器测试流程规范,同时也是本账号输出内容的数据来源!如何测试显示器?荒岛整理总结出多种测试方法和注意事项,以及粗浅的原理解析!
本期内容为「ximagine」频道《显示器测试流程》的规范及标准,我们主要使用Calman、DisplayCAL、i1Profiler等软件及CA410、Spyder X、i1Pro 2等设备,是我们目前制作内容数据的重要来源,我们深知所做的仍是比较表面的活儿,和工程师、科研人员相比有着不小的差距,测试并不复杂,但是相当繁琐,收集整理测试无不花费大量时间精力,内容不完善或者有错误的地方,希望大佬指出我们好改进!
73 16
「ximagine」业余爱好者的非专业显示器测试流程规范,同时也是本账号输出内容的数据来源!如何测试显示器?荒岛整理总结出多种测试方法和注意事项,以及粗浅的原理解析!
|
5天前
|
Java 数据库 开发者
详细介绍SpringBoot启动流程及配置类解析原理
通过对 Spring Boot 启动流程及配置类解析原理的深入分析,我们可以看到 Spring Boot 在启动时的灵活性和可扩展性。理解这些机制不仅有助于开发者更好地使用 Spring Boot 进行应用开发,还能够在面对问题时,迅速定位和解决问题。希望本文能为您在 Spring Boot 开发过程中提供有效的指导和帮助。
42 12
|
3天前
|
开发框架 监控 JavaScript
解锁鸿蒙装饰器:应用、原理与优势全解析
ArkTS提供了多维度的状态管理机制。在UI开发框架中,与UI相关联的数据可以在组件内使用,也可以在不同组件层级间传递,比如父子组件之间、爷孙组件之间,还可以在应用全局范围内传递或跨设备传递。
17 2
|
2天前
|
安全 算法 网络协议
解析:HTTPS通过SSL/TLS证书加密的原理与逻辑
HTTPS通过SSL/TLS证书加密,结合对称与非对称加密及数字证书验证实现安全通信。首先,服务器发送含公钥的数字证书,客户端验证其合法性后生成随机数并用公钥加密发送给服务器,双方据此生成相同的对称密钥。后续通信使用对称加密确保高效性和安全性。同时,数字证书验证服务器身份,防止中间人攻击;哈希算法和数字签名确保数据完整性,防止篡改。整个流程保障了身份认证、数据加密和完整性保护。
|
13天前
|
Java 程序员 开发者
Java社招面试题:一个线程运行时发生异常会怎样?
大家好,我是小米。今天分享一个经典的 Java 面试题:线程运行时发生异常,程序会怎样处理?此问题考察 Java 线程和异常处理机制的理解。线程发生异常,默认会导致线程终止,但可以通过 try-catch 捕获并处理,避免影响其他线程。未捕获的异常可通过 Thread.UncaughtExceptionHandler 处理。线程池中的异常会被自动处理,不影响任务执行。希望这篇文章能帮助你深入理解 Java 线程异常处理机制,为面试做好准备。如果你觉得有帮助,欢迎收藏、转发!
72 14
|
16天前
|
安全 Java 程序员
Java 面试必问!线程构造方法和静态块的执行线程到底是谁?
大家好,我是小米。今天聊聊Java多线程面试题:线程类的构造方法和静态块是由哪个线程调用的?构造方法由创建线程实例的主线程调用,静态块在类加载时由主线程调用。理解这些细节有助于掌握Java多线程机制。下期再见! 简介: 本文通过一个常见的Java多线程面试题,详细讲解了线程类的构造方法和静态块是由哪个线程调用的。构造方法由创建线程实例的主线程调用,静态块在类加载时由主线程调用。理解这些细节对掌握Java多线程编程至关重要。
48 13
|
7天前
|
Java API 数据处理
深潜数据海洋:Java文件读写全面解析与实战指南
通过本文的详细解析与实战示例,您可以系统地掌握Java中各种文件读写操作,从基本的读写到高效的NIO操作,再到文件复制、移动和删除。希望这些内容能够帮助您在实际项目中处理文件数据,提高开发效率和代码质量。
15 0
|
17天前
|
安全 Java 开发者
【JAVA】封装多线程原理
Java 中的多线程封装旨在简化使用、提高安全性和增强可维护性。通过抽象和隐藏底层细节,提供简洁接口。常见封装方式包括基于 Runnable 和 Callable 接口的任务封装,以及线程池的封装。Runnable 适用于无返回值任务,Callable 支持有返回值任务。线程池(如 ExecutorService)则用于管理和复用线程,减少性能开销。示例代码展示了如何实现这些封装,使多线程编程更加高效和安全。

推荐镜像

更多