有关线程安全的探讨--final、static、单例、线程安全
- 什么样的代码是天生线程安全的?而不用加锁
- 线程是否安全的本质是什么?
- 什么是快速把一段代码变成线程安全的通用方法
- final static 单例 线程安全 之间的关系
- 这句话严格一些说可以是这样:线程使用在run( )方法中实例化的局部变量的方法,是线程安全的
- 继承并重写一个Thread类,然后在使用的时候实例化这个类,最后调用这个对象实例的start方法启动
- 这种方式的run方法中,其实能调用的东西就很少了
- 你在继承时加的成员变量。(完全不会有线程是否安全的问题,因为这个类就一个run()方法是多线程方法,就跟在run()中实例化的局部变量一样)
- 通过构造方法从外面传入的变量。(这种方式需要警惕!因为传递的是引用,如果你在线程中对这个引用指向的内容进行修改,那么会影响到原来的东西!)
- 使用其他的代码段(方法)
- 静态方法(类似单例模式)
- 实例方法——通过实例对象
- 使用其他的对象
- 静态对象
- 实例对象
- (两面两大点中,使用实例方法和实例对象都是线程安全的。而使用静态方法和静态对象时,是一定会冲突的)
- 所以总结一下,这种方式中
- 线程安全的有
- 在继承时加的成员变量
- 实例化其他对象,使用这个对象,或者使用这个对象的方法
- 不安全的有
- 通过构造方法从外面传入的变量
- 静态方法
- 静态对象
- 线程安全的有
- 这种方式的run方法中,其实能调用的东西就很少了
- 使用匿名内部类
- 这种方式,在上一种方式的继承上,只少了构造方法的方式,然后多了好几种危险的方式, 需要注意
- 所处方法中的局部变量
- 这个值得一提,本来这项是肯定会线程不安全,而且非常常用,所以危险指数五颗星的,但是JAVA特地为此限定了一条规则,就是这样的局部变量必须是final的,不能修改,于是这个就变得非常安全了
- 但这条其实可以通过引用类型绕过,就是另一回事了,其实也说明了它的不安全
- 所处类中的属性
- 所处类中的方法
- 所处方法中的局部变量
- 这种方式,在上一种方式的继承上,只少了构造方法的方式,然后多了好几种危险的方式, 需要注意
- 方法本身不会有问题,问题的根源是(普通方法、静态方法)方法使用了变量(相对全局变量,或者说叫可共享变量)【比如静态成员、类属性等等】
- 匿名类中更加危险,要谨慎调用
- 不能同时被多个线程调用
- 这个是最普通的,也是常规上我们的线程安全的含义
- 这个问题可以通过加锁解决
- 不能被多个线程调用(不同时也不行)
- 这个在第一类的程度上有所增加,不是常用的情况,可能你不仅是要使用变量,你还需要记录变量的值
- 这个问题一般是把相关变量变成ThreadLocal的
- 不能被超过一次地调用
- 这个的情况更加特殊
- 一般使用单例模式解决
- final
- 意思是,这个对象的值(基本类型就是值,引用类型是引用地址),不会再被改变
- 与线程安全的关系,如上文,一定程度上能使某些变量强制变得线程安全
- static
- 意思是,这个对象是一个全局变量了,你可以在多个地方,多个线程中调用到它,而且调用的是同一个它
- 与线程安全的关系,一般这种的变量很容易造成线程不安全的情况
- 单例
- 这首先是一种特殊的需求,就是某个类的实例在JVM中只能存在一个,跟前面的static,线程安全都不一样
- 与线程安全的关系。实现单例需要考虑复杂的多线程的情况,这个东西需要线程安全
- 因为创建一个 SimpleDateFormat实例的开销比较昂贵,解析字符串时间时频繁创建生命周期短暂的实例导致性能低下
- 在程序中我们应当尽量少的创建SimpleDateFormat 实例,因为创建这么一个实例需要耗费很大的代价。在一个读取数据库数据导出到excel文件的例子当中,每次处理一个时间信息的时候,就需要创建一个SimpleDateFormat实例对象,然后再丢弃这个对象。大量的对象就这样被创建出来,占用大量的内存和 jvm空间。
- 于是,就很容易想到,将 SimpleDateFormat定义为静态类变量,貌似能解决这个问题
- 于是这就引出了,SimpleDateFormat是非线程安全的,这样的使用方式可能引发并发线程安全问题
- SimpleDateFormat类内部有一个Calendar对象引用,它用来储存和这个SimpleDateFormat对象(叫sdf)相关的日期信息,例如sdf.parse(dateStr), sdf.format(date)
- 诸如此类的方法参数传入的日期相关String, Date等等, 都是交友Calendar引用来储存的
- 这样就会导致一个问题:如果你的sdf是个static的, 那么多个thread 之间就会共享这个sdf, 同时也是共享这个Calendar引用, 并且, 观察 sdf.parse() 方法,你会发现有如下的调用:
- Date parse() {
- calendar.clear(); // 清理calendar
- ... // 执行一些操作, 设置 calendar 的日期什么的
- calendar.getTime(); // 获取calendar的时间
- }
- 这里会导致的问题就是:如果 线程A 调用了 sdf.parse(), 并且进行了 calendar.clear()后还未执行calendar.getTime()的时候,线程B又调用了sdf.parse(), 这时候线程B也执行了sdf.clear()方法, 这样就导致线程A的的calendar数据被清空了(实际上A,B的同时被清空了). 又或者当 A 执行了calendar.clear() 后被挂起, 这时候B 开始调用sdf.parse()并顺利i结束, 这样 A 的 calendar内存储的的date 变成了后来B设置的calendar的date
- 你肯定是喜欢使用匿名内部类的,以这个为基础
- 注意如果是调用所在方法中的局部变量,尽量不要绕过final机制,如果需要绕过,而且会对这个局部变量进行修改的话,那一定是知道不会多个这样的线程同时运行(比如作为UI主线程外的一个子线程,这个子线程只会有一个)
- 不要尝试修改不是在自己内部实例化出的对象的值(只能改局部变量的值)(尽量使用局部变量)
- 你还喜欢使用静态工具方法,所有的静态工具方法中使用变量尽量使用局部变量(for循环中的i++ 是没有问题的),尽量少地使用静态变量,更不要尝试对静态变量的值进行修改
有关线程安全的探讨--final、static、单例、线程安全的更多相关文章
- 探讨一下Java单例设计模式
所谓单例模式,简单来说,就是在整个应用中保证只有一个类的实例存在.就像是Java Web中的application,也就是提供了一个全局变量,用处相当广泛,比如保存全局数据,实现全局性的操作等. 1. ...
- Java线程和多线程(五)——单例类中的线程安全
单例模式是最广泛使用的创建模式之一.在现实世界之中,诸如Databae的连接或者是企业信息系统(EIS)等,通常其创建都是受到限制的,应该尽量复用已存在对象而不是频繁创建销毁.为了达到这个目的,开发者 ...
- Android单例线程池
package com.jredu.schooltong.manager; import java.util.concurrent.ExecutorService;import java.util.c ...
- Spring中构造器、init-method、@PostConstruct、afterPropertiesSet孰先孰后,自动注入发生时间以及单例多例的区别、SSH线程安全问题
首先明白,spring的IOC功能需要是利用反射原理,反射获取类的无参构造方法创建对象,如果一个类没有无参的构造方法spring是不会创建对象的.在这里需要提醒一下,如果我们在class中没有显示的声 ...
- static实现单例的隐患
1. 前言 Java的单例有多种实现方式:单线程下的简单版本.无法在指令重排序下正常工作的Double-Check.static.内部类+static.枚举--.这篇文章要讨论的,是在使用static ...
- Swift百万线程攻破单例(Singleton)模式
一.不安全的单例实现 在上一篇文章我们给出了单例的设计模式,直接给出了线程安全的实现方法.单例的实现有多种方法,如下面: class SwiftSingleton { class var shared ...
- Java设计模式之单例
一.Java中的单例: 特点: ① 单例类只有一个实例 ② 单例类必须自己创建自己唯一实例 ③ 单例类必须给所有其他对象提供这一实例 二.两种模式: ①懒汉式单例<线程不安全> 在类加载时 ...
- 创建类模式(五):单例(Singleton)
定义 确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例. 单例模式一般情况下通过使用private的构造函数确保了在一个应用中只产生一个实例,并且是自行实例化. 和静态变量的区别 虽然 ...
- java中你确定用对单例了吗?
作为程序员这样的特殊物种来说,都掌握了一种特殊能力就是编程思想,逻辑比較慎重,可是有时候总会忽略到一些细节,比方我,一直以来总认为Singleton是设计模式里最简单的,不用太在意,然而就是由于这样的 ...
随机推荐
- sql语句建表,并且自增加主键
sql语句建表,并且自增加主键 use [test] CREATE TABLE [dbo].[Table_4] ( [userid] [int] IDENTITY(1,1) NOT NULL, CON ...
- EGit系列第一篇——创建本地仓库
首先,用Eclipse创建一个项目,然后右键项目Team\Share Preject... 然后出来Share Preject对话框,选择git 点击下一步配置Git仓库,通常在项目本地目录创建仓库, ...
- 【Linux】 centos 7 启用端口
网上的大部分资料都是用iptables防火墙的,但是阿里云的centos 7默认防火墙是firewall.最为简单的方法其实就是关闭我们的防火墙: 1 查看下防火墙的状态: systemctl st ...
- Intellij 部署项目java.lang.ClassNotFoundException: org.springframework.web.context.ContextLoaderListener
报错信息: org.apache.catalina.core.StandardContext.listenerStart Error configuring application listener ...
- 求组合数 C++程序
一 递归求组合数 设函数为void comb(int m,int k)为找出从自然数1.2.... .m中任取k个数的所有组合. 分析:当组合的第一个数字选定时,其后的数字是从余下的m-1个数中 ...
- update select 多字段
update Countrys set ( Abbreviation_cn, Abbreviation_en, Two_code,Three_code, Number_code)= (select [ ...
- 阅读代码工具:Visual Studio Code
打开一个文件夹,直接阅读,体验还不错 版本: 1.25.1提交: 1dfc5e557209371715f655691b1235b6b26a06be日期: 2018-07-11T15:43:11.471 ...
- 使用spring提供的ReflectionUtils简化项目中反射代码的复杂性
在项目中有时候我们会使用到反射的功能,如果使用最原始的方法来开发反射的功能的话肯能会比较复杂,需要处理一大堆异常以及访问权限等问题.spring中提供了ReflectionUtils 这个反射的工具类 ...
- 号称简明实用的django上手教程
1 几个基本概念 前置条件:假设读者基本Python语言基础,或者具备某种编程语言的基础.你还熟悉web开发环境,懂些css,js,db等. Django是什么? Django是一个开放源代码的Web ...
- linux 命令行常用快捷键
linux命令行常用快捷键,区别于vim编辑器快捷键.熟练掌握下面的快捷键可提高操作linux的工作效率.当然最重要的是可以装屌. 1.移动光标快捷键Ctrl+a光标回到命令行首* Ctrl+e光标回 ...