Java并发编程实战 第4章 对象的组合
Java监视器模式
java监视器模式就是在将共享的数据封装在一个类里面,然后然后所有访问或者修改这些数据的方法都标注为synchronize。
车辆追踪模拟:
使用监视器模式:
CarTracker对象维护了一个所有汽车坐标的Map,这个Map是竞争资源,线程会同时对它进行更新和读取。所以才每个方法上都加了synchronized。
- package com.zjf;
- import java.util.*;
- //定义坐标
- class Point{
- public int x,y;
- public Point(Point p){
- this.x=p.x;
- this.y=p.y;
- }
- }
- //车辆追踪
- public class CarTracker{
- //维护所有车辆的坐标Map key是汽车的ID 这是竞争资源
- private Map<String,Point> locations;
- public CarTracker(Map<String,Point> points){
- locations=deepCopy(points);
- }
- //获得所有车辆的坐标Map
- public synchronized Map<String,Point> getLocations(){
- return deepCopy(locations);
- }
- //所有某一车辆的坐标
- public synchronized Point getLocation(String id){
- Point p=locations.get(id);
- return (p==null)?null:new Point(p);
- }
- //设置某一个汽车的坐标
- public synchronized void setLocation(String id,int x,int y){
- Point p=locations.get(id);
- if(p==null)
- System.out.print("id not exists");
- p.x=x;
- p.y=y;
- }
- //深拷贝
- public static Map<String,Point> deepCopy(Map<String,Point> m){
- Map<String,Point> result=new HashMap<String,Point>();
- for(String id:m.keySet()){
- result.put(id,new Point(m.get(id)));
- }
- return Collections.unmodifiableMap(result);
- }
- }
使用java的并发集合来重写上面的代码:
- package com.zjf;
- import java.util.*;
- import java.util.concurrent.ConcurrentHashMap;
- import java.util.concurrent.ConcurrentMap;
- //定义坐标 这个是不可变类型 所以可以直接返回 不担心被修改
- class Point {
- public final int x, y;
- public Point(int x, int y) {
- this.x = x;
- this.y = y;
- }
- }
- // 车辆追踪
- public class CarTracker {
- // 维护所有车辆的坐标Map key是汽车的ID 这是竞争资源 使用ConcurrentMap
- private final ConcurrentMap<String, Point> locations;
- //是locations的视图 locations的变化会直接映射到这里 但是它是不可修改的。
- private final Map<String, Point> unmodifiableMap;
- public CarTracker(Map<String,Point> points){
- locations =new ConcurrentHashMap<String,Point>(points);
- unmodifiableMap=Collections.unmodifiableMap(locations);
- }
- // 获得所有车辆的坐标Map 结果是不可修改的
- public Map<String,Point> getLocations(){
- return unmodifiableMap;
- }
- // 获取某一车辆的坐标 结果也是不可修改的
- public Point getLocation(String id){
- return locations.get(id);
- }
- // 设置某一个汽车的坐标 使用replace方法 这是ConcurrentMap提供的并发安全的方法
- public void setLocation(String id,int x,int y){
- if(locations.replace(id,new Point(x,y))==null)
- System.out.print("id not exists");
- }
- }
上面的方式,我们成为委托。我们把对车辆Map的并发管理委托给ConcurrentHashMap类。
Java并发编程实战 第4章 对象的组合的更多相关文章
- Java并发编程实战 第3章 对象的共享
可见性 可见性是由于java对于多线程处理的内存模型导致的.这似乎是一种失败的设计,但是JVM却能充分的利用多核处理器的强大性能,例如在缺乏同步的情况下,Java内存模型允许编译器对操作顺序进行重排序 ...
- 【JAVA并发编程实战】2、对象的组合
1. 设计线程安全的类 1.找出构成对象状态的所有变量 2.找出约束状态变量的不变性条件 3.建立对象状态的并发访问管理策略 package cn.xf.cp.ch04; /** * *功能:JAVA ...
- Java并发编程实战---第六章:任务执行
废话开篇 今天开始学习Java并发编程实战,很多大牛都推荐,所以为了能在并发编程的道路上留下点书本上的知识,所以也就有了这篇博文.今天主要学习的是任务执行章节,主要讲了任务执行定义.Executor. ...
- Java并发编程实战 第16章 Java内存模型
什么是内存模型 JMM(Java内存模型)规定了JVM必须遵循一组最小保证,这组保证规定了对变量的写入操作在何时将对其他线程可见. JMM为程序中所有的操作定义了一个偏序关系,称为Happens-Be ...
- 【java并发编程实战】第一章笔记
1.线程安全的定义 当多个线程访问某个类时,不管允许环境采用何种调度方式或者这些线程如何交替执行,这个类都能表现出正确的行为 如果一个类既不包含任何域,也不包含任何对其他类中域的引用.则它一定是无状态 ...
- Java并发编程实战 第8章 线程池的使用
合理的控制线程池的大小: 下面内容来自网络.不过跟作者说的一致.不想自己敲了.留个记录. 要想合理的配置线程池的大小,首先得分析任务的特性,可以从以下几个角度分析: 任务的性质:CPU密集型任务.IO ...
- 读书笔记-----Java并发编程实战(二)对象的共享
public class NoVisibility{ private static boolean ready; private static int number; private static c ...
- JAVA并发编程实战---第三章:对象的共享(2)
线程封闭 如果仅仅在单线程内访问数据,就不需要同步,这种技术被称为线程封闭,它是实现线程安全性的最简单的方式之一.当某个对象封闭在一个线程中时,这种方法将自动实现线程安全性,即使被封闭的对象本生不是线 ...
- JAVA并发编程实战---第三章:对象的共享
在没有同步的情况下,编译器.处理器以及运行时等都可能对操作的执行顺序进行一些意想不到的调整.在缺乏足够同步的多线程程序中,要对内存操作的执行顺序进行判断几乎无法得到正确的结果. 非原子的64位操作 当 ...
随机推荐
- Linux_Shell基础
目录 目录 查看系统中可以使用的shell 重定向 管道 变量 export指令与echo的不同 算术运算符 用户个性化系统变量文件 通配符 引号 文件比较运算符 查看系统中可以使用的shell ca ...
- apache禁止php解析--安全
#禁止解析php <Directory "/data/www/data/"> php_admin_flag engine off <filesmatch &quo ...
- 关于 if __name__ == '__main__':的使用
当是用 if __name__ == '__main__': 时,下面调用函数只是在当前脚本调试, 而我们需要在别处导入这个脚本中的类或者函数时,这个if __name__ == '__main__ ...
- Java学习之==>int和Integer的区别和联系
一.区别 1.类型 int是java中原始八种基本数据类型之一: Integer是一个类,包装整型提供了很多日常的操作: 2.存储位置和大小 int是由jvm底层提供,由Java虚拟机规范,int型数 ...
- IDEA/Git 提交/commit 忽略 文件夹
commit的时候.idea文件夹被默认选中了,如果忘记点掉就会被提交上去,想要默认忽略其实很简单. 找到项目根目录处的.gitignore文件(如果是用git版本控制的话) 双击打开之后 我们在最 ...
- 【Java基础】JAVA 使用线程的几种方式
之前放在自己网站上的例子,因为网站关闭,已经找不到了,想用的时候,没有的话又重新翻书是很麻烦的事情.所以重新记录一下,以备将来查看. 第一种,让任务类继承Runable接口,然后将任务类对象放入Thr ...
- LeetCode.925-长按的名字(Long Pressed Name)
这是悦乐书的第355次更新,第380篇原创 01 看题和准备 今天介绍的是LeetCode算法题中Easy级别的第217题(顺位题号是925).你的朋友正在键盘上输入他的名字. 有时,在键入字符c时, ...
- 车牌识别1:License Plate Detection and Recognition in Unconstrained Scenarios阅读笔记
一.WHAT 论文下载地址:License Plate Detection and Recognition in Unconstrained Scenarios [pdf] github 的项目地址: ...
- 【电子电路技术】短波红外InGaAs探测器简析
核心提示: 红外线是波长介于微波与可见光之间的电磁波,波长在0.75-1000μm之间,其在军事.通讯.探测.医疗等方面有广泛的应用.目前对红外线的分类还没有统一的标准,各个专业根据应用的需要,有着自 ...
- jQ的toggle() 方法
语法:$(selector).toggle(speed,callback,switch) 实例: <script src="js/jquery.min.js">< ...