同步的问题及解决方案

同步

同步的问题

当给狗狗食物的同时,狗狗又在吃,这会导致在运行过程中会出现食物的数据的错乱,有时候会多出数据,有时候会少出数据,这就让狗狗有时候会很吃亏,那么该如何解决呢?

实验体现

package multiThread2;

public class Animal{
    private String name;
    private int year;
    Animal(){}
    private int food;
    public Animal(String name,int year,int food){
        this.name = name;
        this.year = year;
        this.food = food;
    }

    @Override
    public String toString() {
        return "姓名:"+ name + ",年龄:" + year + ",来干饭了,还剩"+ food + "个食物";
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getYear() {
        return year;
    }

    public void setYear(int year) {
        this.year = year;
    }

    public int getFood() {
        return food;
    }

    public void setFood(int food) {
        this.food = food;
    }
    public void eatFood(){
        this.food--;
    }
    public void addFood(){
        this.food++;
    }
}

package multiThread2;

public class main {
    public static void main(String[] args) throws InterruptedException {
        Animal a2 = new Animal("旺财",3,200);
        System.out.println(a2.getName()+"原本的食物有"+a2.getFood());
        Thread[] addThread = new Thread[200];
        Thread[] reduceThread = new Thread[200];
//        n个线程增加狗的食物
        for (int i = 0; i < 200; i++) {
            Thread t = new Thread(() -> {
                a2.addFood();
                try{
                    Thread.sleep(100);
                }catch (Exception e){
                    e.printStackTrace();
                }
            });
            t.start();
            addThread[i]=t;
        }
//        n个线程减少狗的食物
        for (int i = 0; i < 200; i++) {
            Thread t = new Thread(() ->{
                a2.eatFood();
                try {
                    Thread.sleep(100);
                }catch (Exception e){
                    e.printStackTrace();
                }
            });
            t.start();
            reduceThread[i]=t;
        }
//        等待所有增加线程的结束
        for(Thread t: addThread){
            try {
                t.join();
            }catch (Exception e){
                e.printStackTrace();
            }
        }
        for (Thread t: reduceThread) {
            try {
                t.join();
            }catch (Exception e){
                e.printStackTrace();
            }
        }
        System.out.println(a2.getName()+"的食物变成了"+a2.getFood()+"个");
    }
}

在这里插入图片描述

同步出现问题的原因

增加食物和减少食物是同时发生的事件,这会导致在增加食物的反应时间的同时,减少食物的情况正在发生,来不及增加食物,这会导致食物结果减少,反之亦然。

解决方案

在增加食物这一个线程发生的过程中,其他线程禁止访问

  1. 增加线程获取到食物数据,并进行运算
  2. 在运算期间,减少线程试图来访问食物数据,但是不被允许
  3. 在增加线程执行完毕之后,减少线程才能访问食物数据
  4. 减少数据运行完毕

synchronized 同步对象概念

Object someObject = new Object();
synchronized(someObject){
}
  • synchronized表示当前线程,独占对象someObject当前线程独占 了对象someObject,如果有其他线程试图占有对象someObject,就会等待,直到当前线程释放对someObject的占用。
  • someObject 又叫同步对象,所有的对象,都可以作为同步对象
    为了达到同步的效果,必须使用同一个同步对象
  • 释放同步对象的方式: synchronized 块自然结束,或者有异常抛出
package multiThread2;

public class Animal{
    private String name;
    private int year;
    Animal(){}
    private int food;
    public Animal(String name,int year,int food){
        this.name = name;
        this.year = year;
        this.food = food;
    }

    @Override
    public String toString() {
        return "姓名:"+ name + ",年龄:" + year + ",来干饭了,还剩"+ food + "个食物";
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getYear() {
        return year;
    }

    public void setYear(int year) {
        this.year = year;
    }

    public int getFood() {
        return food;
    }

    public void setFood(int food) {
        this.food = food;
    }
    public void eatFood(){
        this.food--;
    }
    public void addFood(){
        this.food++;
    }
}

package multiThread2;

public class main {
    public static void main(String[] args) throws InterruptedException {
        Animal a2 = new Animal("旺财", 3, 200);
        System.out.println(a2.getName() + "原本的食物有" + a2.getFood());
        Thread[] addThread = new Thread[200];
        Thread[] reduceThread = new Thread[200];
        Object someObject = new Object();
//        n个线程增加狗的食物
//        for (int i = 0; i < 5; i++) {
        Thread t1 = new Thread(() -> {
            a2.addFood();
            try {
                System.out.println("t1试图进行");
                System.out.println("t1试图占有someObject对象");
                synchronized (someObject) {
                    System.out.println("t1占有someObject对象");
                    Thread.sleep(100);
                    System.out.println("t1释放someObject对象");
                }
                System.out.println("t1结束");
                Thread.sleep(100);
            } catch (Exception e) {
                e.printStackTrace();
            }
        });
        t1.start();
//            addThread[i] = t;
//        }
//        n个线程减少狗的食物
//        for (int i = 0; i < 5; i++) {
        Thread t2 = new Thread(() -> {
            a2.eatFood();
            try {
                System.out.println("t2试图进行");
                System.out.println("t2试图占有someObject对象");
                synchronized (someObject) {
                    System.out.println("t2占有someObject对象");
                    Thread.sleep(100);
                    System.out.println("t2释放someObject对象");
                }
                System.out.println("t2结束");
                Thread.sleep(100);
            } catch (Exception e) {
                e.printStackTrace();
            }
        });
        t2.start();
//            reduceThread[i]=t;
//        }
//        等待所有增加线程的结束
//        for(Thread t: addThread){
//            try {
//                t.join();
//            }catch (Exception e){
//                e.printStackTrace();
//            }
//        }
//        for (Thread t: reduceThread) {
//            try {
//                t.join();
//            }catch (Exception e){
//                e.printStackTrace();
//            }
//        }
//        System.out.println(a2.getName()+"的食物变成了"+a2.getFood()+"个");
    }
}

在这里插入图片描述

在方法前加一个修饰符synchronized

此时主函数可以不用加入synchronized

线程安全的类

如果一个类,它的方法都是有synchronized修饰的,那么该类就是线程安全的类

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

实验6 形态学图像处理

1. 实验目的 ①掌握数字图像处理中&#xff0c;形态学方法的基本思想&#xff1b; ②掌握膨胀、腐蚀、开运算、闭运算等形态学基本运算方法&#xff1b; ③能够利用形态学基本运算方法&#xff0c;编程实现图像去噪&#xff0c;边界提取等功能。 2. 实验内容 ①调用Matlab /…

Excel 数据筛选难题解决

人不走空 &#x1f308;个人主页&#xff1a;人不走空 &#x1f496;系列专栏&#xff1a;算法专题 ⏰诗词歌赋&#xff1a;斯是陋室&#xff0c;惟吾德馨 目录 &#x1f308;个人主页&#xff1a;人不走空 &#x1f496;系列专栏&#xff1a;算法专题 ⏰诗词歌…

一个中文和越南语双语版本的助贷平台开源源码

一个中文和越南语双语版本的助贷平台开源源码。后台试nodejs。 后台 代理 前端均为vue源码&#xff0c;前端有中文和越南语。 前端ui黄色大气&#xff0c;逻辑操作简单&#xff0c;注册可对接国际短信&#xff0c;可不对接。 用户注册进去填写资料&#xff0c;后台审批&…

职场必备:三大神器助你完美驾驭工作与生活;从 GTD 到SMART再到OKR:提升效率的终极指南;告别拖延,高效工作的秘密武器!

在现代职场和个人生活中&#xff0c;有效的时间管理和目标设定是成功的关键。我们每天都面临着无数的任务和目标。如何在纷繁复杂的日常中保持专注&#xff0c;高效地完成工作&#xff1f; GTD&#xff08;Getting Things Done&#xff09; GTD&#xff08;Getting Things Don…

10款好用不火的PC软件,真的超好用!

AI视频生成&#xff1a;小说文案智能分镜智能识别角色和场景批量Ai绘图自动配音添加音乐一键合成视频https://aitools.jurilu.com/市场上有很多软件&#xff0c;除了那些常见的大众化软件&#xff0c;还有很多不为人知的小众软件&#xff0c;它们的作用非常强大&#xff0c;简洁…

web全屏api,实现元素放大全屏,requestFullscreen,exitFullscreen

全屏api 主要方法 document.exitFullscreen(); 退出页面全屏状态&#xff0c;document是全局文档对象 dom.requestFullscreen(); 使dom进入全屏状态&#xff0c;异步&#xff0c;dom是一个dom元素 dom.onfullscreenchange&#xff08;&#xff09;; 全…

imx6ull/linux应用编程学习(6)jpeg和png的图片显示

1.JPEG图片显示 JPEG&#xff08;Joint Photographic Experts Group&#xff09;是由国际标准组织为静态图像所建立的第一个国际数字图像压缩标准&#xff0c;也是至今一直在使用的、应用最广的图像压缩标准。JPEG 由于可以提供有损压缩&#xff0c;因此压缩比可以达到其他传统…

SpringBoot | 使用jwt令牌实现登录认证,使用Md5加密实现注册

对于登录认证中的令牌&#xff0c;其实就是一段字符串&#xff0c;那为什么要那么麻烦去用jwt令牌&#xff1f;其实对于登录这个业务&#xff0c;在平常我们实现这个功能时&#xff0c;可能大部分都是通过比对用户名和密码&#xff0c;只要正确&#xff0c;就登录成功&#xff…

美团外卖搜索基于Elasticsearch的优化实践--图文解析

美团外卖搜索基于Elasticsearch的优化实践–图文解析 前言 美团在外卖搜索业务场景中大规模地使用了 Elasticsearch 作为底层检索引擎&#xff0c;随着业务量越来越大&#xff0c;检索速度变慢了&#xff0c;CPU快累趴了&#xff0c;所以要进行优化。经过检测&#xff0c;发现…

智慧校园-办公管理系统总体概述

智慧校园行政办公系统是专为高校及教育机构定制的数字化办公解决方案&#xff0c;它整合了众多办公应用与服务&#xff0c;旨在全面提升校园行政管理的效率与便捷性&#xff0c;推动信息的自由流动&#xff0c;实现绿色无纸化办公环境。该系统作为一个综合平台&#xff0c;将日…

redis实战-缓存穿透问题及解决方案

定义理解 缓存穿透&#xff1a;缓存穿透是指客户端请求的数据在缓存中和数据库中都不存在&#xff0c;这样缓存永远都不会生效&#xff08;只有数据库查到了&#xff0c;才会让redis缓存&#xff0c;但现在的问题是查不到&#xff09;&#xff0c;会频繁的去访问数据库。 解决…

【Spring】DAO 和 Repository 的区别

DAO 和 Repository 的区别 1.概述2.DAO 模式2.1 User2.2 UserDao2.3 UserDaoImpl 3.Repository 模式3.1 UserRepository3.2 UserRepositoryImpl 4.具有多个 DAO 的 Repository 模式4.1 Tweet4.2 TweetDao 和 TweetDaoImpl4.3 增强 User 域4.4 UserRepositoryImpl 5.比较两种模式…

RabbitMQ实践——临时队列

临时队列是一种自动删除队列。当这个队列被创建后&#xff0c;如果没有消费者监听&#xff0c;则会一直存在&#xff0c;还可以不断向其发布消息。但是一旦的消费者开始监听&#xff0c;然后断开监听后&#xff0c;它就会被自动删除。 新建自动删除队列 我们创建一个名字叫qu…

【CodinGame】CLASH OF CODE - 20240630

前言 本文是CodinGame&#xff08;图片来自此&#xff09;随手做的几个&#xff0c;供记录用 要求&#xff1a; 代码 import math import syss input()for n in range(len(s)):print(s[n:])要求 代码 import sys import math# Auto-generated code below aims at helpi…

大模型压缩量化方案怎么选?无问芯穹Qllm-Eval量化方案全面评估:多模型、多参数、多维度

基于 Transformer架构的大型语言模型在各种基准测试中展现出优异性能&#xff0c;但数百亿、千亿乃至万亿量级的参数规模会带来高昂的服务成本。例如GPT-3有1750亿参数&#xff0c;采用FP16存储&#xff0c;模型大小约为350GB&#xff0c;而即使是英伟达最新的B200 GPU 内存也只…

SpringBoot使用redis 笔记(视频摘抄 哔哩哔哩博主(感谢!):遇见狂神)

springboot集成redis步骤 1.创建springboot项目 2.配置连接 3.测试 创建springboot项目 创建以一个Maven项目 创建之后查看pom.xml配置文件&#xff0c;可以看到 pom文件里面导入了 data-redis 的依赖&#xff0c;那我们就可以在知道&#xff0c;springboot集成redis操作…

详解flink sql, calcite logical转flink logical

文章目录 背景示例FlinkLogicalCalcConverterBatchPhysicalCalcRuleStreamPhysicalCalcRule其它算子FlinkLogicalAggregateFlinkLogicalCorrelateFlinkLogicalDataStreamTableScanFlinkLogicalDistributionFlinkLogicalExpandFlinkLogicalIntermediateTableScanFlinkLogicalInt…

20240623日志:大模型压缩-sliceGPT

context 1. 剪枝方案图释2. 正交矩阵Q 1. 剪枝方案图释 Fig. 1.1 剪枝方案 图中的阴影是表示丢弃掉这部分数据。通过引入正交矩阵 Q Q Q使 Q ⊤ Q Q Q ⊤ I \mathrm{Q}^\top\mathrm{Q}\mathrm{Q}\mathrm{Q}^\top\mathrm{I} Q⊤QQQ⊤I&#xff0c;来大量缩减 X X X的列数和 W …

【操作系统】内存管理——页面分配策略(个人笔记)

学习日期&#xff1a;2024.6.28 内容摘要&#xff1a;页面分配策略和内存映射文件&#xff0c;内存映射文件 页面分配置换策略 基本概念 驻留集&#xff0c;指请求分页存储管理中给进程分配的物理块的集合&#xff0c;在采用了虚拟存储技术的系统中&#xff0c;驻留集大小一…

第3章-数据类型和运算符

#本章目标 掌握Python中的保留字与标识符 理解Python中变量的定义及使用 掌握Python中基本数据类型 掌握数据类型之间的相互转换 掌握eval()函数的使用 了解不同的进制数 掌握Python中常用的运算符及优先级1&#xff0c;保留字与标识符 保留字 指在Python中被赋予特定意义的一…