【并发】共享模型之管程

2024-01-14 17:44
文章标签 模型 并发 共享 管程

本文主要是介绍【并发】共享模型之管程,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

共享模型之管程

共享问题

package 并发;public class Test1 {static int a=0;public static void main(String[] args) throws InterruptedException {Thread t1=new Thread(new Runnable() {@Overridepublic void run() {for(int i=0;i<5000;i++){a++;}}});Thread t2=new Thread(new Runnable() {@Overridepublic void run() {for(int i=0;i<5000;i++){a--;}}});t1.start();;t2.start();t1.join();;t2.join();System.out.println("a="+a);}}

与预期的结果不同

问题分析

以上的结果可能是正数,负数,0为什么呢? 因为Java中对静态变量的自增,自减并不是原子操作,要彻底理解,必须从字节码进行分析。

例如:对于i++而言,实际会产生如下的JVM字节码指令:

getstatic i //获取静态变量 iconst_1 //准备常量1 iadd //自增 putstatic i //将修改后的值存入静态变量i

而JAVA 的内存模型如下,完成静态变量的自增,自减需要在主存和工作内存中进行数据交换:

如果是单线程,上面的代码是顺序执行(不会交错) 没有问题:

临界区

  • 一个程序运行多个线程本身是没有问题的。
  • 问题出在多个线程访问共享资源。
    • 多个线程读取共享资源其实也没有问题
    • 在多个线程对共享资源读写操作时发生指令交集,就会出现问题。
  • 一般代码块如果存在对共享资源的多线程读写操作。那么这段代码称为临界区。
Thread t1=new Thread(new Runnable() {@Overridepublic void run() {//临界区a++;}
});
Thread t2=new Thread(new Runnable() {@Overridepublic void run() {//临界区a--;}
});

解决方案

  • 阻塞式的解决方案:synchronized \ Lock
  • 非阻塞式的解决方案:原子变量

本次课程使用的解决方案式:synchronzied ,来解决上述问题,俗称【对象锁】。

它采用互斥的方式让同一时刻至多只有一个线程能持有【对象锁】,其他想获取这个对象锁就会被阻塞住,这样就能保证拥有锁的线程可以安全的执行临界区内的代码,不用担心上下文的切换。

注意:

虽然java中的互斥和同步都是可以采用synchronized来完成,但还是有区别的。

  • 互斥是保证临界区的竟态条件发生,同一时刻只有一个线程执行临界区的代码。
  • 同步是由于线程执行的先后,顺序不同,需要一个线程等待其他线程运行到这个点,

synchronzied

语法
synchronized(){  临界区
}
解决
package 并发;import java.util.Date;public class Test1 {static Integer a=0;static Object flag=new Object();public static void main(String[] args) throws InterruptedException {Thread t1=new Thread(new Runnable() {@Overridepublic void run() {for(int i=0;i<5000;i++){//加锁synchronized (flag){a++;}}}});Thread t2=new Thread(new Runnable() {@Overridepublic void run() {for(int i=0;i<5000;i++){//加锁synchronized (flag){a--;}}}});t1.start();;t2.start();t1.join();;t2.join();System.out.println("a="+a);}}
向对象思想改进面
package 并发;import java.util.Date;class Test1 {static Integer a=0;static   Room room=new Room();public static void main(String[] args) throws InterruptedException {Thread t1=new Thread(new Runnable() {@Overridepublic void run() {for(int i=0;i<5000;i++){room.increase();;}}});Thread t2=new Thread(new Runnable() {@Overridepublic void run() {for(int i=0;i<5000;i++){room.decrease();;}}});t1.start();;t2.start();t1.join();;t2.join();System.out.println("结果是"+room.count);}}class Room {public static int count=0;static Object flag=new Object();public void increase(){synchronized (flag){count++;}}public void decrease(){synchronized (flag){count--;}}}

方法上的synchronized

语法

synchronized加在普通方法上

class Room {public static int count=0;static Object flag=new Object();public synchronized void increase(){count++;}//等价于 锁住的是自己的对象public void increase(){synchronized(this){count++;}}}

synchronized加在静态方法上

class Room {public static int count=0;static Object flag=new Object();public synchronized  static void increase(){count++;}//等价于 锁住的是自己的类对象public static void increase(){synchronized(Room.class){count++;}}}

不加synchronized方法无法保证原子性

线程安全分析

成员变量和静态变量是否是安全的?

  • 如果他们没有共享,则线程安全
  • 如果他们被共享了,根据他们的线程是否能改变,又分为两种:

                只有读操作,则线程安全。

                如果有读写操作,则这段代码是临界区,需要考虑线程安全。

局部变量是否是线程安全的?

  • 局部变量是线程安全的
  • 但局部变量引用的对象未必。(堆中的变量就可能被共享)
    • 如果该对象没有逃离方法的作用范围,则是线程安全的。
    • 如果该对象逃离方法的作用范围,则需要考虑线程安全
局部变量线程安全分析

public static void test1(){ int i=10; i++; }

每个线程调用test1()方法时局部变量i,会在每个线程的栈帧内存中被创建多份,因此不存在共享

如图:

局部变量的引用稍有不同

先看一个成员变量的例子

class ThreadUnsafe { ArrayList list = new ArrayList<>(); public void method1(int loopNumber) { for (int i = 0; i < loopNumber; i++) { // { 临界区, 会产生竞态条件 method2(); method3();执行其中一种情况是,如果线程2 还未 add,线程1 remove 就会报错:Exception in thread "Thread1" java.lang.IndexOutOfBoundsException: Index: 0, Size: 0 at java.util.ArrayList.rangeCheck(ArrayList.java:657) at java.util.ArrayList.remove(ArrayList.java:496) at cn.itcast.n6.ThreadUnsafe.method3(TestThreadSafe.java:35) at cn.itcast.n6.ThreadUnsafe.method1(TestThreadSafe.java:26) at cn.itcast.n6.TestThreadSafe.lambda$main$0(TestThreadSafe.java:14) at java.lang.Thread.run(Thread.java:748)分析:无论哪个线程中的 method2 引用的都是同一个对象中的 list 成员变量method3 与 method2 分析相同 // } 临界区 } } private void method2() { list.add("1"); } private void method3() { list.remove(0); }}

执行

static final int THREAD_NUMBER = 2;
static final int LOOP_NUMBER = 200;
public static void main(String[] args) { ThreadUnsafe test = new ThreadUnsafe(); for (int i = 0; i < THREAD_NUMBER; i++) { new Thread(() -> { test.method1(LOOP_NUMBER); }, "Thread" + i).start(); }}

其中一种情况是,如果线程2 还未 add,线程1 remove 就会报错

Exception in thread "Thread1" java.lang.IndexOutOfBoundsException: Index: 0, Size: 0 at java.util.ArrayList.rangeCheck(ArrayList.java:657) at java.util.ArrayList.remove(ArrayList.java:496) at cn.itcast.n6.ThreadUnsafe.method3(TestThreadSafe.java:35) at cn.itcast.n6.ThreadUnsafe.method1(TestThreadSafe.java:26) at cn.itcast.n6.TestThreadSafe.lambda$main$0(TestThreadSafe.java:14) at java.lang.Thread.run(Thread.java:748)

分析:

无论哪个线程中的 method2 引用的都是同一个对象中的 list 成员变量

method3 与 method2 分析相同

这篇关于【并发】共享模型之管程的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/605993

相关文章

NFS实现多服务器文件的共享的方法步骤

《NFS实现多服务器文件的共享的方法步骤》NFS允许网络中的计算机之间共享资源,客户端可以透明地读写远端NFS服务器上的文件,本文就来介绍一下NFS实现多服务器文件的共享的方法步骤,感兴趣的可以了解一... 目录一、简介二、部署1、准备1、服务端和客户端:安装nfs-utils2、服务端:创建共享目录3、服

Golang的CSP模型简介(最新推荐)

《Golang的CSP模型简介(最新推荐)》Golang采用了CSP(CommunicatingSequentialProcesses,通信顺序进程)并发模型,通过goroutine和channe... 目录前言一、介绍1. 什么是 CSP 模型2. Goroutine3. Channel4. Channe

使用Nginx来共享文件的详细教程

《使用Nginx来共享文件的详细教程》有时我们想共享电脑上的某些文件,一个比较方便的做法是,开一个HTTP服务,指向文件所在的目录,这次我们用nginx来实现这个需求,本文将通过代码示例一步步教你使用... 在本教程中,我们将向您展示如何使用开源 Web 服务器 Nginx 设置文件共享服务器步骤 0 —

Python基于火山引擎豆包大模型搭建QQ机器人详细教程(2024年最新)

《Python基于火山引擎豆包大模型搭建QQ机器人详细教程(2024年最新)》:本文主要介绍Python基于火山引擎豆包大模型搭建QQ机器人详细的相关资料,包括开通模型、配置APIKEY鉴权和SD... 目录豆包大模型概述开通模型付费安装 SDK 环境配置 API KEY 鉴权Ark 模型接口Prompt

Python使用pysmb库访问Windows共享文件夹的详细教程

《Python使用pysmb库访问Windows共享文件夹的详细教程》本教程旨在帮助您使用pysmb库,通过SMB(ServerMessageBlock)协议,轻松连接到Windows共享文件夹,并列... 目录前置条件步骤一:导入必要的模块步骤二:配置连接参数步骤三:实例化SMB连接对象并尝试连接步骤四:

Linux使用粘滞位 (t-bit)共享文件的方法教程

《Linux使用粘滞位(t-bit)共享文件的方法教程》在Linux系统中,共享文件是日常管理和协作中的常见任务,而粘滞位(StickyBit或t-bit)是实现共享目录安全性的重要工具之一,本文将... 目录文件共享的常见场景基础概念linux 文件权限粘滞位 (Sticky Bit)设置共享目录并配置粘

大模型研发全揭秘:客服工单数据标注的完整攻略

在人工智能(AI)领域,数据标注是模型训练过程中至关重要的一步。无论你是新手还是有经验的从业者,掌握数据标注的技术细节和常见问题的解决方案都能为你的AI项目增添不少价值。在电信运营商的客服系统中,工单数据是客户问题和解决方案的重要记录。通过对这些工单数据进行有效标注,不仅能够帮助提升客服自动化系统的智能化水平,还能优化客户服务流程,提高客户满意度。本文将详细介绍如何在电信运营商客服工单的背景下进行

Andrej Karpathy最新采访:认知核心模型10亿参数就够了,AI会打破教育不公的僵局

夕小瑶科技说 原创  作者 | 海野 AI圈子的红人,AI大神Andrej Karpathy,曾是OpenAI联合创始人之一,特斯拉AI总监。上一次的动态是官宣创办一家名为 Eureka Labs 的人工智能+教育公司 ,宣布将长期致力于AI原生教育。 近日,Andrej Karpathy接受了No Priors(投资博客)的采访,与硅谷知名投资人 Sara Guo 和 Elad G

Retrieval-based-Voice-Conversion-WebUI模型构建指南

一、模型介绍 Retrieval-based-Voice-Conversion-WebUI(简称 RVC)模型是一个基于 VITS(Variational Inference with adversarial learning for end-to-end Text-to-Speech)的简单易用的语音转换框架。 具有以下特点 简单易用:RVC 模型通过简单易用的网页界面,使得用户无需深入了

透彻!驯服大型语言模型(LLMs)的五种方法,及具体方法选择思路

引言 随着时间的发展,大型语言模型不再停留在演示阶段而是逐步面向生产系统的应用,随着人们期望的不断增加,目标也发生了巨大的变化。在短短的几个月的时间里,人们对大模型的认识已经从对其zero-shot能力感到惊讶,转变为考虑改进模型质量、提高模型可用性。 「大语言模型(LLMs)其实就是利用高容量的模型架构(例如Transformer)对海量的、多种多样的数据分布进行建模得到,它包含了大量的先验