多线程下一些点

最近看了一下java中多线程并发同步问题。

一、对于开发中做键值对的操作我们常用的Map来存储对应信息。

Map map = new HasMap();

Map map = new HasTable<>();

Array以及Linked一样,只是表示不同数据存储结构。

对于HashMap和HashTable来说,主要区别点在于:

1、HashMap是线程不安全的,而HashTable是采用synchronized来进行整体的锁同步的。

2、HashMap的key和value都可以是null,而HashTable不接受这种操作。

3、iteator的区别

故:对于单线程操作来说不存在数据同步的问题,使用HashMap优于HashTable,HashTable的synchronized锁会导致性能较低。而对于多线程操作Map的case来说,数据的同步性是最重要的,HashTable可以满足需求。

二、

HashTable虽然可以保证数据的同步性,但是HashTable是通过synchronized锁方式来实现的,而一个HashTable是一把锁来控制的,



如上就是HashTable的实现方式,全局采用一个数据结构来保存数据并对操作数据的方法进行synchronized同步,这样虽然保证了数据的同步性,但是效率较低,尤其在多线程中一个线程获取到锁,其他线程即使操作其他方法也需要等待,在代码实现有问题的某些情况下甚至会导致cpu占用100%,死锁的产生。

在java5以上建议使用ConcurrentHashMap来替代HashTable进行多线程时的操作。

ConcurrentHashMap将数据进行再哈希,把数据分段,每一段数据有自己的锁,这样不同数据分段间的数据对于锁的获取释放互不影响。


三、java内存模型与volitale关键字

对于内存模型来说,在程序工作时,cpu负责计算,存储负责数据。存储有处理器的L1\L2 cache memory。

对于线程来说,分为堆内存与栈内存,堆内存是进程所持有的共有的,每个线程的数据使用路线为:

堆内存----->copy到私有内存引用副本------>操作--------->未来时机刷新堆内存

而在具体的操作需要关注的三个点:

1、原子性

如i=10是原子操作,i=j,i++不是原子操作,简单来说,原子操作是要么一次性执行完,要么不执行,不存在执行中被调度或其他中断。

一般来说,i=10只需直接刷新内存中值即可,一步完成,而对于i++这样的操作,需要读取,自增、刷新内存三步,所以不是原子的。

对于Fragment以及转账等业务一般是通过开启事务来保证操作的整体原子性。

目前java也提供了一些封装的保证基本类型各种操作的整体原子性的类,如AtomicInteger等。

2、可见性,

可见性是指一个线程对于变量的操作应该对于其他线程可见,避免数据的不同步,volitale就是保证可见性的,一个被volitale修饰的变量,如果在一个线程中修改,会强制立即刷新到内存,且其他线程已读取并缓存在私有内存中的对应变量值会失效,其他线程的后续读取会无法命中,重新来公共内存加载读取。

3、有序性

对于代码来说,书写的顺序并不等同于在编译器或处理器中真正执行的顺序,编译器会对生成的代码进行优化重排,但是重排的前提是保证最终的执行结果不变!

这种优化重排执行顺序的处理在单线程中是OK的,最终的结果不变。但是对于多线程来说,线程的调度无法保证各个线程细化到具体代码行的执行顺序。这样如果线程间有数据依赖,就会导致结果不可预期。

volatile的作用并不会阻止代码重排,但是volatile就像一个分界线,重排只会发生在volatile的前后代码块儿内部,不可能越过volitale进行重排。


看起来volatile如同synchronized一样既保证数据的立即可见性也保证执行的一定有序性。

但是在多线程中还是要谨慎使用volitale,因为这个关键字无法保证变量操作的原子性。

如:

volitale int i=0;

i++;

两个线程都进行这种操作,如果线程1已经命中成功读取i的值,此时线程中断,此时没有改变值,所以并不会刷新内存。

线程2执行操作后, i变为1,此时刷新内存,但是线程1已经执行完读取的操作,所以即使线程私有内存失效也无法影响cpu操作的值。最终结果还是1。

当然这些case是很极端的,但是要充分理解操作的原子性。真正的执行过程可能在硬件上可能分为很多步骤。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351

推荐阅读更多精彩内容