`
stephen80
  • 浏览: 106170 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

java concurrency 笔记

    博客分类:
  • j2se
阅读更多
重读:<java concurrency in practice> ,再次获益。决定买本书。 

顺序:
         1. immutable
         2. confined :method ,object ,thread
       
          3. concurrent utils
         4. lock


备忘:并准备逐条核实。

java并发编程实践笔记

1, 保证线程安全的三种方法:
    a, 不要跨线程访问共享变量
    b, 使共享变量是final类型的
    c, 将共享变量的操作加上同步

2, 一开始就将类设计成线程安全的, 比在后期重新修复它,更容易.

3, 编写多线程程序, 首先保证它是正确的, 其次再考虑性能.

4, 无状态或只读对象永远是线程安全的.

5, 不要将一个共享变量裸露在多线程环境下(无同步或不可变性保护)

6, 多线程环境下的延迟加载需要同步的保护, 因为延迟加载会造成对象重复实例化

7, 对于volatile声明的数值类型变量进行运算, 往往是不安全的(volatile只能保证可见性,不能保证原子性).
详见volatile原理与技巧中, 脏数据问题讨论.

8, 当一个线程请求获得它自己占有的锁时(同一把锁的嵌套使用), 我们称该锁为可重入锁.
在jdk1.5并发包中, 提供了可重入锁的java实现-ReentrantLock.

9, 每个共享变量,都应该由一个唯一确定的锁保护.
创建与变量相同数目的ReentrantLock, 使他们负责每个变量的线程安全.

10,虽然缩小同步块的范围, 可以提升系统性能.
但在保证原子性的情况下, 不可将原子操作分解成多个synchronized块.

11, 在没有同步的情况下, 编译器与处理器运行时的指令执行顺序可能完全出乎意料.
原因是, 编译器或处理器为了优化自身执行效率, 而对指令进行了的重排序(reordering).

12, 当一个线程在没有同步的情况下读取变量, 它可能会得到一个过期值, 但是至少它可以看到那个
线程在当时设定的一个真实数值. 而不是凭空而来的值. 这种安全保证, 称之为最低限的安全性(out-of-thin-air safety)

在开发并发应用程序时, 有时为了大幅度提高系统的吞吐量与性能, 会采用这种无保障的做法.
但是针对, 数值的运算, 仍旧是被否决的.

13, volatile变量,只能保证可见性, 无法保证原子性.
详见 volatile原理与技巧

14, 某些耗时较长的网络操作或IO, 确保执行时, 不要占有锁.

15, 发布(publish)对象, 指的是使它能够被当前范围之外的代码所使用.(引用传递)
对象逸出(escape), 指的是一个对象在尚未准备好时将它发布.

原则: 为防止逸出, 对象必须要被完全构造完后, 才可以被发布(最好的解决方式是采用同步)

this关键字引用对象逸出
例子: 在构造函数中, 开启线程, 并将自身对象this传入线程, 造成引用传递.
而此时, 构造函数尚未执行完, 就会发生对象逸出了.

16, 必要时, 使用ThreadLocal变量确保线程封闭性(封闭线程往往是比较安全的, 但一定程度上会造成性能损耗)
封闭对象的例子在实际使用过程中, 比较常见, 例如 hibernate openSessionInView机制, jdbc的connection机制.

17, 单一不可变对象往往是线程安全的(复杂不可变对象需要保证其内部成员变量也是不可变的)
良好的多线程编程习惯是: 将所有的域都声明为final, 除非它们是可变的

18, 保证共享变量的发布是安全的
    a, 通过静态初始化器初始化对象(jls 12.4.2叙述, jvm会保证静态初始化变量是同步的)
    b, 将对象申明为volatile或使用AtomicReference
    c, 保证对象是不可变的
    d, 将引用或可变操作都由锁来保护

19, 设计线程安全的类, 应该包括的基本要素:
    a, 确定哪些是可变共享变量
    b, 确定哪些是不可变的变量
    c, 指定一个管理并发访问对象状态的策略

20, 将数据封装在对象内部, 并保证对数据的访问是原子的.
建议采用volatile javabean模型或者构造同步的getter,setter.

21, 线程限制性使构造线程安全的类变得更容易, 因为类的状态被限制后, 分析它的线程安全性时, 就不必检查完整的程序.

22, 编写并发程序, 需要更全的注释, 更完整的文档说明.

23, 在需要细分锁的分配时, 使用java监视器模式好于使用自身对象的监视器锁.
前者的灵活性更好.

Object target = new Object();
// 这里使用外部对象来作为监视器, 而非this
synchronized(target) {
    // TODO
}

针对java monitor pattern, 实际上ReentrantLock的实现更易于并发编程.
功能上, 也更强大.

24, 设计并发程序时, 在保证伸缩性与性能折中的前提下, 优先考虑将共享变量委托给线程安全的类.
由它来控制全局的并发访问.

25, 使用普通同步容器(Vector, Hashtable)的迭代器, 需要外部锁来保证其原子性.
原因是, 普通同步容器产生的迭代器是非线程安全的.

26, 在并发编程中, 需要容器支持的时候, 优先考虑使用jdk并发容器
(ConcurrentHashMap, ConcurrentLinkedQueue, CopyOnWriteArrayList...).

27, ConcurrentHashMap, CopyOnWriteArrayList
并发容器的迭代器,以及全范围的size(), isEmpty() 都表现出弱一致性.
他们只能标示容器当时的一个数据状态. 无法完整响应容器之后的变化和修改.

28, 使用有界队列, 在队列充满或为空时, 阻塞所有的读与写操作. (实现生产-消费的良好方案)
BlockQueue下的实现有LinkedBlockingQueue与ArrayBlockingQueue, 前者为链表, 可变操作频繁优先考虑,后者为数组, 读取操作频繁优先考虑.
PriorityBlockingQueue是一个按优先级顺序排列的阻塞队列, 它可以对所有置入的元素进行排序(实现Comparator接口)

29, 当一个方法, 能抛出InterruptedException, 则意味着, 这个方法是一个可阻塞的方法, 如果它被中断, 将提前结束阻塞状态.
当你调用一个阻塞方法, 也就意味着, 本身也称为了一个阻塞方法, 因为你必须等待阻塞方法返回.

如果阻塞方法抛出了中断异常, 我们需要做的是, 将其往上层抛, 除非当前已经是需要捕获异常的层次.
如果当前方法, 不能抛出InterruptedException, 可以使用Thread.currentThread.interrupt()方法, 手动进行中断.
分享到:
评论

相关推荐

    Java Concurrency In Practice Learning Note

    本笔记将深入探讨《Java Concurrency In Practice》这本书中的核心概念,结合Guava库的实际使用案例,帮助读者理解并掌握Java并发编程的精髓。 首先,我们来了解Java并发的基础知识。Java提供了丰富的并发工具类,...

    《Java Concurrency in Practice》中文版笔记.docx

    《Java Concurrency in Practice》中文版笔记

    《Java并发编程之美》学习笔记 《Java concurrency programming》learning note.zip

    《Java并发编程之美》学习笔记 《Java concurrency programming》learning note

    Java重要的基础知识/高级特性/工作经验170页笔记

    精心制作的170页Java学习笔记 这本笔记要讨论的主题范围大概有: 1 . JMM: Java Concurrency Java运行的基础之一 2 . Java Concurrency: Java Java最重要的高级特性之一 3 . JVM:即Java Virtual Machine.我们都...

    java面试笔试题库java学习笔记开发教程互联网公司面试资料大全合集.zip

    Addison.Wesley.Java.Concurrency.in.Practice.May.2006.chm Agile Java 测试驱动开发的编程技术.pdf Java 8 默认方法和多继承.docx Java NIO通信框架在电信领域的实践.docx java.png javaconcurrencyinpractice.pdf...

    Java任小龙版基础笔记.zip

    10. **Java标准库**:熟悉常用的Java API,如Collections、Concurrency、Networking等包,它们提供了大量的类和接口,极大地丰富了Java的功能。 最后,配合任小龙的X-mind思维脑图,可以更直观地理解这些知识结构,...

    j7cc:Java 7 Concurrency Cookbook 翻译+学习笔记

    《Java 7 Concurrency Cookbook》是一本专注于Java并发编程的权威指南,对于任何希望深入理解并熟练运用Java并发特性的开发者来说,都是不可或缺的参考书。这本书不仅包含了对Java 7新特性的详细解读,还提供了大量...

    JAVA上课笔记class_17

    根据给定的文件信息,以下是对“JAVA上课笔记class_17”中涉及的重要知识点的详细解析: ### 泛型(Generics) 泛型是Java SE 5.0引入的一个特性,它允许编译时类型安全地处理不同类型的对象。在代码中使用泛型...

    Java基础笔记(包括底层原理)

    JUC(Java Concurrency Utilities)提供了高级同步工具,如Semaphore、CyclicBarrier和CountDownLatch,以及并发容器如ConcurrentHashMap,提升了多线程编程的便利性和性能。 总的来说,Java基础知识覆盖了从简单的...

    良葛格java5.0的学习笔记

    10. **Concurrency Updates(并发更新)**:Java 5.0增强了多线程编程的支持,如引入了`java.util.concurrent`包,包含Semaphore、ExecutorService、Future、Callable等类,提供了更高效的线程管理和任务调度。...

    Java高并发笔记.pdf

    **并发(Concurrency)** - **定义**:并发是指多个任务看起来同时执行的现象。但实际上,这些任务可能是在交替执行的。在一个单处理器系统中,由于处理器一次只能执行一个指令,因此并发实际上是通过任务间的快速切换...

    notes-learning-java-concurrency:java 并发学习笔记

    Java并发编程学习笔记 这是我两年前(2012-05)的学习笔记。。 -- 本文不会详细介绍java5以前的多线程开发的知识,而是重点介绍java5以来多线程开发的一些新变化。部分文字、代码都是摘抄、提炼过来的,大家有兴趣可...

    java面试笔试题库java软件设计java笔试题大集合及答案文档资料合集300MB.zip

    Addison.Wesley.Java.Concurrency.in.Practice.May.2006.chm Agile Java 测试驱动开发的编程技术.pdf Java 8 默认方法和多继承.docx Java NIO通信框架在电信领域的实践.docx java.png javaconcurrencyinpractice.pdf...

    java个人学习笔记总结

    本文将从 JVM、分布式与高并发、以及相关的技术如 Spring、MyBatis、JUC(Java Concurrency Utilities)包、mq、Redis 和 Zookeeper 等方面进行深入探讨。 一、JVM(Java 虚拟机) 1. JVM 是 Java 应用程序的运行...

    javaconcurrent源码-java7-source-code:Java7源码/Concurrency同步

    Concurrency in Practice 读书笔记和示例代码 包括 线程基础/同步sync/并发concurrent/容器/线程池 示例代码 JUC容器的源码解析 Synchronized 同步容器 Concurrent 并发容器: Map/List/Queue 阻塞队列 JUC锁: ...

Global site tag (gtag.js) - Google Analytics