线程安全问题的三种解决方案_c# winfrom 线程安全去掉-程序员宅基地

技术标签: JavaSE  

关于线程安全问题:
1.什么是线程安全问题?.为什么会出现线程安全问题?
首先我们需要了解一个概念,临界资源:即多个线程同时访问的资源(共享资源)。
当多个线程同时操作临界资源的时候,就容易出现线程安全问题,线程安全问题只会影响到线程对同一个共享的全局变量的写操作。

》接下来来演示一下出现线程安全问题的案例:窗口售票

public class Ticket implements Runnable{//创建对象继承Runnable接口  作为参数传入Thread对象构造中
    private  int ticket=100;//设定临界资源100张票
    @Override
    public void run() {
        while (true){//死循环
            if(ticket<=0){票数少于等于0结束循环  不再买票
                break;
            }
            try {
                Thread.sleep(10);//休眠只是为了更明显突出多线程安全问题
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName()+"卖出了第"+ticket+"张票");//打印卖票
            ticket--;//每卖出一张票  票数减一
        }
    }
    }
public class TirketDemo {
    public static void main(String[] args) {
        Ticket ticket=new Ticket();//创建票对象
        new Thread(ticket,"窗口一").start();//作为参数传入
        new Thread(ticket,"窗口二").start();
        new Thread(ticket,"窗口三").start();
        new Thread(ticket,"窗口四").start();
    }
}

运行结果:在这里插入图片描述
可以看出来不仅出现重复,还有出现负数票的情况,这是因为在线程运行时候,刚判定结束后,cpu倍另外的线程抢夺而出现的问题,再次醒来之后,票数已经被其他线程修改,该线程仍然继续向下执行减一 就有可能出现负数或者重复。解决方法大致有三种:

一同步代码块

说到同步代码块就要说锁了,一个线程在访问临界资源的时候,如果给这个资源“上一把锁”,这个时候如果其他线程也要访问这个资源,就得在“锁”外面等待。
锁对象可以是任意对象,但是要求必须是同一把锁才行。
加进同步代码块的代码,执行时候其他线程必须等待,待当前进入的线程执行完,释放锁对象之后其他线程才能进去执行,以此类推。
同步中的线程没有执行完毕不会释放锁,同步外的线程没有锁对象,进不去同步

代码实现:加了同步代码块之后就不会出现以上问题

public class Ticket implements Runnable{
    private  int ticket=100;
    @Override
    public void run() {
        while (true){
            synchronized (this) {//在这里加上同步代码块并传入所对象
                if(ticket<=0){
                    break;
                }
                try {
                    Thread.sleep(10);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println(Thread.currentThread().getName()+"卖出了第"+ticket+"张票");
                ticket--;
            }
        }
    }

二.同步方法

创建一个方法,该方法必须用synchronized关键字修饰,将出现安全问题的代码放入到同步方法中,再在Runnable实现类重写的run方法中调用该方法即可。
同步方法默认的锁对象是调用该方法的类 也就是this

public class Ticket implements Runnable{
    private  int ticket=100;
    @Override
    public void run() {
        while (true) {
            sellTicket();
        }
    }
    public synchronized  boolean sellTicket(){
//注意:while循环不能加到方法里面  因为这样的话  调用一次方法  一个线程就会把票卖完了  这里我们返回boolean来判定
            if(ticket<=0){//票数少于等于0结束循环  不再买票
               return false;
            }
            try {
                Thread.sleep(10);//休眠只是为了更明显突出多线程安全问题
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName()+"卖出了第"+ticket+"张票");//打印卖票
            ticket--;//每卖出一张票  票数减一
        return true;

    }

静态同步方法:即在方法加静态修饰符,静态方法的锁不能是this,因为this只有在类加载之后创建对象才会有,而静态方法在创建对象之前就已经加载,所以同步静态方法的锁是本类的class

public class Ticket implements Runnable{
    private   static int ticket=100;//静态方法只能访问静态变量
    @Override
    public void run() {
        while (true) {
            sellTicket();
        }
    }
    public synchronized  static boolean sellTicket(){
//注意:while循环不能加到方法里面  因为这样的话  调用一次方法  一个线程就会把票卖完了
            if(ticket<=0){//票数少于等于0结束循环  不再买票
               return false;
            }
            try {
                Thread.sleep(10);//休眠只是为了更明显突出多线程安全问题
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName()+"卖出了第"+ticket+"张票");//打印卖票
            ticket--;//每卖出一张票  票数减一
        return true;

    }

三.Lock锁

从jdk1.5之后加入新的接口 Lock,ReentrantLock是Lock接口的实现类。
通过显式定义同步锁对象来实现同步,同步锁提供了比synchronized代码块更广泛的锁定操
注意:最好将 unlock的操作放到finally块中
通过使用ReentrantLock这个类来进行锁的操作,它实现了Lock接口,使用ReentrantLock可以显式地加锁、释放锁。

lock锁使用步骤:1.在成员位置创建一个Reentrantlock对象,因为lock是一个接口需要实现类才能创建对象。
2.在可能会出现安全问题的代码前调用接口方法中的lock()方法获取锁
3.在可能会出现安全问题的代码后面调用接口方法中的Unlock()方法释放锁
注意:最好将有安全问题的代码放到try…finally{ … }中,finally中用于释放锁。

代码:

public class Ticket implements Runnable{
    private   int ticket=100;

    Lock lock=new ReentrantLock();//创建锁对象
    @Override
    public void run() {
        while (true){
            lock.lock();//获取锁
            try {
                if(ticket<=0){
                    break;
                }
                System.out.println(Thread.currentThread().getName()+"卖出了第"+ticket+"张票");
                ticket--;
            } catch (Exception e) {
                e.printStackTrace();
            }finally {
                lock.unlock();//释放锁
            }
        }

    }
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/shunshizhen120412/article/details/99185893

智能推荐

蓝凌EIS智慧协同平台saveImg接口存在任意文件上传漏洞_蓝凌eis智慧协同平台文件上传漏洞-程序员宅基地

文章浏览阅读979次。蓝凌智慧协同平台eis集合了非常丰富的模块,满足组织企业在知识、协同、项目管理系统建设等需求。_蓝凌eis智慧协同平台文件上传漏洞

LLaVA-1.5-程序员宅基地

文章浏览阅读193次。与InstructBLIP或Qwen-VL在数亿甚至数十几亿的图像文本配对数据上训练的、专门设计的视觉重新采样器相比,LLaVA用的是最简单的LMM架构设计,只需要在600K个图像-文本对上,训练一个简单的完全连接映射层即可。结果表明,LLaVA-1.5不仅可以使用更少的预训练和指令微调数据,而且还可以利用最简单的架构、学术计算和公共数据集来实现最佳的性能——在12个基准中的11个上取得了SOTA。为了解决这个问题,研究人员建议在VQA问题的末尾,添加一个可以明确输出格式的提示,进而让模型生成简短回答。

ORACLE基本数据类型总结_oracle 数值类型最大值-程序员宅基地

文章浏览阅读442次。2013-08-17 21:04 by 潇湘隐者, 100246 阅读, 5 评论, 收藏, 编辑 ORACLE基本数据类型(亦叫内置数据类型 built-in datatypes)可以按类型分为:字符串类型、数字类型、日期类型、LOB类型、LONG RAW& RAW类型、ROWID & UROWID类型。在讲叙字符串类型前,先要讲一下编码。字符串类型的数据可依编码方式分成_oracle 数值类型最大值

10种机器学习算法_决策树和mlp-程序员宅基地

文章浏览阅读315次。作为数据科学家的实践者,我们必须了解一些通用机器学习的基础知识算法,这将帮助我们解决所遇到的新领域问题。本文对通用机器学习算法进行了简要的阐述,并列举了它们的相关资源,从而帮助你能够快速掌握其中的奥妙。▌1.主成分分析(PCA)/ SVDPCA是一种无监督的方法,用于对由向量组成的数据集的全局属性进行理解。本文分析了数据点的协方差矩阵,以了解哪些维度(大部分情况)/数据点(少数情况)更为重要,即它..._决策树和mlp

桥接模式的实现-程序员宅基地

文章浏览阅读148次。在这个示例中,我们使用std::shared_ptr来管理Implementor对象的生命周期,确保在不再需要时自动释放资源。通过智能指针的使用,我们避免了手动管理内存的复杂性,提高了代码的可靠性和可维护性。希望这个示例能帮助你理解如何使用智能指针来实现桥接模式。当使用智能指针来实现桥接模式时,我们可以利用std::shared_ptr或std::unique_ptr来管理对象的生命周期,确保资源的正确释放。

制造业敏感文件外发不安全?一招解锁更高效的加密方式!-程序员宅基地

文章浏览阅读440次,点赞11次,收藏8次。云盒子在制造业上有丰富的部署经验,在面向制造类企业的重要文件,可以通过审计、授权、文件加密进行多重保护,使得图纸文件、专利技术、采购订单等敏感数据等到有效保护,做到无处可泄,同时安全可靠,也不会对日常工作效率有影响 ,实现真正有效的企业文件保护的目的,达到既防止机密文件外泄和扩散,又支持内部知识积累和文件共享的目的。云盒子的加密方式是通过将本地文件数据上传到云盘进行统一加密存储,而不是对设备加密,通过【本地加密】+【云加密】双重组合下,不管用什么设备打开文件都受到管控,使管理者管理起来能够更高效。

随便推点

计算几何讲义——计算几何中的欧拉定理-程序员宅基地

文章浏览阅读188次。在处理计算几何的问题中,有时候我们会将其看成图论中的graph图,结合我们在图论中学习过的欧拉定理,我们可以通过图形的节点数(v)和边数(e)得到不是那么好求的面数f。 平面图中的欧拉定理: 定理:设G为任意的连通的平面图,则v-e+f=2,v是G的顶点数,e是G的边数,f是G的面数。证明:其实有点类似几何学中的欧拉公式的证明方法,这里采用归纳证明的方法。对m..._怎么证明平面图欧拉定理

c语言中各种括号的作用,C语言中各种类型指针的特性与用法介绍-程序员宅基地

文章浏览阅读750次。C语言中各种类型指针的特性与用法介绍本文主要介绍了C语言中各种类型指针的特性与用法,有需要的朋友可以参考一下!想了解更多相关信息请持续关注我们应届毕业生考试网!指针为什么要区分类型:在同一种编译器环境下,一个指针变量所占用的内存空间是固定的。比如,在16位编译器环境 下,任何一个指针变量都只占用8个字节,并不会随所指向变量的类型而改变。虽然所有的指针都只占8个字节,但不同类型的变量却占不同的字节数..._c语言带括号指针

缅甸文字库 缅甸语字库 缅甸字库算法_0x103c-程序员宅基地

文章浏览阅读9.5k次。字库交流 QQ:2229691219 缅甸语比较特殊、缅甸语有官方和民间之分,二者不同的是编码机制不同,因此这2种缅甸语的字串翻译、处理引擎、字库都是不同的。我们这里只讨论官方语言。 缅文、泰文等婆罗米系文字大多是元音附标文字,一般辅音字母自带默认元音可以发音,真正拼写词句时元音像标点符号一样附标在辅音上下左右的相应位置。由于每个元音位于辅音的具体位置是有自己的规则的,当只书写..._0x103c

Python+django+vue校园二手闲置物品拍卖系统pycharm毕业设计项目推荐_基于python+django+vue实现的校园二手交易平台-程序员宅基地

文章浏览阅读200次。在校园,随着学生数量的增多,存在许多生活和学习物品,许多学习用品经过一学期学习之后往往被闲置,一些出于一时喜欢而购买的物品使用机会少而被闲置,还有一些物品以低廉的价格卖给资源回收站,造成巨大的资源浪费。校园闲置物品拍卖系统使用python技术,MySQL数据库进行开发,系统后台使用django框架进行开发,具有低耦合、高内聚的特点,其中校园用户通过人脸识别的方法增加系统安全性,在闲置物品推荐中,使用协同过滤算法进行商品推荐。系统的开发,帮助高校有效的对闲置物品进行管理,提高了闲置物品销售的效率。_基于python+django+vue实现的校园二手交易平台

【推荐系统论文精读系列】(十)--Wide&Deep Learning for Recommender Systems_引用《wide & deep learning for recommender systems》-程序员宅基地

文章浏览阅读1.1k次,点赞3次,收藏3次。文章目录Wide & Deep Learning for Recommender Systems一、摘要二、介绍三、推荐系统综述四、Wide&Deep学习4.1 Wide部分4.2 Deep部分4.3 联合训练 Wide&Deep ModelPreferenceWide & Deep Learning for Recommender Systems一、摘要具有非线性特征转化能力的广义线性模型被广泛用于大规模的分类和回归问题,对于那些输入数据是极度稀疏的情况下。通过使用交_引用《wide & deep learning for recommender systems》

c++ sleep函数_Linux 多线程应用中如何编写安全的信号处理函数-程序员宅基地

文章浏览阅读171次。关于代码的可重入性,设计开发人员一般只考虑到线程安全,异步信号处理函数的安全却往往被忽略。本文首先介绍如何编写安全的异步信号处理函数;然后举例说明在多线程应用中如何构建模型让异步信号在指定的线程中以同步的方式处理。Linux 多线程应用中编写安全的信号处理函数在开发多线程应用时,开发人员一般都会考虑线程安全,会使用 pthread_mutex 去保护全局变量。如果应用中使用了信号,而且信号的产生不..._linux c++ sleep 不被中断

推荐文章

热门文章

相关标签