多线程系列之三:Immutable 模式
一,什么是Immutable模式?
immutable就是不变的,不发生改变的。Immutable模式中存在着确保实例状态不发生变化改变的类。这些实例不需要互斥处理。
String就是一个Immutable类,String实例所表示的字符串的内容不会变化。
二,定义一个使用Immutable模式的类
public final class Person {
private final String name;
private final String address;
public Person(String name,String address){
this.name = name;
this.address = address;
}
public String getName(){
return name;
}
public String getAddress(){
return address;
}
@Override
public String toString() {
return "[ Person : name = "+name+",address = "+address+"]";
}
}
final修饰类,无法创建子类,防止子类修改其子段值
private修饰字段,内部可见,防止子类修改字段值
final修饰了字段,赋值后不在改变
三,何时使用这种模式?
1.实例创建后,状态不再发生变化
2.实例是共享的,且被频繁访问时
四,集合类与多线程
1.ArrayList类用于提供可调整大小的数组,是非线程安全的。当多个线程并发执行读写时,是不安全的。
public class WriterThread extends Thread {
private final List<Integer> list;
public WriterThread(List<Integer> list){
super("WriterThread");
this.list = list;
}
@Override
public void run() {
for (int i = 0;true ; i++) {
list.add(i);
list.remove(0);
}
}
}
public class ReaderThread extends Thread{
private final List<Integer> list;
public ReaderThread(List<Integer> list){
super("ReaderThread");
this.list = list;
}
@Override
public void run() {
while (true){
for (int n :list){
System.out.println(n);
}
System.out.println("------");
}
}
}
/**
* 并发读写List,会出异常
* Exception in thread "ReaderThread" java.util.NoSuchElementException
* Exception in thread "ReaderThread" java.util.ConcurrentModificationException
*
*/
public class ListTest {
public static void main(String[] args) {
List<Integer> list = new ArrayList<>();new WriterThread(list).start();
new ReaderThread(list).start();
}
}
2.利用Collections.synchronizedList方法锁进行的同步
利用Collections.synchronizedList方法进行同步,就能够得到线程安全的实例
public class WriterThread extends Thread {
private final List<Integer> list;
public WriterThread(List<Integer> list){
super("WriterThread");
this.list = list;
}
@Override
public void run() {
for (int i = 0;true ; i++) {
list.add(i);
list.remove(0);
}
}
}
public class ReaderThread extends Thread{
private final List<Integer> list;
public ReaderThread(List<Integer> list){
super("ReaderThread");
this.list = list;
}
@Override
public void run() {
while (true){
//使用了synchronizedList,读数据时必须加锁
synchronized (list){
for (int n :list){
System.out.println(n);
}
}
System.out.println("------");
}
}
}
public class ListTest {
public static void main(String[] args) {
List<Integer> arrayList = new ArrayList<>();
List<Integer> list= Collections.synchronizedList(arrayList);
new WriterThread(list).start();
new ReaderThread(list).start();
}
}
3.使用copy-on-write 的java.util.concurrent.CopyOnWriteArrayList类
copy-on-write,就是写时复制,如果使用copy-on-write,当对集合执行 写操作时,内部已确保安全的数组就会被整体复制。复制之后,就无需在使用迭代器依次读取数据时
担心元素被修改了。所以该类不会抛出并发修改异常
public class WriterThread extends Thread {
private final List<Integer> list;
public WriterThread(List<Integer> list){
super("WriterThread");
this.list = list;
}
@Override
public void run() {
for (int i = 0;true ; i++) {
list.add(i);
list.remove(0);
}
}
}
public class ReaderThread extends Thread{
private final List<Integer> list;
public ReaderThread(List<Integer> list){
super("ReaderThread");
this.list = list;
}
@Override
public void run() {
while (true){
for (int n :list){
System.out.println(n);
}
System.out.println("------");
}
}
}
public class CopyOnWriteListTest {
public static void main(String[] args) {
final List<Integer> list = new CopyOnWriteArrayList<>();
new WriterThread(list).start();
new ReaderThread(list).start();
}
}
使用copy-on-write时,每次执行 写操作时都会执行复制。因此程序频繁执行写操作时,如果使用CopyOnWriteArrayList,会比较花费时间。
如果写操作比较少,读炒作频繁时,很适合用CopyOnWriteArrayList。
具体根据情况而定。
多线程系列之三:Immutable 模式的更多相关文章
- 完毕port(CompletionPort)具体解释 - 手把手教你玩转网络编程系列之三
手把手叫你玩转网络编程系列之三 完毕port(Completion Port)具体解释 ...
- Android多线程分析之三:Handler,Looper的实现
Android多线程分析之三:Handler,Looper的实现 罗朝辉 (http://www.cnblogs.com/kesalin/) CC 许可,转载请注明出处 在前文<Android多 ...
- Java多线程系列--“JUC集合”05之 ConcurrentSkipListMap
概要 本章对Java.util.concurrent包中的ConcurrentSkipListMap类进行详细的介绍.内容包括:ConcurrentSkipListMap介绍ConcurrentSki ...
- Red Gate系列之三 SQL Server 开发利器 SQL Prompt 5.3.4.1 Edition T-SQL智能感知分析器 完全破解+使用教程
原文:Red Gate系列之三 SQL Server 开发利器 SQL Prompt 5.3.4.1 Edition T-SQL智能感知分析器 完全破解+使用教程 Red Gate系列之三 SQL S ...
- .NET 4 并行(多核)编程系列之三 从Task的取消
原文:.NET 4 并行(多核)编程系列之三 从Task的取消 .NET 4 并行(多核)编程系列之三 从Task的取消 前言:因为Task是.NET 4并行编程最为核心的一个类,也我们在是在并行编程 ...
- java多线程系列(三)---等待通知机制
等待通知机制 前言:本系列将从零开始讲解java多线程相关的技术,内容参考于<java多线程核心技术>与<java并发编程实战>等相关资料,希望站在巨人的肩膀上,再通过我的理解 ...
- java多线程系列 目录
Java多线程系列1 线程创建以及状态切换 Java多线程系列2 线程常见方法介绍 Java多线程系列3 synchronized 关键词 Java多线程系列4 线程交互(wait和 ...
- Java多线程系列——原子类的实现(CAS算法)
1.什么是CAS? CAS:Compare and Swap,即比较再交换. jdk5增加了并发包java.util.concurrent.*,其下面的类使用CAS算法实现了区别于synchronou ...
- Java多线程系列——从菜鸟到入门
持续更新系列. 参考自Java多线程系列目录(共43篇).<Java并发编程实战>.<实战Java高并发程序设计>.<Java并发编程的艺术>. 基础 Java多线 ...
随机推荐
- oracle外部表
关于外部表的描述 正确描述 the create table as select statement can be used to upload data into a normal table in ...
- php学习----运算符
PHP 1.运算符 加减乘除与数学运算无异 但PHP的赋值运算符有两种,分别是: (1)"=":把右边表达式的值赋给左边的运算数.它将右边表达式值复制一份,交给左边的运算数.换而言 ...
- virtualbox+ievms:还你一个原装IE8
在web开发中,不可避免的一件事是浏览器兼容性问题,你永远无法想象项目正式上线后,坐在电脑前操作这套系统的人用的是什么版本的浏览器,IE(7,8,...),360,Chrome,火狐等,后面几个还好说 ...
- Linux:自动删除n天前日志
linux是一个很能自动产生文件的系统,日志.邮件.备份等.虽然现在硬盘廉价,我们可以有很多硬盘空间供这些文件浪费,让系统定时清理一些不需要的文件很有一种爽快的事情.不用你去每天惦记着是否需要清理日志 ...
- input accept属性限制文件上传格式
上传文件的类型:具体做法如下所示: 注意:accept属性可以限制上传格式,其有兼容性如下 <1>上传.csv格式的 <input text="file" acc ...
- (转)Spring Boot(六):如何优雅的使用 Mybatis
http://www.ityouknow.com/springboot/2016/11/06/spring-boot-mybatis.html 这两天启动了一个新项目因为项目组成员一直都使用的是 My ...
- 【JSOI2018】潜入行动
[JSOI2018]潜入行动 树形\(DP\).设\(f_{i,j,0/1,0/1}\)表示以\(i\)为根的子树中,用了\(j\)个监听器,是否放置了监听器,是否被监听的方案数.转移就多讨论几种情况 ...
- CentOS 7下安装Python3.6
CentOS 7下安装Python3.6.4 CentOS 7下安装Python3.5 •安装python3.6可能使用的依赖 yum install openssl-devel bzip2-de ...
- postgresql命令
连接数据库, 默认的用户和数据库是postgrespsql -U user -d dbname 切换数据库,相当于mysql的use dbname\c dbname列举数据库,相当于mysql的sho ...
- 【JAVA8】双冒号
现在JDK双冒号是: public class MyTest { public static void printValur(String str){ System.out. ...