Java并发编程学习路线(转)

时间:2021-12-19 15:17:04

以前特地学过并发编程,但是没怎么学进去,不太喜欢。最近发现,作为一个资深工程师,却没有完整深入系统的学习过,而反是现在的BAT大并发是必须的,感觉甚是惭愧。

故找了一片学习文章,如下,准备集中一段时间来学习并发编程技术

--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- 

一年前由于工作需要从微软技术栈入坑Java,并陆陆续续做了一个Java后台项目,目前在搞Scala+Java混合的后台开发,一直觉得并发编程是所有后台工程师的基本功,所以也学习了小一年Java的并发工具,对整体的并发理解乃至分布式都有一定的提高,所以想和大家分享一下。

我的学习路线

首先说说学习路线,我一开始是直接上手JCIP(Java Concurrency in Practice),发现不是很好懂,把握不了那本书的主线,所以思索着从国内的作者开始先,所以便读了下方腾飞的《Java并发编程的艺术》的,虽然豆瓣上的评价一般,但是对于构建Java并发的整体映像还是有所提高的,至少我知道了有哪些东西要深入学习。接着我想加强下并发的理论,继续读了The Art of Multiprocessor Programming,这本书比较艰涩,不是很好懂,但是过一遍还是好处多多,建议初学者了解下概念的过过,后期可以再来翻看。有了以上两步的支持,接下来就又开始啃JCIP了,发现比以前有了不同的感觉,我能比较轻松的跟上书的脉络,知道书的整体框架,读起来不那么费劲了,这本书号称Java并发编程的圣经,确实可以看出作者有很丰富的并发实践经验。再后来我过了一遍Oracle官网上的Java Tutorial关于并发的那一章,发现讲的也不错,对于了解基础库有哪些组件帮助挺大。

到了这一步,接下来怎么继续提高呢?我发现了一本很有趣的书,《七周七并发模型》,之前的视野一直是在Java并发编程的工具包中深入了解,感觉,应该跳出来,从模型的角度看看各个语言的并发实现的原理,我目前正处于这一步,发现很有意思,第一章讲Java的线程和锁这个模型就感觉很精髓,只用了小三章把Java整体的脉络过了一遍,强烈推荐用来复习。

下一步我的计划是jdk的concurrent包以及Java specification的并发部分,并发理解,除了基础概念,就是要深刻领会各个应用场景下,有无并发问题以及如何写出线程安全的代码,个人觉得学习下无锁的实现对理解有一定的帮助,但不用太费心思,到了Java Memory Model这一层基本就够用了。

根据上面的阐述,我的路线图可以总结如下:

学习心得 -- Java并发包的基础概念

了解Java并发包有哪些工具以及相关基础概念,有Java tutorial的concurrent章节和JCIP一书就足够了。

JCIP一书的整体脉络如下:

  1. 介绍多线程的利弊;
  2. 解释线程安全是什么以及如何获得线程安全;
  3. 从高频的使用场景出发,介绍对象传递,类的设计等如何获得线程安全;
  4. 从Java并发包出发,介绍高层的并发组件有啥以及相关原理;
  5. 介绍并发的一些弊端以及如何避免;
  6. 从Java并发包出发,介绍底层的并发组件以及原理;

总体看,该书有两条主线,1 从高到低介绍Java并发包的一些重要组件和原理; 2 从并发场景出发,介绍如何利用这些组件来获得线程安全。其中第二部分是这本书最大的特色,也是书名中有Practice的原因。

书中提到了几个比较有意思的地方,

首先,到底什么是线程安全?

A class is thread-safe if it behaves correctly when accessed from multiple threads, regardless of the scheduling or interleaving of the execution of those threads by the runtime environment, and with no additional synchronization or other coordination on the part of the calling code.
这个定义中,作者强调了正确地被多线程访问, 同时要求没有外加其他同步的手段。

那么,如何获得线程安全?
Writing thread-safe code is, at its core, about managing access to state, and in particular to shared, mutable state.
书中将获得线程安全总结为维护代码的状态,如果一个类是无状态的(immutable),则自带线程安全的属性(函数式编程便是通过这种方式达到自带的线程安全)。这些状态大致可以理解为类中的非常量变量。 
通过这个可以了解到线程安全的本质,其实是共享变量,也就是状态,有状态的多线程访问就需要同步机制来保证线程安全。

如何理解Java提供的用于处理并发的组件?
JDK提供的并发组件,大致可以分为两类, 一类是预防为主,防止错误发生(race condition, visibility),大部分组件都是这类,还有一类是发生了错误但是能够知道并及时重试(Atomic类提供的CAS),形象的例子有如 十字路口的信号灯,在流量小的时候,采用过多的预防措施反而会适得其反,例如白白的在大部分时间都没有车的道路上等红灯,这个时候适合采用犯错(例如去掉红绿灯,让车*行驶,遇到其他车的时候互相让位即可)后解决的方法,能够获得最大的效率,在流量大的时候,红绿灯的作用就能够凸显出来,其实规则的制定一定是在规模较大的时候才有意义,这也是预防的初衷。
类比到并发领域就是,在线程数量大,采用预防的措施比较好,这样大部分线程就不会因为概率小的CAS重试浪费大量的cpu周期,在线程数量小的时候,CAS的意义就比较大,因为预防措施带来的线程切换等的开销可能大于CAS的等待,而且较少的线程也会让CAS重试的等待时间变少。

以下是我根据这两个资料概括出来的基础概念,
Java并发编程学习路线(转)

理解这些基础概念的核心,我觉得其实就是解决两点问题:

  • Thread Interleaving,即多个线程读写共享变量造成的不一致问题;
  • Visibility,为了提高性能,处理器的每个执行单元其实都有缓存,这个虽然提高了某些数据的访问性能但是却给并发编程带来了数据读取的不一致性问题;

当然要更深入理解并发,还需要知道如何提升并发的性能,例如锁的粒度如何把握?(经典的例子可以JDK的ConcurrentHashMap),底层一点的知识也得了解,例如CAS和Java Memory Model。

从高维视角了解并发

有了Java并发的基础知识,接下来很适合阅读七周七并发,我目前就在读七周七并发,发现站在多种语言从范式的角度了解并发很有意思,原来Java提供的线程和锁的机制其实相当于比较原始的工具了,其离底层最近。最近接触了Scala,其使用了AKKA,则是一种高层的并发抽象。

七周七并发试图从历史的角度阐述作为锁和线程的代表之Java的并发包的进化历程,首先最早加入JDK的,其实是synchronized及其statement,但是发现缺少相关timeout和不能中断等等功能,加入了可重入锁,读写锁等等,再后来又加入了各种线程安全的数据结构和高级同步机制。

接下来,七周七并发从函数式编程等等各种范式的角度阐述,除了线程和锁,还有很多其他高层抽象可以更加方便的编写并发代码。

这本书对于充分理解并发,拓宽视野很有帮助,推荐大家阅读。

从实现角度透彻理解并发

再深入下去的话,没有比经典的JDK更合适的了,当然Google的Guava包也值得学习,从这些经典代码了解各种组件的实现可以加深理解并更好的使用它们,但是作为应用端的程序员,倒是并不需要写出这种较为底层的代码(无锁化)。

回顾这小一年的学习曲线,收获良多,不过最后最值得强调的一点其实是,在做技术选择的时候,并发只是工具箱中的一种手段,学习它只是为了能够灵活运用,设计的首要选择依然是在当时情境下的最简化,能不用并发就不要用。

Java并发编程学习路线(转)的更多相关文章

  1. Java并发编程学习路线

    一年前由于工作需要从微软技术栈入坑Java,并陆陆续续做了一个Java后台项目,目前在搞Scala+Java混合的后台开发,一直觉得并发编程是所有后台工程师的基本功,所以也学习了小一年Java的并发工 ...

  2. Java并发编程学习前期知识下篇

    Java并发编程学习前期知识下篇 通过上一篇<Java并发编程学习前期知识上篇>我们知道了在Java并发中的可见性是什么?volatile的定义以及JMM的定义.我们先来看看几个大厂真实的 ...

  3. 学习笔记:java并发编程学习之初识Concurrent

    一.初识Concurrent 第一次看见concurrent的使用是在同事写的一个抽取系统代码里,当时这部分代码没有完成,有许多的问题,另一个同事接手了这部分代码的功能开发,由于他没有多线程开发的经验 ...

  4. Java并发编程学习笔记

    Java编程思想,并发编程学习笔记. 一.基本的线程机制 1.定义任务:Runnable接口 线程可以驱动任务,因此需要一种描述任务的方式,这可以由Runnable接口来提供.要想定义任务,只需实现R ...

  5. Java并发指南开篇:Java并发编程学习大纲

    Java并发编程一直是Java程序员必须懂但又是很难懂的技术内容. 这里不仅仅是指使用简单的多线程编程,或者使用juc的某个类.当然这些都是并发编程的基本知识,除了使用这些工具以外,Java并发编程中 ...

  6. Java并发编程学习:volatile关键字解析

    转载:https://www.cnblogs.com/dolphin0520/p/3920373.html 写的非常棒,好东西要分享一下 Java并发编程:volatile关键字解析 volatile ...

  7. &lbrack;Todo&rsqb; Java并发编程学习

    有两个系列的博文,交替着可以看看: 1. Java并发编程与技术内幕 http://blog.csdn.net/Evankaka/article/details/51866242 2. [Java并发 ...

  8. java并发编程学习:用 Semaphore (信号量)控制并发资源

    并发编程这方面以前关注得比较少,恶补一下,推荐一个好的网站:并发编程网 - ifeve.com,上面全是各种大牛原创或编译的并发编程文章. 今天先来学习Semaphore(信号量),字面上看,根本不知 ...

  9. java并发编程学习&colon; ThreadLocal使用及原理

    多线程应用中,如果希望一个变量隔离在某个线程内,即:该变量只能由某个线程本身可见,其它线程无法访问,那么ThreadLocal可以很方便的帮你做到这一点. 先来看一下示例: package yjmyz ...

随机推荐

  1. 基于Nginx dyups模块的站点动态上下线并实现简单服务治理

    简介 今天主要讨论一下,对于分布式服务,站点如何平滑的上下线问题. 分布式服务 在分布式服务下,我们会用nginx做负载均衡, 业务站点访问某服务站点的时候, 统一走nginx, 然后nginx根据一 ...

  2. JQuery中对各种域进行隐藏和显示操作

    操作的基本步骤: (1)导入jquery相关jar <script type="text/javascript" src="jquery-1.1.3.pack.js ...

  3. Lintcode&colon; Nth to Last Node in List

    Find the nth to last element of a singly linked list. The minimum number of nodes in list is n. Exam ...

  4. JS 操作 radio input(cc问卷管理)

    1.选中特定的单选按钮 function showDetail(content){ $("input[name^='radio']").removeAttr("check ...

  5. python 学习资料

    Python是一种面向对象.直译式计算机程序设计语言.它的语法简捷和清晰,尽量使用无异义的英语单词,与其它大多数程序设计语言使用大括号不一样,它使用縮进来定义语句块.与Scheme.Ruby.Perl ...

  6. CodeForces 22B Bargaining Table 简单DP

    题目很好理解,问你的是在所给的图中周长最长的矩形是多长嗯用坐标(x1, y1, x2, y2)表示一个矩形,暴力图中所有矩形易得递推式:(x1, y1, x2, y2)为矩形的充要条件为: (x1, ...

  7. laravel无法显示路由界面

    安装完laravel项目后,开启了重写,/app/storage也设置好了权限,但是始终无法显示出页面,并出现: "Whoops, looks like something went wro ...

  8. 数字信号处理MATLAB简单序列

    数字信号处理应用的几个基本序列: 1 单位样本序列 function mainImseq() clc clear disp('生成抽样序列'); y=imseq(,,); %调用样本函数,此时序列下标 ...

  9. MSIL实用指南-创建方法和定义参数

    本篇讲解实现创建方法.指定参数的名称.实现参数加out和ref修饰符.以及参数加默认值. 创建方法 创建方法用类TypeAttributes的 DefineMethod(string name, Me ...

  10. Codeforces Round &num;417 &lpar;Div&period; 2&rpar;-A&period; Sagheer and Crossroad

    [题意概述] 在一个十字路口 ,给定红绿灯的情况, 按逆时针方向一次给出各个路口的左转,直行,右转,以及行人车道,判断汽车是否有可能撞到行人 [题目分析] 需要在逻辑上清晰,只需要把所有情况列出来即可 ...