MVC本来是存在于Desktop程序中的,M是指数据
模型,V是指用户界面,C则是控制器。使用MVC的目的是将M和V的实现代码分离,从而使同一个程序可以使用不同的表现形式。比如一批统计数据你可以分别用柱状图、饼图来表示。C存在的目的则是确保M和V的同步,一旦M改变,V应该同步更新。模型-
视图-控制器(MVC)是Xerox PARC在八十年代为编程语言Smalltalk-80发明的一种软件设计模式,至今已被广泛使用。谈竖仔最近几年被推荐为Sun公司J2EE平台的设计模式,并且受到越来越多的使用 ColdFusion 和 PHP 的开发者的欢迎。模型-视图-控制器模式是一个有用的工具箱,它有很多好处,但也有一些缺点。 MVC如何工作 MVC是一个设计模式,它强制性的使
应用程序的输入、处理和输出分开。使用MVC应用程序被分成三个核心部件:模型、视图、控制器。它们各自处理自己的任务。 视图 视图是用户看到并与之交互含汪的界面。对老式的Web应用程序来说,视图就是由HTML元素组成的界面,在新式的Web应用程序中,HTML依旧在视图中扮演着重要的角色,但一些新的技术已层出不穷,它们包括Macromedia Flash和象XHTML,XML/XSL,WML等一些标识语言和Web services. 如何处理应用程序的界面变得越来越有挑战性。MVC一个大的好处是它能为你的应用程序处理很多不同的视图。在视图中其实没有真正的处理发生,不管这些数据是联机存储的还纤悔是一个雇员列表,作为视图来讲,它只是作为一种输出数据并允许用户 *** 纵的方式。 模型 模型表示企业数据和业务规则。在MVC的三个部件中,模型拥有最多的处理任务。例如它可能用象EJBs和ColdFusion Components这样的构件对象来处理数据库。被模型返回的数据是中立的,就是说模型与数据格式无关,这样一个模型能为多个视图提供数据。由于应用于模型的代码只需写一次就可以被多个视图重用,所以减少了代码的重复性。 控制器 控制器接受用户的输入并调用模型和视图去完成用户的需求。所以当单击Web页面中的超链接和发送HTML表单时,控制器本身不输出任何东西和做任何处理。它只是接收请求并决定调用哪个模型构件去处理请求,然后用确定用哪个视图来显示模型处理返回的数据。 现在我们总结MVC的处理过程,首先控制器接收用户的请求,并决定应该调用哪个模型来进行处理,然后模型用业务逻辑来处理用户的请求并返回数据,最后控制器用相应的视图格式化模型返回的数据,并通过表示层呈现给用户。 为什么要使用 MVC 大部分Web应用程序都是用像ASP,PHP,或者CFML这样的过程化语言来创建的。它们将像数据库查询语句这样的数据层代码和像HTML这样的表示层代码混在一起。经验比较丰富的开发者会将数据从表示层分离开来,但这通常不是很容易做到的,它需要精心的计划和不断的尝试。MVC从根本上强制性的将它们分开。尽管构造MVC应用程序需要一些额外的工作,但是它给我们带来的好处是无庸质疑的。 首先,最重要的一点是多个视图能共享一个模型,正如我所提及的,现在需要用越来越多的方式来访问你的应用程序。对此,其中一个解决之道是使用MVC,无论你的用户想要Flash界面或是 WAP 界面;用一个模型就能处理它们。由于你已经将数据和业务规则从表示层分开,所以你可以最大化的重用你的代码了。 由于模型返回的数据没有进行格式化,所以同样的构件能被不同界面使用。例如,很多数据可能用HTML来表示,但是它们也有可能要用Macromedia Flash和WAP来表示。模型也有状态管理和数据持久性处理的功能,例如,基于会话的购物车和电子商务过程也能被Flash网站或者无线联网的应用程序所重用。 因为模型是自包含的,并且与控制器和视图相分离,所以很容易改变你的应用程序的数据层和业务规则。如果你想把你的数据库从MySQL移植到Oracle,或者改变你的基于RDBMS数据源到LDAP,只需改变你的模型即可。一旦你正确的实现了模型,不管你的数据来自数据库或是LDAP服务器,视图将会正确的显示它们。由于运用MVC的应用程序的三个部件是相互对立,改变其中一个不会影响其它两个,所以依据这种设计思想你能构造良好的松偶合的构件。 对我来说,控制器的也提供了一个好处,就是可以使用控制器来联接不同的模型和视图去完成用户的需求,这样控制器可以为构造应用程序提供强有力的手段。给定一些可重用的模型和视图,控制器可以根据用户的需求选择模型进行处理,然后选择视图将处理结果显示给用户。 MVC的缺点 MVC的缺点是由于它没有明确的定义,所以完全理解MVC并不是很容易。使用MVC需要精心的计划,由于它的内部原理比较复杂,所以需要花费一些时间去思考。 你将不得不花费相当可观的时间去考虑如何将MVC运用到你的应用程序,同时由于模型和视图要严格的分离,这样也给调试应用程序到来了一定的困难。每个构件在使用之前都需要经过彻底的测试。一旦你的构件经过了测试,你就可以毫无顾忌的重用它们了。 根据我个人经验,由于我们将一个应用程序分成了三个部件,所以使用MVC同时也意味着你将要管理比以前更多的文件,这一点是显而易见的。这样好像我们的工作量增加了,但是请记住这比起它所能带给我们的好处是不值一提。 MVC并不适合小型甚至中等规模的应用程序,花费大量时间将MVC应用到规模并不是很大的应用程序通常会得不偿失。 MVC是一条创建软件的好途径 MVC设计模式是一个很好创建软件的途径,它所提倡的一些原则,像内容和显示互相分离可能比较好理解。但是如果你要隔离模型、视图和控制器的构件,你可能需要重新思考你的应用程序,尤其是应用程序的构架方面。如果你肯接受MVC,并且有能力应付它所带来的额外的工作和复杂性,MVC将会使你的软件在健壮性,代码重用和结构方面上一个新的台阶。 引言在用Delphi Visual Basic等可视化快速开发工具编写Windows应用程序时 常会遇到这样几个问题 ) 希望程序界面美观 在Delphi中 开发人员通常使用各种控件来实现界面的风格化 但缺点是造成应用程序体积较大 且在升级时常会被控件版本与Delphi版本不兼容带来的问题所困扰 ) 希望应用程序在功能不变的情况下具有不同的界面风格 这常常通过换 皮肤 的技术来实现 但一般实现 换肤 功能的控件体积都较大 且界面反应速度比较慢 而且 皮肤 的制作比较麻烦 ) 程序界面的维护困难 为了使界面与代码实现相分离而获得肆卖缺 换肤 等灵活性 通常要用到一些设计模式的技术 这对于不熟悉设计模式的开发人员来说比较困难 微软公司预计将于 年发布下一代 *** 作系统(开发代号为Longhorn)中 应用程序的结构及部署将有重大变革 其中一项就是应用程序的界面完全以XML的一个扩展集XAML语言来描述 以便达到界面的高度可定制性 这无疑能够方便地解决上述几个问题 问题是在目前来说有没有类似的方法呢?答案就是使用浏览器控件 微软公司的网页浏览器Internet Explorer的核心被设计为可以嵌入到应用程序中重用的ActiveX组件 它有极强的可裂辩编程能力和与容器交互的能力 使得开发人员能够快速地开发出功能强劲的应用程序 从下面的Internet Explorer的架构图可以看到 我们平常运行的iexplorer exe其实只是一个外壳程序 真正的浏览网页 记录历史等工作是由嵌入其窗口的封装在shdocvw dll中的WebBrowser Control来完成的 Shdocvw dll的功能则是调用ms dll来解析网页 以及在它的窗口中嵌入其它活动文档组件(如Microsoft Office Adobe Acrobat等应用程序的文档都可以嵌入到浏览器窗口中查看) 而ms dll一方面处理HTML解析以及作为脚本引擎 java虚拟机 ActiveX控件 插件的宿主 另一方面 它实现了活动文档服务器接口 允许应用程序以标准的接口来把它嵌入到程序中并通过它暴露的接口来访问其中的网页及网页元素 通过shdocvw dll提供的丰富接口 网页中的元素可以访问外壳应用程序提供的属性和配友方法(如window external AddFavorite(location href document title)则是调用IE的AddFavorite方法把当前页添加到收藏夹) 而通过ms dll提供的接口 外壳应用程序则反过来可以访问网页中元素的属性 方法 行为 事件等等 解决文章开头提出的几个问题的方法就是基于shdocvw dll和ms dll实现的 一些著名软件如 Microsoft Money Microsoft Visual Studio NET Macromedia Dreamweaver MX 等都运用了这种技术 原理 ) 程序的界面完全由制作网页来完成 网页在文字 图像 声音等方面具有强大的表现能力 运用所见即所得的网页制作工具可以轻松制作出图文并茂的网页 以网页作为程序的界面 其效果胜过任何界面控件 ) 换肤 功能容易实现 只需制作不同风格的网页 即可轻松实现样式各异的程序界面 ) 程序的功能在应用程序内部编写代码来实现 并通过一个自动化接口提供给网页中的元素调用 这就实现了程序界面和代码的分离 网页布局及风格的改变不会影响到程序的实现 从网页调用外壳程序的属性和方法 GetExternal接口方法WebBrowser Control提供的接口使得外壳应用程序可以用自己的对象 方法和属性等来扩展IE的对象模型(DOM) 以达到个性化定制的目的 在网页中访问外壳应用程序的扩展则通过文档的 external 对象来实现 如外壳程序提供了名为AddFavorite的方法 网页中就通过window external AddFavorite()来调用 实现这一功能的核心是IDocHostUIHandler接口的GetExternal方法 HRESULT GetExternal(IDispatch **ppDispatch)在自定义的WebBrowser Control中实现IDocHostUIHandler接口 当网页元素通过 external 对象访问外壳扩展的属性和方法时 GetExternal方法就会被调用 在此方法的中将实现外壳程序属性和方法的自动化接口传递给ppDispatch即可 自定义的WebBrowser Control示例代码如下 在其中将GetExternal包装为OnGetExternal事件供外部程序调用 IDocHostUIHandler接口有 个方法 此处我们只关心GetExternal方法 故略去其余 个(省略号处为略去的代码) unit ZoCWebBrowserinterfaceuses Variants IEConst Windows SysUtils Classes SHDocVw ActiveX shlObj MSHTML objtype …… TGetExternalEvent = function(out ppDispatch: IDispatch): HRESULT of object //定义OnGetExternal事件类型 TZoCWebBrowser = class(ebBrowser IDocHostUIHandler) private……FOnGetExternal: TGetExternalEvent protected……function GetExternal(out ppDispatch: IDispatch): HRESULTstdcall published……property OnGetExternal: TGetExternalEvent read FOnGetExternal write FOnGetExternal end ……implementation……function TZoCWebBrowser GetExternal(out ppDispatch: IDispatch): HRESULTbegin if Assigned(FOnGetExternal) thenResult := FOnGetExternal(ppDispatch) elseResult := S_FALSEendinitialization OleInitialize(nil)finalization tryOleUninitialize except endend 实现外壳程序扩展自动化接口在Delphi的 New Items 对话框中 切换到 ActiveX 页 选择 Automation Object 新建一个自动化对象 并在 CoClass Name 一栏中填入接口名 MyExternal Instancing 选择为 Internal 表示该对象只能在程序内部被创建 外部程序不能直接创建 点击 OK 按钮后在Type Library编辑对话框中为IMyExternal接口添加两个方法ShowAboutBox和SwitchUI 此时代码大致如下所示 unit MyExternalImpl{$WARN SYMBOL_PLATFORM OFF}interfaceuses ComObj ActiveX Project _TLB StdVcltype TMyExternal = class(TAutoObject IMyExternal) protectedprocedure ShowAboutBoxsafecallprocedure SwitchUIsafecall endimplementationuses ComServprocedure TMyExternal ShowAboutBoxbegin MessageBox(MainForm Handle GetExternal Demo ZoCWebBrowser MB_OK or MB_ICONASTERISK)endprocedure TMyExternal SwitchUIbegin ShowSwitchUIForm //显示切换程序界面对话框endinitialization TAutoObjectFactory Create(ComServer TMyExternal Class_MyExternal ciInternal tmApartment)end 从网页中调用外壳程序接口在程序主窗口中放置一个自定义的WebBrowser Control 命名为ZoCWebBrowser 编写它的OnGetExternal事件(由网页中的window external调用触发) 代码如下 function TMainForm ZoCWebBrowserGetExternal( out ppDispatch: IDispatch): HRESULTvar MyExternal: TMyExternalbegin MyExternal:= TMyExternal Create//创建实现自动化接口的对象 ppDispatch :=MyExternal//将对象接口传递给WebBrowser Control//这样当 external 对象被调用时 真正被调用的是我们实现的TMyExternal对象 Result :=S_OKend假设我们制作了两个风格迥异的的网页l和l作为程序界面 这两个网页中都有两个按钮(也可以是其它网页元素) 其HTML代码示例如下 <BUTTON window external ShowAboutBox >关于</BUTTON><BUTTON window external SwitchUI >切换界面</BUTTON>在程序开始运行时让WebBrowser Control布满整个Form 且显示l页面 则当点击 关于 按钮时程序将显示一个关于信息对话框 而点击 切换界面 按钮时将显示切换界面的对话框 在其中选择l并让WebBrowser Control显示它即可获得风格完全不同的界面 但在功能上与l完全一样 总结从上面的例子可以看到 我们以及其简单的方式实现了程序界面与实现的分离 这有利于程序的维护和扩展 传统方式下 界面设计和编码通常都由程序员来完成 一来造成程序员负担较重 二来难以保证界面质量 实用上述方法 程序界面可以由专业美工人员来设计 他可以在完全不知道程序如何实现的情况下设计出完整的界面 而程序员只需专注于代码的编写 并将必要的方法和属性通过一个自动化接口暴露出来 合并的时候 在网页中合适的位置放入所需的按钮或其它网页元素 并赋予简单的脚本调用即可 lishixinzhi/Article/program/Delphi/201311/8412
评论列表(0条)