来自:冰河技术

前言

面向对象思想与并发编程有关系吗?本来二者是没有什么鸟关系的!它们是分属两个不同的领域,但是,Java却将二者融合在一起了!而且融合的效果不错:我们利用Java的面向对象的思想能够让并发编程变得更加简单!!

那我们如何利用面向对象的思想写好并发程序呢?我们可以从下面三个角度进行分析。

  • 封装共享变量

  • 识别共享变量间的约束条件

  • 指定并发访问策略

封装共享变量

在编写并发程序时,我们关注的一个核心问题,其实就是解决多线程同时访问共享变量的问题!

面向对象思想中有一个很重要的特性:封装。简单的说,封装就是将属性和实现细节封装到对象的内部,外界对象只能通过目标对象提供的公共方法来间接访问内部属性。我们把共享变量作为对象的属性,那么,对于共享变量的访问路径就是对象的公共方法,所有公共方法的入口都要设置并发访问策略。

所以,我们得出一个结论:利用面向对象思想写并发程序其实挺简单,就是将共享变量作为对象属性封装在内部,对所有的公共方法指定并发访问策略!

比如,我们在很多业务场景中都会用到计数器,我们可以将计数器类定义成如下所示。

public class Counter{private long count;public synchronized long incrementCount(){return ++count;}public synchronized long getCount(){return count;}
}

在上面的Counter类中,存在一个共享变量count,对外提供的两个公共方法incrementCount()和getCount()设置了synchronized同步锁,此时,Counter类就是一个线程安全的类了。

在实际工作中,很多场景比计数器的实现复杂的多,比如,我们的银行账户中,有卡号、姓名、身份证、余额等共享变量,我们没有必要对每个共享变量都要考虑并发问题。此时,我们就需要仔细分析这些共享变量,看这些共享变量中哪些变量是不变的。对于我们的银行账户来说,卡号、姓名、身份证这三个共享变量就是不变的。对于这些不变的共享变量,我们可以使用final关键字来修饰它们,避免并发问题。

最后,需要注意的是,对共享变量进行封装时,要注意”对象逃逸“的问题!例如,下面的程序代码,在构造函数中将this赋值给了全局变量global.obj,此时对象初始化还没有完成,此时对象初始化还没有完成,此时对象初始化还没有完成,重要的事情说三遍!!线程通过global.obj读取的x值可能为0。此时对象this就“逃逸”了。

final x = 0;
public FinalFieldExample() { // bad!x = 3;y = 4;// bad construction - allowing this to escapeglobal.obj = this;
}

以上示例来源于:http://www.cs.umd.edu/~pugh/java/memoryModel/jsr-133-faq.html#finalWrong

识别共享变量间的约束条件

共享变量间的约束条件非常重要,因为它们决定了并发访问策略。

例如,在商城业务中,对于商品的库存管理中有个合理库存的概念,库存量不能太高,也不能太低,这个值有一个上限和一个下限。例如,下面的类模拟了这个合理的库存概念。

public class Stock{//库存的上限private final AtomicLong upper = new AtomicLong(0);//库存的下限private final AtomicLong lower = new AtomicLong(0);//设置库存上限public void setUpper(long v){upper.set(v);}//设置库存下限public void setLower(long v){lower.set(v);}//其他众多的代码省略
}

乍一看,上面的程序没问题啊!但是,其忽略了一个约束条件,就是库存的下限要小于库存的上限。这也是很多人容易忽略的问题。

看到这里,很多人的第一反应就是在setUpper()方法和setLower()方法中,添加参数校验逻辑,例如,改造后的Stock类如下所示。

public class Stock{//库存的上限private final AtomicLong upper = new AtomicLong(0);//库存的下限private final AtomicLong lower = new AtomicLong(0);//设置库存上限public void setUpper(long v){if(v < lower.get()){throw new IllegalArgumentException();}upper.set(v);}//设置库存下限public void setLower(long v){if(v > upper.get()){throw new IllegalArgumentException();}lower.set(v);}//其他众多的代码省略
}

这样设置正确吗?答案是:这样设置完全不同保证库存的下限小于库存的上限。

其实,这里存在竞态条件(当程序中出现 if 语句的时候,应该首先反应出程序是否有竞态条件),关于竞态条件的详细讲解可以参见《【高并发】要想学好并发编程,关键是要理解这三个核心问题》。

假设,原有库存的上限为10,下限为3。此时线程A调用setUpper(5)将库存的上限设置为5,线程B调用setLower(7)将库存的下限设置为8,如果线程A和线程B同时执行,线程A会通过参数校验,因为此时库存的下限还没有被线程B设置完毕,此时的库存下限还是3,5>3成立,所以,线程A会将库存的上限设置为5。同样的,线程B也能够通过参数校验,因为此时库存的上限还没有被线程A设置完毕,此时库存的上限还是10,8<10成立,线程B会将库存的下限设置为8。最终的结果为:库存的上限为5,下限为8。库存的上限小于下限,不满足上限小于下限的约束条件。

所以,大家在识别共享变量间的约束条件时,一定要注意竞态条件的问题!

制定并发访问策略

制定并发访问策略比较复杂,它需要结合具体的业务场景进行选择。但是从方案上,我们可以将其总结成如下方案。

避免共享

可以利用线程本地存储和为每个任务分配独立的线程来避免共享。

不变模式

这个在Java中使用的比较少,在其他的领域使用的比较多,例如Actor模式,CSP模式和函数式编程。

管程和其他同步工具

Java中对于并发编程万能的解决方案就是管程(关于什么是管程后面的文章会讲解),但是对于很多特定的并发场景来说,使用Java并发包提供的读写锁、并发容器等同步工具比较好。

我们在编写并发程序时,也要遵循一定的原则,这些原则可以归纳如下。

优先使用成熟的工具类

对于并发编程来说,我们最好优先使用Java中提供的并发工具类,因为这些并发工具类基本上能够满足大部分并发的业务场景。

尽量不要使用低级的同步原语

低级的同步原语指的是synchronized,Lock和Semaphore等,这些使用起来虽然简单,但实际上并没有那么简单,使用的时候一定要小心。不到万不得已的时候,尽量不要使用它们。

避免过早优化

安全第一,并发编程首先要保证的就是线程安全,出现性能瓶颈之后再优化,不要过早和过度的优化。

写在最后

最后,附上并发编程需要掌握的核心技能知识图,祝大家在学习并发编程时,少走弯路。

特别推荐一个分享架构+算法的优质内容,还没关注的小伙伴,可以长按关注一下:长按订阅更多精彩▼如有收获,点个在看,诚挚感谢

信不信?以面向对象的思想是可以写好高并发程序的!相关推荐

  1. 至于你信不信,我反正是信了——以类为单位的编程思想

    一.打开和关闭数据库       你需要知道这么几个信息:你要连接的是哪台机器(机器名:SPXY-WYH):你要采取什么身份验证方式(Window身份验证方式,还是SQL Server身份验证方式): ...

  2. java编写计算类加减乘除_老师要求张浩使用面向对象的思想编写一个计算器类(Calculator),可以实现两个整数的加减乘除的运算.java...

    导航:网站首页 > 老师要求张浩使用面向对象的思想编写一个计算器类(Calculator),可以实现两个整数的加减乘除的运算.java 时间:2019-4-10 老师要求张浩使用面向对象的思想编 ...

  3. 女人体相——面相知识——信不信由你

    女人体相--面相知识--信不信由你     俗话说:好头不如好面,好面不如好身,一个人的运势,可以从她的身材体貌略窥一二.       圆脸 脸圆的人在相学上主金水,对人际关系及财运都有加分的作用,但 ...

  4. Java面向对象编程思想

    面向对象三个特征: 封装.继承.多态 封装: 语法: 属性私有化(private).提供相对应的get/set 的方法进行访问(public). 在set/get的方法中对属性的数据 做相对应的业务逻 ...

  5. Silverlight游戏设计(Game Design):(五)面向对象的思想塑造游戏对象

    传说,面向对象的开发模式最初是因为程序员偷懒而不小心诞生的.发展至今,人们从最初的热忠于讨论某某语言是否足够面向对象到现在开始更广泛的关注面向对象的思想而不是具体内容.面向对象的思想其实并不深奥,它存 ...

  6. 面向对象编程思想(2)--策略模式

    定义 策略模式 官方定义:定义了一系列的算法,并将每一个算法封装起来,而且使它们还可以相互替换. 个人理解:选择执行多个规则中的某个规则. C#实现 需求1: 开发一个商场收银系统v1.0 三下五除二 ...

  7. 以面向对象的思想实现数据表的添加和查询,JDBC代码超详细

    以面向对象的思想编写JDBC程序,实现使用java程序向数据表中添加学生信息,并且可以实现给定×××号查询学生信息或给定准考证号查询学生信息. 创建的数据表如下: CREATE TABLE EXAMS ...

  8. U3D 飞机大战(MVC模式)解析--面向对象编程思想

    在自己研究U3D游戏的时候,看过一些人的简单的游戏开发视频,写的不错,只是个人是java web 开发的人,所以结合着MVC思想,对游戏开发进行了一番考虑. 如果能把游戏更加的思想化,分工化,开发便明 ...

  9. C#基础第七天-作业-利用面向对象的思想去实现名片-动态添加

    1.利用面向对象的思想去实现: (增加,修改,删除,查询,查询全部) 需求:根据人名去(删除/查询). 指定列:姓名,年龄,性别,爱好,电话. 多条添加 , 动态添加 名片 本系列教程: C#基础总结 ...

最新文章

  1. 气候变迁给社会带来什么变化?
  2. MVC起始页面路径设置
  3. 模拟断电oracle数据不一致,Oracle数据库案例整理-Oracle系统运行时故障-断电导致数据文件状态变为RECOVER...
  4. C++11 并发指南四(future 详解二 std::packaged_task 介绍)
  5. c++课程学习(未完待续)
  6. ubuntu14.04 安装php5-fpm
  7. 用fileupload处理文件上传
  8. CLR via C# 边读边想 03 - 本地程序集和强命名程序集
  9. 以WIN10 22H2为例,下载ISO、制作安装U盘的办法
  10. java filenet_为FileNet P8组件集成器开发Java组件
  11. 手机幻灯片html代码,html5手机幻灯片制作手指滑动触屏手机幻灯片代码
  12. Opencv中BGR、YUV、YUV_I420\NV12分析
  13. 【C语言】结构体指针与结构体数组
  14. 算法设计与分析:分治法输出数字旋转方阵
  15. 卸载计算机应用程序的步骤,图文教您win10系统快速卸载应用程序的具体步骤
  16. JavaWeb - 黑马旅游网(2):用户注册
  17. Flume基础知识(个人总结)
  18. 怎么实现接口解耦_将接口与实现解耦-使用分离的接口
  19. TTL与CMOS 电路
  20. 宁津县大数据中心建成 属“智慧宁津”子项目

热门文章

  1. android ble不配对接收广播数据_蓝牙低功耗(BLE)学习笔记_0
  2. RNN情感分类问题实战
  3. 关于学习Python的一点学习总结(30->递归实例)
  4. 关于学习Python的一点学习总结(9->字典创建及相关操作)
  5. python raise valueerror_raise ValueError('无法设置没有定义索引的帧'ValueError:
  6. centos python2.7升级到3.7_centos系统升级python 2.7.3
  7. 《繁凡的论文精读》(一)CVPR 2019 基于决策的高效人脸识别黑盒对抗攻击(清华朱军)
  8. (每日一题)P3768 简单的数学题(确信)(莫反 + 欧拉反演 + 杜教筛 )
  9. CF429D Tricky Function(求解公式、经分析转为求平面最近点对、思维)
  10. 没有头文件调用cpp_VS2017中同一个解决方案下不同工程的调用