Java开发——37.多线程_(线程安全)

简介: 什么是线程安全?窗口售票是经典案例。

进程:系统分配资源的单位;

线程:处理器任务调度和执行的单位,线程之间共享进程资源。


线程不同步问题:

   问题:会出现多个窗口买一张票的情况/卖错票(票号为0/-1)

   问题出现原因:一个线程已经运行了,但是还没有执行ticket--,后续线程也进来了,导致他们操作了同一张票号...

   解决:如果一个线程已经运行了,但是在运行的过程中阻塞了,也不会给后续线程分配CPU资源,直至该线程运行完毕。


线程安全的两种方法:1.方式一:同步代码块(synchronized);2.方式二:同步方法。

image.png


注意:同步代码块和同步方法统一的规则:不能包含过多的代码,也不能包含过少的代码,只包含存在安全隐患的代码。


1.方式一:同步代码块;


书写格式:



synchronized(同步监视器){
//需要同步的代码块}
/***   参数说明:*        1.需要被同步的代码:操作共享数据的代码(注意不能多包含代码,也不能少包含代码);*            共享数据:多个线程共同操作的变量(例如,ticket票号)*        2.同步监视器,俗称:锁。 任何类型的对象都能充当锁*           要求:所有的线程必须保证共享一把锁。*              注意类也是对象(在学习反射的时候会详讲),可以使用(类名.class)来同步监视器;*                  类只会被加载一次。*              注意使用继承Thread类时 慎用(this)来充当同步监视器;*        解释:(卫生间的锁不能从外面决定关锁/解锁,只能被进卫生间的人从里面关锁/解锁,也就是一个卫生间在同一时间的关锁/解锁只能被一个已经进入卫生间的人操作)。*/


满足条件:

1.多个线程共用一把锁;

2.任何类型的对象都能充当同步监视器(锁);

3.需要同步的代码,不能多包也不能少包。


1.1.继承Thread类


publicclassPraSaleTicket {
publicstaticvoidmain(String[] args) {
Salesale01=newSale();
Salesale02=newSale();
Salesale03=newSale();
sale01.setName("窗口01");
sale02.setName("窗口02");
sale03.setName("窗口03");
sale01.start();
sale02.start();
sale03.start();
    }
}
classSaleextendsThread{
privatestaticintticket=100;
ObjectmyLock=newObject();
@Overridepublicvoidrun() {
//        synchronized (myLock) { // 不能多包含代码,也不能少包含代码while (true){
synchronized (Sale.class) {//当前类为静态类,只能加载一次,所以可以让当前类作为锁if (ticket>0) {
try {
Thread.currentThread().sleep(100);
                    } catch (InterruptedExceptione) {
e.printStackTrace();
                    }
System.out.println(getName() +"_票号为:"+ticket);
ticket--;
                } else {
break;
                }
            }
        }
    }
}


1.2.实现Runnable接口


publicclassPraSaleTicket_Runnable {
publicstaticvoidmain(String[] args) {
SaleTicketssale=newSaleTickets();
Threadthread=newThread(sale);
Threadthread2=newThread(sale);
Threadthread3=newThread(sale);
thread.setName("窗口01");
thread2.setName("窗口02");
thread3.setName("窗口03");
thread.start();
thread2.start();
thread3.start();
    }
}
classSaleTicketsimplementsRunnable{
privateintticket=100;
//    Object olock = new Object(); // 任何类型的对象都能充当锁。MyLockmyLock=newMyLock();
@Overridepublicvoidrun() {
//    错误的存放位置:Object olock = new Object(); // 任何类型的对象都能充当锁。while (true) {
//    错误的存放位置:Object olock = new Object(); // 任何类型的对象都能充当锁。synchronized(myLock) {
if (ticket>0) {
//如果票号大于0,假设现在票号为1,线程1抢到了CPU,但是还没往后面走,线程2页签到了资源,线程3也抢到了...导致输出的票号出现了:1 0 -1 的现象。//如果此处没有调用sleep(),也会出现重复票号的问题...try {
Thread.currentThread().sleep(100);
                    } catch (InterruptedExceptione) {
e.printStackTrace();
                    }
System.out.println(Thread.currentThread().getName() +"_票号为:"+ticket);
ticket--;
                } else {
break;
                }
        }
        }
    }
}
//任何类型的对象都能作为同步监视器。classMyLock{
}


2.方式二:同步方法;

将存在线程安全隐患的代码形成一个方法,即为同步方法。


同步方法写法:


修饰符 (static) synchronized返回值类型方法名(){}


注意事项:

1.多个线程共用一把锁;

2.需要同步的代码,不能多包也不能少包。

3.静态的同步方法:默认使用的同步监视器是,当前类.class;

4.非静态的同步方法:默认使用的同步监视器是,this,指代当前类。


注意此时将需要同步的方法,使用了同步方法来保证线程安全,针对买100张票的话,我们没办法终止循环,所以在循环结束后需要手动终止循环!!!


2.1.继承Thread类


publicclassPraSaleTicket02 {
publicstaticvoidmain(String[] args) {
Salessale01=newSales();
Salessale02=newSales();
Salessale03=newSales();
sale01.setName("窗口01");
sale02.setName("窗口02");
sale03.setName("窗口03");
sale01.start();
sale02.start();
sale03.start();
    }
}
classSalesextendsThread{
privatestaticintticket=100;
@Overridepublicvoidrun() {
while (true){
showTicket();
        }
    }
privatestaticsynchronizedvoidshowTicket(){//默认的同步监视器是当前类(Sales.class)//        private synchronized void showTicket(){ //使用该方法是错的,因为这样是默认传入的同步监视器是this(也就是sale01,sale02,sale03)//不能保证多个线程共用一把锁,所以线程是不安全的。if (ticket>0) {
try {
Thread.currentThread().sleep(100);
            } catch (InterruptedExceptione) {
e.printStackTrace();
            }
System.out.println(Thread.currentThread().getName() +"_票号为:"+ticket);
ticket--;
        }
    }
}


2.2.实现Runnable接口



publicclassPraSaleTicket_Runnable02 {
publicstaticvoidmain(String[] args) {
SaleTicketsale=newSaleTicket();
Threadthread=newThread(sale);
Threadthread2=newThread(sale);
Threadthread3=newThread(sale);
thread.setName("窗口01");
thread2.setName("窗口02");
thread3.setName("窗口03");
thread.start();
thread2.start();
thread3.start();
    }
}
classSaleTicketimplementsRunnable{
privateintticket=100;
@Overridepublicvoidrun() {
while (true) {
showTickets();
        }
    }
privatesynchronizedvoidshowTickets(){//默认的同步监视器是SaleTicket.classif (ticket>0) {
try {
Thread.currentThread().sleep(100);
            } catch (InterruptedExceptione) {
e.printStackTrace();
            }
System.out.println(Thread.currentThread().getName() +"_票号为:"+ticket);
ticket--;
        }
    }


线程安全的好处/局限:

好处:解决线程安全问题;

局限:让线程效率低,相当于把多线程问题变成了单线程的问题。

相关文章
|
20天前
|
Java 开发者 微服务
Spring Boot 入门:简化 Java Web 开发的强大工具
Spring Boot 是一个开源的 Java 基础框架,用于创建独立、生产级别的基于Spring框架的应用程序。它旨在简化Spring应用的初始搭建以及开发过程。
38 6
Spring Boot 入门:简化 Java Web 开发的强大工具
|
7天前
|
存储 JavaScript 前端开发
基于 SpringBoot 和 Vue 开发校园点餐订餐外卖跑腿Java源码
一个非常实用的校园外卖系统,基于 SpringBoot 和 Vue 的开发。这一系统源于黑马的外卖案例项目 经过站长的进一步改进和优化,提供了更丰富的功能和更高的可用性。 这个项目的架构设计非常有趣。虽然它采用了SpringBoot和Vue的组合,但并不是一个完全分离的项目。 前端视图通过JS的方式引入了Vue和Element UI,既能利用Vue的快速开发优势,
54 13
|
6天前
|
安全 Java API
java如何请求接口然后终止某个线程
通过本文的介绍,您应该能够理解如何在Java中请求接口并根据返回结果终止某个线程。合理使用标志位或 `interrupt`方法可以确保线程的安全终止,而处理好网络请求中的各种异常情况,可以提高程序的稳定性和可靠性。
37 6
|
12天前
|
算法 Java API
如何使用Java开发获得淘宝商品描述API接口?
本文详细介绍如何使用Java开发调用淘宝商品描述API接口,涵盖从注册淘宝开放平台账号、阅读平台规则、创建应用并申请接口权限,到安装开发工具、配置开发环境、获取访问令牌,以及具体的Java代码实现和注意事项。通过遵循这些步骤,开发者可以高效地获取商品详情、描述及图片等信息,为项目和业务增添价值。
44 10
|
5天前
|
前端开发 Java 测试技术
java日常开发中如何写出优雅的好维护的代码
代码可读性太差,实际是给团队后续开发中埋坑,优化在平时,没有那个团队会说我专门给你一个月来优化之前的代码,所以在日常开发中就要多注意可读性问题,不要写出几天之后自己都看不懂的代码。
42 2
|
19天前
|
存储 监控 小程序
Java中的线程池优化实践####
本文深入探讨了Java中线程池的工作原理,分析了常见的线程池类型及其适用场景,并通过实际案例展示了如何根据应用需求进行线程池的优化配置。文章首先介绍了线程池的基本概念和核心参数,随后详细阐述了几种常见的线程池实现(如FixedThreadPool、CachedThreadPool、ScheduledThreadPool等)的特点及使用场景。接着,通过一个电商系统订单处理的实际案例,分析了线程池参数设置不当导致的性能问题,并提出了相应的优化策略。最终,总结了线程池优化的最佳实践,旨在帮助开发者更好地利用Java线程池提升应用性能和稳定性。 ####
|
15天前
|
安全 算法 Java
Java多线程编程中的陷阱与最佳实践####
本文探讨了Java多线程编程中常见的陷阱,并介绍了如何通过最佳实践来避免这些问题。我们将从基础概念入手,逐步深入到具体的代码示例,帮助开发者更好地理解和应用多线程技术。无论是初学者还是有经验的开发者,都能从中获得有价值的见解和建议。 ####
|
15天前
|
Java 调度
Java中的多线程编程与并发控制
本文深入探讨了Java编程语言中多线程编程的基础知识和并发控制机制。文章首先介绍了多线程的基本概念,包括线程的定义、生命周期以及在Java中创建和管理线程的方法。接着,详细讲解了Java提供的同步机制,如synchronized关键字、wait()和notify()方法等,以及如何通过这些机制实现线程间的协调与通信。最后,本文还讨论了一些常见的并发问题,例如死锁、竞态条件等,并提供了相应的解决策略。
40 3
|
15天前
|
JavaScript 安全 Java
java版药品不良反应智能监测系统源码,采用SpringBoot、Vue、MySQL技术开发
基于B/S架构,采用Java、SpringBoot、Vue、MySQL等技术自主研发的ADR智能监测系统,适用于三甲医院,支持二次开发。该系统能自动监测全院患者药物不良反应,通过移动端和PC端实时反馈,提升用药安全。系统涵盖规则管理、监测报告、系统管理三大模块,确保精准、高效地处理ADR事件。
|
16天前
|
监控 Java 开发者
深入理解Java中的线程池实现原理及其性能优化####
本文旨在揭示Java中线程池的核心工作机制,通过剖析其背后的设计思想与实现细节,为读者提供一份详尽的线程池性能优化指南。不同于传统的技术教程,本文将采用一种互动式探索的方式,带领大家从理论到实践,逐步揭开线程池高效管理线程资源的奥秘。无论你是Java并发编程的初学者,还是寻求性能调优技巧的资深开发者,都能在本文中找到有价值的内容。 ####
下一篇
DataWorks