




已阅读5页,还剩5页未读, 继续免费阅读
版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
Java多线程特性为构建高性能的应用提供了极大的方便,但是也带来了不少的麻烦。线程间同步、数据一致性等烦琐的问题需要细心的考虑,一不小心就会出现一些微妙的,难以调试的错误。另外,应用逻辑和线程逻辑纠缠在一起,会导致程序的逻辑结构混乱,难以复用和维护。本文试图给出一个解决这个问题的方案,通过构建一个并发模型框架(framework),使得开发多线程的应用变得容易。基础知识Java语言提供了对于线程很好的支持,实现方法小巧、优雅。对于方法重入的保护,信号量(semaphore)和临界区(critical section)机制的实现都非常简洁。可以很容易的实现多线程间的同步操作从而保护关键数据的一致性。这些特点使得Java成为面向对象语言中对于多线程特性支持方面的佼佼者(C+正在试图把boost库中的对于线程的支持部分纳入语言标准)。Java中内置了对于对象并发访问的支持,每一个对象都有一个监视器(monitor),同时只允许一个线程持有监视器从而进行对对象的访问,那些没有获得监视器的线程必须等待直到持有监视器的线程释放监视器。对象通过synchronized关键字来声明线程必须获得监视器才能进行对自己的访问。synchronized声明仅仅对于一些较为简单的线程间同步问题比较有效,对于哪些复杂的同步问题,比如带有条件的同步问题,Java提供了另外的解决方法,wait/notify/notifyAll。获得对象监视器的线程可以通过调用该对象的wait方法主动释放监视器,等待在该对象的线程等待队列上,此时其他线程可以得到监视器从而访问该对象,之后可以通过调用notify/notifyAll方法来唤醒先前因调用wait方法而等待的线程。一般情况下,对于wait/notify/notifyAll方法的调用都是根据一定的条件来进行的,比如:经典的生产者/消费者问题中对于队列空、满的判断。熟悉POSIX的读者会发现,使用wait/notify/notifyAll可以很容易的实现POSIX中的一个线程间的高级同步技术:条件变量。简单例子本文将围绕一个简单的例子展开论述,这样可以更容易突出我们解决问题的思路、方法。本文想向读者展现的正是这些思路、方法。这些思路、方法更加适用于解决大规模、复杂应用中的并发问题。考虑一个简单的例子,我们有一个服务提供者,它通过一个接口对外提供服务,服务内容非常简单,就是在标准输出上打印Hello World。类结构图如下:代码如下:1.interface Service2.3. public void sayHello();4.5.class ServiceImp implements Service6.7. public void sayHello() 8. System.out.println(Hello World!);9. 10.11.class Client12.13. public Client(Service s) 14. _service = s;15.16. public void requestService() 17. _service.sayHello();18. 19. private Service _service;20.如果现在有新的需求,要求该服务必须支持Client的并发访问。一种简单的方法就是在ServicImp类中的每个方法前面加上synchronized声明,来保证自己内部数据的一致性(当然对于本例来说,目前是没有必要的,因为ServiceImp没有需要保护的数据,但是随着需求的变化,以后可能会有的)。但是这样做至少会存在以下几个问题:1.现在要维护ServiceImp的两个版本:多线程版本和单线程版本(有些地方,比如其他项目,可能没有并发的问题),容易带来同步更新和正确选择版本的问题,给维护带来麻烦。2.如果多个并发的Client频繁调用该服务,由于是直接同步调用,会造成Client阻塞,降低服务质量。3.很难进行一些灵活的控制,比如:根据Client的优先级进行排队等等。4.这些问题对于大型的多线程应用服务器尤为突出,对于一些简单的应用(如本文中的例子)可能根本不用考虑。本文正是要讨论这些问题的解决方案,文中的简单的例子只是提供了一个说明问题,展示思路、方法的平台。5.如何才能较好的解决这些问题,有没有一个可以重用的解决方案呢?让我们先把这些问题放一放,先来谈谈和框架有关的一些问题。框架概述熟悉面向对象的读者一定知道面向对象的最大的优势之一就是:软件复用。通过复用,可以减少很多的工作量,提高软件开发生产率。复用本身也是分层次的,代码级的复用和设计架构的复用。大家可能非常熟悉C语言中的一些标准库,它们提供了一些通用的功能让你的程序使用。但是这些标准库并不能影响你的程序结构和设计思路,仅仅是提供一些机能,帮助你的程序完成工作。它们使你不必重头编写一般性的通用功能(比如printf),它们强调的是程序代码本身的复用性,而不是设计架构的复用性。那么什么是框架呢?所谓框架,它不同于一般的标准库,是指一组紧密关联的(类)classes,强调彼此的配合以完成某种可以重复运用的设计概念。这些类之间以特定的方式合作,彼此不可或缺。它们相当程度的影响了你的程序的形貌。框架本身规划了应用程序的骨干,让程序遵循一定的流程和动线,展现一定的风貌和功能。这样就使程序员不必费力于通用性的功能的繁文缛节,集中精力于专业领域。有一点必须要强调,放之四海而皆准的框架是不存在的,也是最没有用处的。框架往往都是针对某个特定应用领域的,是在对这个应用领域进行深刻理解的基础上,抽象出该应用的概念模型,在这些抽象的概念上搭建的一个模型,是一个有形无体的框架。不同的具体应用根据自身的特点对框架中的抽象概念进行实现,从而赋予框架生命,完成应用的功能。基于框架的应用都有两部分构成:框架部分和特定应用部分。要想达到框架复用的目标,必须要做到框架部分和特定应用部分的隔离。使用面向对象的一个强大功能:多态,可以实现这一点。在框架中完成抽象概念之间的交互、关联,把具体的实现交给特定的应用来完成。其中一般都会大量使用了Template Method设计模式。Java中的Collection Framework以及微软的MFC都是框架方面很好的例子。有兴趣的读者可以自行研究。构建框架如何构建一个Java并发模型框架呢?让我们先回到原来的问题,先来分析一下原因。造成要维护多线程和单线程两个版本的原因是由于把应用逻辑和并发逻辑混在一起,如果能够做到把应用逻辑和并发模型进行很好的隔离,那么应用逻辑本身就可以很好的被复用,而且也很容易把并发逻辑添加进来而不会对应用逻辑造成任何影响。造成Client阻塞,性能降低以及无法进行额外的控制的原因是由于所有的服务调用都是同步的,解决方案很简单,改为异步调用方式,把服务的调用和服务的执行分离。首先来介绍一个概念,活动对象(Active Object)。所谓活动对象是相对于被动对象(passive object)而言的,被动对象的方法的调用和执行都是在同一个线程中的,被动对象方法的调用是同步的、阻塞的,一般的对象都属于被动对象;主动对象的方法的调用和执行是分离的,主动对象有自己独立的执行线程,主动对象的方法的调用是由其他线程发起的,但是方法是在自己的线程中执行的,主动对象方法的调用是异步的,非阻塞的。本框架的核心就是使用主动对象来封装并发逻辑,然后把Client的请求转发给实际的服务提供者(应用逻辑),这样无论是Client还是实际的服务提供者都不用关心并发的存在,不用考虑并发所带来的数据一致性问题。从而实现应用逻辑和并发逻辑的隔离,服务调用和服务执行的隔离。下面给出关键的实现细节。本框架有如下几部分构成:1.一个ActiveObject类,从Thread继承,封装了并发逻辑的活动对象;2.一个ActiveQueue类,主要用来存放调用者请求;3.一个MethodRequest接口,主要用来封装调用者的请求,Command设计模式的一种实现方式。它们的一个简单的实现如下:1. /MethodRequest接口定义2. interface MethodRequest3.4. public void call();5.6./ActiveQueue定义,其实就是一个producer/consumer队列7. class ActiveQueue8.9. public ActiveQueue() 10. _queue = new Stack();11. 12. public synchronized void enqueue(MethodRequest mr) 13. while(_queue.size() QUEUE_SIZE) 14. try 15. wait();16. catch (InterruptedException e) 17. e.printStackTrace();18. 19. 20.21. _queue.push(mr);22. notifyAll();23. System.out.println(Leave Queue);24. 25. public synchronized MethodRequest dequeue() 26. MethodRequest mr;27.28. while(_queue.empty() 29. try 30. wait();31. catch (InterruptedException e) 32. e.printStackTrace();33. 34. 35. mr = (MethodRequest)_queue.pop();36. notifyAll();37.38. return mr;39. 40. private Stack _queue;41. private final static int QUEUE_SIZE = 20;42.43./ActiveObject的定义44.class ActiveObject extends Thread45.46. public ActiveObject() 47. _queue = new ActiveQueue();48. start();49. 50. public void enqueue(MethodRequest mr) 51. _queue.enqueue(mr);52. 53. public void run() 54. while(true) 55. MethodRequest mr = _queue.dequeue();56. mr.call();57. 58. 59. private ActiveQueue _queue;60.通过上面的代码可以看出正是这些类相互合作完成了对并发逻辑的封装。开发者只需要根据需要实现MethodRequest接口,另外再定义一个服务代理类提供给使用者,在服务代理者类中把服务调用者的请求转化为MethodRequest实现,交给活动对象即可。使用该框架,可以较好的做到应用逻辑和并发模型的分离,从而使开发者集中精力于应用领域,然后平滑的和并发模型结合起来,并且可以针对ActiveQueue定制排队机制,比如基于优先级等。基于框架的解决方案本小节将使用上述的框架重新实现前面的例子,提供对于并发的支持。第一步先完成对于MethodRequest的实现,对于我们的例子来说实现如下:1.class SayHello implements MethodRequest2.3. public SayHello(Service s) 4. _service = s;5. 6. public void call() 7. _service.sayHello();8. 9. private Service _service;10.该类完成了对于服务提供接口sayHello方法的封装。接下来定义一个服务代理类,来完成请求的封装、排队功能,当然为了做到对Client透明,该类必须实现Service接口。定义如下:11.class ServiceProxy implements Service12.13. public ServiceProxy() 14. _service = new ServiceImp();15. _active_object = new ActiveObject();16. 17.18. public void sayHello() 19. MethodRequest mr = new SayHello(_service);20. _active_object.enqueue(mr);21. 22. private Service _service;23. private ActiveObject _active_object;24.其他的类和接口定义不变,下面对比一下并发逻辑增加前后的服务调用的变化,并发逻辑增加前,对于sayHello服务的调用方法:25.Service s = new ServiceImp();26.Client c = new Client(s);27.c.requestService();并发逻辑增加后,对于sayHello服务的调用方法:28.Service s = new ServiceP
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 大学生疾病健康科普
- 2025版轨道交通设备采购与运营维护合作合同
- 二零二五年度工厂保安人员派遣及生产安全协议
- 2025版年度财务代理与财务培训服务合同
- 二零二五年度智慧城市土地开发合同
- 二零二五年度瓷砖出口退税代理服务合同
- 2025版车库租赁合同包含车位租赁合同解除及违约责任
- 2025版电子产品区域代理销售返点合同样本
- 二零二五年度财务信息系统安全保密服务合同
- 二零二五年度特色餐厅厨师餐饮培训合同
- 农业法律法规知识课件
- 耕地质量评价课件
- 关于加强医药卫生领域廉政建设的意见(2025年版)解读
- 人力资源管理毕业论文-事业单位人力资源管理创新策略研究
- 砖砌围墙工程施工方案
- 人力资源管理试题及答案
- 2025年新乡市市直事业单位招考(338名)高频重点模拟试卷提升(共500题附带答案详解)
- 外研版九年级上册英语阅读理解60题含参考答案
- 建筑工程施工工序管控
- 2025年昆明铁道职业技术学院高职单招职业技能测试近5年常考版参考题库含答案解析
- 废钢知识培训课件
评论
0/150
提交评论