Concurrent Container 2020-03-20 Vector、Hashtable Vector 特点 类似一个线程安全的 ArrayList * 大量容器操作方法都采用同步方法 Vector<String> vector = new Vector<>(); vector.add("test"); System.out.println(vector.get(0)); Hashtable 特点 类似一个线程安全的 HashMap * 大量容器操作方法都采用同步方法 Hashtable<String, String> hashtable = new Hashtable<>(); hashtable.put("a", "80%"); System.out.println(hashtable.get("a")); Collections.synchronizedXXX() 虽然这两个类不是线程安全的,但是可以用 Collections.synchronizedList(new ArrayList<Integer>())、Conllections.synchroniz....
Immutable 2020-03-18 不变性(Immutable) 如果对象在被创建后,状态就不能被修改,那么它就是不可变的。 具有不变性的对象一定是线程安全的,我们不需要对其采取任何二外的安全措施,也能保证线程安全。 示例:person 对象,age、name 都不能再变。 /** * 描述: 不可变的对象,演示其他类无法修改这个对象,public也不行 */ public class Person { final int age = 18; String alice = new String("Alice");//这样就破坏了不可变性 final String name = alice; final TestFinal testFinal = new TestFinal(); public static void main(String[] args) { Person person = new Person(); person.alice = "44"; System.out.println(person.name); } } final 的作用 类防止被继承,方法防止被重写,变量防止被修改 天生是线程安全的....
CAS 2020-03-17 CAS 简介 compare and swap 一种思想、一种用来实现线程安全的算法。 同时也是一种 CPU 指令: compare and swap(比较和交换),一种不会被打断的操作组合。 实现思路 我认为V的值应该是A,如果是的话我就把它修改成B,如果不是A(说明被别人修改过),那我就不修改了,避免多人同时修改导致出错。 CAS 有三个操作数:内存值V、预期值A、要修改的值B,当且仅当预期值 A 和内存值 V 相同时,才将内存值修改为 B,否则什么都不做。最后返回现在的值。 * CPU 的特殊指令保证了原子性 CAS 等价代码(Java) /** * 描述: Java 代码 模拟CAS操作,等价代码 */ public class SimulatedCAS { private volatile int value; //整个方法作用相当于 CPU 的一条指令(CAS 指令) public synchronized int compareAndSwap(int expectedValue, int newValue) { int oldValue = value; i....
Lock 2020-03-15 Lock 简介、地位、作用 锁是一种工具, 用于控制对共享资源的访问。 Lock 和 synchronzied,这两个是最常见的锁,它们都可以达到线程安全的目的,但是在使用上和功能上又有较大的不同。 Lock 并不是用来代替 synchronized 的,而是当使用 synchronized 不适合或不足以满足要求的时候,来提供高级功能的。 Lock 接口最常见的实现类是 ReentrantLock. 为什么需要 Lock? synchronized 的痛点 效率低:锁的释放情况少(等待 IO 时候也不释放锁,无法中途跳出,白白浪费 CPU 时间片),试图获得锁时不能设置超时(死锁的成因之一)、不能中断一个正在试图获得锁的线程。 不够灵活(读写锁更灵活):加锁和释放的时机单一,每个锁仅有单一的条件(某个对象),可能是不够的。 无法知道是否成功获取到锁。 Lock 主要方法介绍 在 Lock 中声明了四个方法来获取锁。 lock() tryLock() tryLock(long time, TimeUnit unit) lockInterruptibly() lock()....
ThreadLocal 2020-03-12 ThreadLocal 适用场景 场景 1:每个线程需要一个独享的对象(通常是工具类,典型需要使用的类有 SimpleDateFormat、Random、工具类线程不安全,所以每个线程需要拥有自己独立的工具类) * 每个 Thread 内有自己的实例副本,不共享 SimpleDateFormat 进化过程 1. 2个线程分别用自己的 SimpleDateFormat 2. 10个线程变为for循环,10个线程new 了10个SimpleDateFormat 3. 1000个线程,必然用线程池(否则消耗太多内存) 4. static SimpleDateFormat ,所有线程公用同一个 SimpleDateFormat 对象 (发现线程不安全) 5. 用 synchronized 加锁修复了线程安全性,但是影响了并发能力。 6. ThreadLocal 登场。线程池中的每个线程中,都有一个 SimpleDateFormat 对象副本,提高并发能力,还不会像 synchronized 那样带来线程安全问题 示例:两个线程分别打印出自己的日期 (用 synchronized 加锁,修复....
死锁 2020-03-10 死锁 发生在并发中,发生在两个线程或多个线程之间。 互不相让:当两个(或更多)线程(或进程)相互持有对方所需要的资源,又不主动释放,导致所有人都无法继续前进,导致程序陷入无尽的阻塞(等待状态),这就是死锁。 多个线程造成死锁的情况 如果多个线程之间的依赖关系是环形,存在环路的锁的依赖关系,那么也可能会发生死锁。 死锁的影响 死锁的影响在不同系统中是不一样的,这取决于系统对死锁的处理能力。 * 数据库中:检测并放弃事务(事务之间发生死锁) * JVM中:无法自动处理 几率不高但危害大 不一定发生,但是遵循“墨菲定律”。 一旦发生,多是高并发场景,影响用户多。 整个系统崩溃,子系统崩溃、性能降低。 压力测试无法找出所有潜在的死锁。(死锁和并发量只是一个正相关的关系,并不是必然的关系,所以无法通过增加并发量来模拟出死锁,也无法保证高并发的情况下可以 100% 的模拟出死锁。) 发生死锁的例子 最简单的情况 /** * 描述: 必定发生死锁的情况 */ public class MustDeadLock implements Runnable { int flag = 1;....