什么情况下用线程池
1.单个任务处理的时间比较短
2.将需处理的任务的数量大
使用线程池的好处:
1.减少在创建和销毁线程上所花的时间以及系统资源的开销
2.如不使用线程池,有可能造成系统创建大量线程而导致消耗完系统内存以及”过度切换”。
同一时刻只能对一个变量进行更新。ABA问题版本号解决.
为什么wait()方法要放在同步块中?
参考 java多线程并发实战
使用多线程的原因:资源利用率,公平性,便利性
###关键字:竞态条件
在多线程环境下存在的一种并发性问题,运行时多个线程交替操作共享的内存空间,因此它们可能会访问或修改其他线程正在使用的变量。导致出现竞态条件。
###多线程需要考虑的问题:
安全性含义:永远不会发生糟糕的事情,而活跃性则是关注另一个目标,即某件正确的事情最终会发生。
编写线程安全的代码其核心在于要对状态访问操作进行管理,特别是对"共享"的和"可变"的状态的访问。反之,如果数据"不共享"或"不可变",则代码线程安全。
"共享"意味着变量可以由多个线程同时访问,而"可变"则意味着变量的值在其生命周期是可以发生变化的。
当多个线程访问某个状态变量并且其中一个线程执行写入操作时,必须采用同步的机制来协同这些线程对变量访问。java中主要同步机制的关键字是synchronized,它提供了一种独占的加锁方式,同步的术语还包括volatile类型的变量。显示的加锁(synchronized)与原子变量(volatile)。
如果当多个线程访问同一个可变的状态变量时,没有合适的使用同步,那么程序就会出现错误,三种方式避免出现此问题:
1.不在线程之间共享此变量
2.将共享的变量变为不可变的常量
3.在访问变量时,使用同步机制
在任何情况中,只有当类中仅包含自己的状态时,线程安全类才是有意义的。
线程安全性最核心的概念就是正确性,正确性的含义是,某个类的行为与其规范完全一致。
关键字:数据竞争
"数据竞争"术语很容易与另外一个相关术语"竞态条件"混淆,"数据竞争"是指,如果在访问共享的非final类型的域时没有采用同步来进行协调,那么就会出现数据竞争,当一个数据写入一个变量而另一个线程接下来读取这个变量,或者读取一个之前由另一个线程写入的变量时,并且在这两个线程之间没有使用同步,那么就可能出现数据竞争。
竞态条件出现的场景:比较经典的场景就是单例模式,单例模式首先判断对象是否初始化,如果没有则初始化,并创建一个新的对象并返回一个引用,从而在后台的调用中无须再执行这段高开销的代码。其中先检查后执行的场景存在竞态条件。
复合操作形式如 延迟初始化,读取-修改-写入等
java提供了一种内置锁机制来支持原子性,内置锁 synchronized同步代码块儿,以及同步机制的另一个重要方面,可见性(voliate)。
synchronized
同步代码块主要包含两个部分,一个是作为锁的对象的引用,一个作为由这个锁保护的代码段。以关键字synchronized修饰的方法就是一种横跨整个方法体的同步代码,其中该同步代码锁的就是方法调用的所在对象。静态的synchronized就是以Class作为锁。
每个java对象都可以用来做一个实现同步的锁,这些锁被称为内置锁,或监视器锁。线程在进入同步代码块儿之前会自动获得锁,并且在退出同步代码块时自动释放锁,而无论是通过正常的控制路径退出,还是通过从代码中抛出异常退出,获得内置锁的唯一途径就是进入由这个锁保护的同步代码或方法。
关键字:重入
重入意味着获取锁的操作的粒度是"线程",而不是"调用"。相当于一个拿到钥匙的人,怎么开门进门都无所谓,因为他有钥匙。但是当他失去钥匙,就需要重新竞争钥匙了。
AQS中的count对同一个线程是不会拦截过滤的,每次释放count减一,直到最终释放。