返回顶部
首页 > 资讯 > 后端开发 > Python >彻底搞懂Java多线程(四)
  • 461
分享到

彻底搞懂Java多线程(四)

2024-04-02 19:04:59 461人浏览 八月长安

Python 官方文档:入门教程 => 点击学习

摘要

目录SimpleDateFORMat非线程安全问题ThreadLocalThreadLocal的原理ThreadLocal常用方法ThreadLocal的初始化Inheritable

SimpleDateFormat非线程安全问题

实现1000个线程的时间格式化


package SimpleDateFormat;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.concurrent.LinkedBlockingDeque;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;

public class SimpleDateFormat1 {
    private static SimpleDateFormat simpleDateFormat = new SimpleDateFormat("mm:ss");
    public static void main(String[] args) {
        ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(10,10,100,
                TimeUnit.MILLISECONDS,new LinkedBlockingDeque<>(1000),new ThreadPoolExecutor.DiscardPolicy());

        for (int i = 0; i < 1001; i++) {
            int finalI = i;
            threadPoolExecutor.submit(new Runnable() {
                @Override
                public void run() {
                    Date date = new Date(finalI * 1000);
                    myFormatTime(date);
                }
            });
        }
        threadPoolExecutor.shutdown();
    }
    private static void myFormatTime(Date date){
        System.out.println(simpleDateFormat.format(date));
    }
}

产生了线程不安全的问题👇:

在这里插入图片描述

这是因为:

在这里插入图片描述

多线程的情况下:

在这里插入图片描述

线程1在时间片用完之后,线程2来setTime()那么线程1的得到了线程2的时间。

所以可以使用加的操作:

在这里插入图片描述

就不会有重复的时间了

在这里插入图片描述

但是虽然可以解决线程不安全的问题,但是排队等待锁,性能就会变得低

所以可以使用局部变量:

在这里插入图片描述

也解决了线程不安全的问题:

在这里插入图片描述

但是每次也都会创建新的私有变量

那么有没有一种方案既可以避免加锁排队执行,又不会每次创建任务的时候不会创建私有的变量呢?

那就是ThreadLocal👇:

ThreadLocal

ThreadLocal的作用就是让每一个线程都拥有自己的变量。

那么选择锁还是ThreadLocal?

看创建实列对象的复用率,如果复用率比较高的话,就使用ThreadLocal。

ThreadLocal的原理

类ThreadLocal的主要作用就是将数据放到当前对象的Map中,这个Map时thread类的实列变量。类ThreadLocal自己不管理、不存储任何的数据,它只是数据和Map之间的桥梁。

执行的流程:数据—>ThreadLocal—>currentThread()—>Map。

执行后每个Map存有自己的数据,Map中的key中存储的就是ThreadLocal对象,value就是存储的值。每个Thread的Map值只对当前的线程可见,其它的线程不可以访问当前线程对象中Map的值。当前的线程被销毁,Map也随之被销毁,Map中的数据如果没有被引用、没有被使用,则随时GC回收。

ThreadLocal常用方法

在这里插入图片描述

set(T):将内容存储到ThreadLocal

get():从线程去私有的变量

remove():从线程中移除私有变量


package ThreadLocalDemo;
import java.text.SimpleDateFormat;

public class ThreadLocalDemo1 {
    private static ThreadLocal<SimpleDateFormat> threadLocal = new ThreadLocal<>();
    public static void main(String[] args) {
        //设置私有变量
        threadLocal.set(new SimpleDateFormat("mm:ss"));
        //得到ThreadLocal
        SimpleDateFormat simpleDateFormat = threadLocal.get();
        //移除
        threadLocal.remove();
    }
}

ThreadLocal的初始化

ThreadLocal提供了两种初始化的方法

initialValue()和

initialValue()初始化:


package ThreadLocalDemo;
import java.text.SimpleDateFormat;
import java.util.Date;

public class ThreadLocalDemo2 {
    //创建并初始化ThreadLocal
    private static ThreadLocal<SimpleDateFormat> threadLocal = new ThreadLocal() {
        @Override
        protected SimpleDateFormat initialValue() {
            System.out.println(Thread.currentThread().getName() + "执行了自己的threadLocal中的初始化方法initialValue()");
            return new SimpleDateFormat("mm:ss");
        }
    };
    public static void main(String[] args) {
        Thread thread1 = new Thread(() -> {
            Date date = new Date(5000);
            System.out.println("thread0格式化时间之后得结果时:" + threadLocal.get().format(date));
        });
        thread1.setName("thread0");
        thread1.start();

        Thread thread2 = new Thread(() -> {
            Date date = new Date(6000);
            System.out.println("thread1格式化时间之后得结果时:" + threadLocal.get().format(date));
        });
        thread2.setName("thread1");
        thread2.start();
    }
}

在这里插入图片描述

withInitial方法初始化:


package ThreadLocalDemo;
import java.util.function.Supplier;

public class ThreadLocalDemo3 {
    private static ThreadLocal<String> stringThreadLocal =
            ThreadLocal.withInitial(new Supplier<String>() {
                @Override
                public String get() {
                    System.out.println("执行了withInitial()方法");
                    return "我是" + Thread.currentThread().getName() + "的ThreadLocal";
                }
            });
    public static void main(String[] args) {
        Thread thread1 = new Thread(() -> {
            System.out.println(stringThreadLocal.get());
        });
        thread1.start();

        Thread thread2 = new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println(stringThreadLocal.get());
            }
        });
        thread2.start();
    }
}

在这里插入图片描述

注意:

ThreadLocal如果使用了set()方法的话,那么它的初始化方法就不会起作用了。

来看:👇


package ThreadLocalDemo;

class Tools {
    public static ThreadLocal t1 = new ThreadLocal();
}
class ThreadA extends Thread {
    @Override
    public void run() {
        for (int i = 0; i < 10; i++) {
            System.out.println("在ThreadA中取值:" + Tools.t1.get());
            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}
public class ThreadLocalDemo4 {
    public static void main(String[] args) throws InterruptedException {
        //main是ThreadA 的 父线程 让main线程set,ThreadA,是get不到的
        if (Tools.t1.get() == null) {
            Tools.t1.set("main父线程的set");
        }
        System.out.println("main get 到了: " + Tools.t1.get());

        Thread.sleep(1000);
        ThreadA a = new ThreadA();
        a.start();
    }
}

在这里插入图片描述

类ThreadLocal不能实现值的继承,那么就可以使用InheritableThreadLocal了👇

InheritableThreadLocal的使用

使用InheritableThreadLocal可以使子线程继承父线程的值

在这里插入图片描述

在来看运行的结果:

在这里插入图片描述

子线程有最新的值,父线程依旧是旧的值


package ThreadLocalDemo;

class ThreadB extends Thread{
    @Override
    public void run() {
        for (int i = 0; i < 10; i++) {
            System.out.println("在ThreadB中取值:" + Tools.t1.get());
            if (i == 5){
                Tools.t1.set("我是ThreadB中新set()");
            }
            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}
public class ThreadLocalDemo5 {
    public static void main(String[] args) throws InterruptedException {
        if (Tools.t1.get() == null) {
            Tools.t1.set("main父线程的set");
        }
        System.out.println("main get 到了: " + Tools.t1.get());

        Thread.sleep(1000);
        ThreadA a = new ThreadA();
        a.start();
        Thread.sleep(5000);
        for (int i = 0; i < 10; i++) {
            System.out.println("main的get是:" + Tools.t1.get());
            Thread.sleep(100);
        }
    }
}

在这里插入图片描述

ThreadLocal的脏读问题来看👇


package ThreadLocalDemo;
import java.util.concurrent.LinkedBlockingDeque;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;

public class ThreadLocalDemo6 {
    private static ThreadLocal<String> threadLocal = new ThreadLocal<>();
    private static class MyThread extends Thread {
        private static boolean flag = false;
        @Override
        public void run() {
            String name = this.getName();
            if (!flag) {
                threadLocal.set(name);
                System.out.println(name + "设置了" + name);
                flag = true;
            }
            System.out.println(name + "得到了" + threadLocal.get());
        }
    }
    public static void main(String[] args) {
        ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(1, 1, 0,
                TimeUnit.MILLISECONDS, new LinkedBlockingDeque<>(10));

        for (int i = 0; i < 2; i++) {
            threadPoolExecutor.execute(new MyThread());
        }
        threadPoolExecutor.shutdown();
    }
}

在这里插入图片描述

发生了脏读:

线程池复用了线程,也复用了这个线程相关的静态属性,就导致了脏读

那么如何避免脏读呢?

去掉static 之后:

在这里插入图片描述

在这里插入图片描述

总结

本篇文章就到这里了,希望对你有些帮助,也希望你可以多多关注编程网的更多内容!

--结束END--

本文标题: 彻底搞懂Java多线程(四)

本文链接: https://lsjlt.com/news/129741.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • 彻底搞懂Java多线程(四)
    目录SimpleDateFormat非线程安全问题ThreadLocalThreadLocal的原理ThreadLocal常用方法ThreadLocal的初始化Inheritable...
    99+
    2024-04-02
  • 彻底搞懂Java多线程(一)
    目录Java多线程线程的创建线程常用方法线程的终止1.自定义实现线程的终止2.使用Thread的interrupted来中断3.Thraed.interrupted()方法和Thre...
    99+
    2024-04-02
  • 彻底搞懂Java多线程(二)
    目录Java中的锁1.synchronized锁(jvm层的解决方案,也叫监视器锁)2.手动锁Locksynchronized锁synchronized使用场景1.使用synchro...
    99+
    2024-04-02
  • 彻底搞懂Java多线程(三)
    目录Java线程池线程池的优点线程池的6种创建方式创建单个线程池的作用是什么?线程池的第七种创建方式ThreadPoolExecutor的执行方式ThreadPoolExecutor...
    99+
    2024-04-02
  • 彻底搞懂Java多线程(五)
    目录单例模式与多线程立即加载/饿汉模式延时加载/懒汉模式饿汉/懒汉对比阻塞队列的实现常见的锁策略乐观锁CASCAS在java中的应用CAS 的ABA问题ABA 问题的解决悲观锁独占锁...
    99+
    2024-04-02
  • Java基础:彻底搞懂java多线程
    目录进程与线程使用多线程的优势线程的状态创建线程线程中断总结进程与线程 进程 进程是操作系统结构的基础,是程序在一个数据集合上运行的过程,是系统进行资源分配和调度的基本单位。进程可...
    99+
    2024-04-02
  • 一文彻底搞懂java多线程和线程池
    目录 什么是线程 一. Java实现线程的三种方式1.1、继承Thread类1.2、实现Runnable接口,并覆写run方法二. Callable接口...
    99+
    2024-04-02
  • 如何彻底搞懂jdk8线程池
    这篇文章将为大家详细讲解有关如何彻底搞懂jdk8线程池,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。顶层设计,定义执行接口Interface Executor(){ &n...
    99+
    2023-06-25
  • 一篇文章彻底搞懂jdk8线程池
    这可能是最简短的线程池分析文章了。 顶层设计,定义执行接口 Interface Executor(){ void execute(Runnable command); ...
    99+
    2024-04-02
  • 彻底搞懂 javascript的Promise
    目录一、为什么要引入PromisePromise解决了什么问题?Promise有哪些具体的使用场景?二、手写Prromise身上的方法手写Promise.all手写Promise.r...
    99+
    2024-04-02
  • 彻底搞懂 Python 编码
    因为中文的特殊编码,导致 Python2 和 Python3 使用过程中的各种编码问题,如果不清楚其中的关联关系,那么这就一直是个大坑,不是懵逼就还是懵逼,所以就目前碰到的情况彻底梳理下 Python2 和 Python3 中编码的关系和区...
    99+
    2023-01-31
    Python
  • 彻底搞懂java并发ThreadPoolExecutor使用
    目录前言正文一. 线程池的简单原理二. 线程池的创建三. 线程池执行任务1. 执行无返回值任务2. 执行有返回值任务3. 执行有返回值任务时抛出错误4. ThreadPoolExec...
    99+
    2023-02-28
    java并发ThreadPoolExecutor java 并发
  • 一篇搞懂Java多线程
    Java多线程是指在一个程序中可以同时执行多个线程,每个线程执行不同的任务。多线程可以提高程序的并发性和效率。Java多线程的实现有...
    99+
    2023-09-14
    Java
  • 一文彻底搞懂Kotlin中的协程
    产生背景 为了解决异步线程产生的回调地狱 //传统回调方式 api.login(phone,psd).enquene(new Callback<User>(){ ...
    99+
    2024-04-02
  • 一文彻底搞懂IO底层原理
    目录一、混乱的 IO 概念二、用户空间和内核空间三、IO模型3.1、BIO(Blocking IO)3.2、“C10K”问题3.3、NIO非阻塞模型3.4、IO多路复用模型3.4.1...
    99+
    2024-04-02
  • 彻底搞懂 PHP 运算符 ?: 和 ??
    文章目录 快速掌握: 短三元运算符 NULL 合并运算符 附上官方文档查阅方式 快速掌握 : 短三元运算符 : 称之为短三元运算符,它是我们熟悉的三元运算符(也叫做条件运算符)的一种特殊写法,也就是省略了三元运算符中间的部分...
    99+
    2023-08-30
    php
  • 一文带你彻底搞懂Vuex
    大家可以思考一下,组件之间的传值有哪些?有父子通讯,兄弟组件通讯......但是传参对于多层嵌套就显得非常繁琐,代码维护也会非常麻烦。因此vuex就是把组件共享状态抽取出来以一个全局单例模式管理,把共享的数据函数放进vuex中,任何组件都可...
    99+
    2022-11-22
    Vue vue3 vue.js VueX
  • 彻底搞懂MySQL存储过程和函数
    目录1.0  创建存储过程和函数1. 创建存储过程2. 创建存储函数2|0变量1. 定义变量2. 变量赋值3|0定义条件和处理程序1. 定义条件2. 定义处理程序4|0光标...
    99+
    2024-04-02
  • 一文彻底搞懂PHP进程信号处理
    本篇文章给大家带来了关于PHP的相关知识,其中主要详细介绍了PHP 进程信号处理,感兴趣的朋友下面一起来看一下吧,希望对大家有帮助。背景前两周老大给我安排了一个任务,写一个监听信号的包。因为我司的项目是运行在容器里边的,每次上线,需要重新打...
    99+
    2023-05-14
    进程 PHP
  • Java基础之让你彻底搞懂代理模式
    目录一、代理模式二、静态代理三、动态代理四、总结一、代理模式 什么是代理模式? 先来生活常用例子:你想买票,你没必要去车站买;而是可以去一个代售点,代售点代理车站卖票,这就是一个简单...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作