多线程--JUC-安全

简介: 多线程--JUC-安全

@[toc]

从底层原理来进行分类

互斥同步

 **使用各种互斥锁
                 Synchronized
                 ReentrantLock
                 Readwritelock**

**使用同步工具
Collections.synchronized(new ArrayList)等
Vector deng**

非互斥同步

原子类

atomic包 原子类

基本类型
AtomicInteger 整形,
AtomicLong 长整型
AtiomicBoolean 布尔

Array数组
AtomicIntergerArray 整形数组
AtomicLongArray 长整形数组
AtomicRefernceArray 引入数组

引用类型原子类
AtomicReference :引用类型
AtomicStampedReference 引用类型 带时间戳,可以解决ABA问题
升级原子类
后面文章会讲到

Adder加法器
LongAdder
DoubleAdder
Accumulator累加器
LongAccumulator
DoubleAccumulator

结合互斥和非互斥同步

并发容器
ConcurrentHashMap
并发队列

无同步方案不可变

final 关键字
ThreadLocal

1.线程安全问题

什么是线程安全问题:当多个线程同时访问一个全局变量,注意(做写的操作的时候可能会受到别的线程的干扰),做读的操作的时候不会发生线程安全问题

如 ++ , -- 修改等操作 抢火车票的操作,就会引发线程安全问题

在这里插入图片描述

模拟场景:

package com.yxl.demo.ThreadTest;

public class test5 {

    public static void main(String[] args) {

        TestDemo thread = new TestDemo();
        Thread t1 = new Thread(thread,"窗口一");
        Thread t2 = new Thread(thread,"窗口二");
        t1.start();
        t2.start();
    }
}

class TestDemo implements Runnable{
    //共享的火车票变量
    private  int count = 100;

    //重寫run方法
    @Override
    public void run() {
        while (count > 0){
            try {
                //休眠一下 方便出现并发问题
                Thread.sleep(50);
            }catch (Exception e){
                e.getMessage();
            }
            sale();
        }
    }
    //卖票
    public void sale(){
        if(count > 0){
            System.out.println(Thread.currentThread().getName() +"出售 :" +(100 -  count + 1));
            count--;
        }
    }
}

运行结果如下:会发现出现卖重复票的问题

在这里插入图片描述

解决这个线程安全问题

内置锁(Synchronized)

  • 保证线程原子性,当先车管进入方法自动的获取锁,当某一个线程获取到锁后,其他线程就会等待,
  • 锁的特性:只有一个线程进行使用。
  • 放程序执行完毕之久,就会把锁释放,但是锁会降低程序的运行效率,

Synchronized 也是 重入锁,内置锁也是互斥锁

使用 Synchronized 有两种方式, 同步方法(使用的是this锁),同步代码块

synchronized 修饰方法使用锁是当前this锁。

synchronized 修饰静态方法使用锁是当前类的字节码文件

显示锁 (lock锁)

解决方案 :

1.同步方法

在这里插入图片描述

2. 同步代码块

在这里插入图片描述

使用同步代码块的时候 注意(锁一定要使用同一个锁 )

问:如何解决多线程之间线程安全问题

*答:使用多线程之间同步synchronized或使用锁(lock)。*

问:为什么使用线程同步或使用锁能解决线程安全问题呢?

*答:将可能会发生数据冲突问题(线程不安全问题),只能让当前一个线程进行执行。代码执行完成后释放锁,让后才能让其他线程进行执行。这样的话就可以解决线程不安全问题。*

问:什么是多线程之间同步

*答:当多个线程共享同一个资源,不会受到其他线程的干扰。*

相关文章
|
3月前
|
安全 Java
java线程之List集合并发安全问题及解决方案
java线程之List集合并发安全问题及解决方案
360 1
|
8天前
|
监控 Java 调度
【Java学习】多线程&JUC万字超详解
本文详细介绍了多线程的概念和三种实现方式,还有一些常见的成员方法,CPU的调动方式,多线程的生命周期,还有线程安全问题,锁和死锁的概念,以及等待唤醒机制,阻塞队列,多线程的六种状态,线程池等
64 6
【Java学习】多线程&JUC万字超详解
|
28天前
|
算法 Java
JUC(1)线程和进程、并发和并行、线程的状态、lock锁、生产者和消费者问题
该博客文章综合介绍了Java并发编程的基础知识,包括线程与进程的区别、并发与并行的概念、线程的生命周期状态、`sleep`与`wait`方法的差异、`Lock`接口及其实现类与`synchronized`关键字的对比,以及生产者和消费者问题的解决方案和使用`Condition`对象替代`synchronized`关键字的方法。
JUC(1)线程和进程、并发和并行、线程的状态、lock锁、生产者和消费者问题
|
20天前
|
设计模式 Java 调度
JUC线程池: ScheduledThreadPoolExecutor详解
`ScheduledThreadPoolExecutor`是Java标准库提供的一个强大的定时任务调度工具,它让并发编程中的任务调度变得简单而可靠。这个类的设计兼顾了灵活性与功能性,使其成为实现复杂定时任务逻辑的理想选择。不过,使用时仍需留意任务的执行时间以及系统的实际响应能力,以避免潜在的调度问题影响应用程序的行为。
47 1
|
22天前
|
Java
【Java集合类面试十二】、HashMap为什么线程不安全?
HashMap在并发环境下执行put操作可能导致循环链表的形成,进而引起死循环,因而它是线程不安全的。
|
22天前
|
安全 算法 Java
【Java集合类面试二】、 Java中的容器,线程安全和线程不安全的分别有哪些?
这篇文章讨论了Java集合类的线程安全性,列举了线程不安全的集合类(如HashSet、ArrayList、HashMap)和线程安全的集合类(如Vector、Hashtable),同时介绍了Java 5之后提供的java.util.concurrent包中的高效并发集合类,如ConcurrentHashMap和CopyOnWriteArrayList。
【Java集合类面试二】、 Java中的容器,线程安全和线程不安全的分别有哪些?
|
13天前
|
Java API 调度
JUC线程池: FutureTask详解
总而言之,FutureTask是Java并发编程中一个非常实用的类,它在异步任务执行及结果处理方面提供了优雅的解决方案。在实现细节方面可以搭配线程池的使用,以及与Callable接口的配合使用,来完成高效的并发任务执行和结果处理。
21 0
|
1月前
|
存储 安全 Java
解锁Java并发编程奥秘:深入剖析Synchronized关键字的同步机制与实现原理,让多线程安全如磐石般稳固!
【8月更文挑战第4天】Java并发编程中,Synchronized关键字是确保多线程环境下数据一致性与线程安全的基础机制。它可通过修饰实例方法、静态方法或代码块来控制对共享资源的独占访问。Synchronized基于Java对象头中的监视器锁实现,通过MonitorEnter/MonitorExit指令管理锁的获取与释放。示例展示了如何使用Synchronized修饰方法以实现线程间的同步,避免数据竞争。掌握其原理对编写高效安全的多线程程序极为关键。
48 1
|
22天前
|
Java 程序员 容器
【多线程面试题二十四】、 说说你对JUC的了解
这篇文章介绍了Java并发包java.util.concurrent(简称JUC),它是JSR 166规范的实现,提供了并发编程所需的基础组件,包括原子更新类、锁与条件变量、线程池、阻塞队列、并发容器和同步器等多种工具。
|
2月前
|
缓存 安全 Java
多线程线程池问题之为什么手动创建的线程池比使用Executors类提供的线程池更安全
多线程线程池问题之为什么手动创建的线程池比使用Executors类提供的线程池更安全