楼主: jieforest

常用Flex IOC框架比较分析

[复制链接]
论坛徽章:
277
马上加薪
日期:2014-02-19 11:55:14马上有对象
日期:2014-02-19 11:55:14马上有钱
日期:2014-02-19 11:55:14马上有房
日期:2014-02-19 11:55:14马上有车
日期:2014-02-19 11:55:14马上有车
日期:2014-02-18 16:41:112014年新春福章
日期:2014-02-18 16:41:11版主9段
日期:2012-11-25 02:21:03ITPUB年度最佳版主
日期:2014-02-19 10:05:27现任管理团队成员
日期:2011-05-07 01:45:08
11#
 楼主| 发表于 2009-9-30 23:55 | 只看该作者
核心概念

任何使用过Spring的Java或.NET版本的人,都会很快熟悉Spring ActionScript。你在运行时加载的配置文件,可以给工厂提供足够信息,用于实例化被应用请求的任何对象。

基本配置

在标准例子工程中使用Spring ActionScript,需如下三个基本步骤:
1. 创建文件application-context.xml
2. 初始化应用中的工厂对象
3. 在你的视图层(或其他任何地方),根据需要从工厂获得对象以供使用

使用道具 举报

回复
论坛徽章:
277
马上加薪
日期:2014-02-19 11:55:14马上有对象
日期:2014-02-19 11:55:14马上有钱
日期:2014-02-19 11:55:14马上有房
日期:2014-02-19 11:55:14马上有车
日期:2014-02-19 11:55:14马上有车
日期:2014-02-18 16:41:112014年新春福章
日期:2014-02-18 16:41:11版主9段
日期:2012-11-25 02:21:03ITPUB年度最佳版主
日期:2014-02-19 10:05:27现任管理团队成员
日期:2011-05-07 01:45:08
12#
 楼主| 发表于 2009-9-30 23:55 | 只看该作者
对象工厂和对象配置

在Spring ActionScript中,对象声明在应用可访问的XML文件(通常命名为application-context.xml)中。此配置文件由XMLApplicationContext(是ObjectFactory的子类)加载。

在本例中,初始化工作由如下两个对象承担:ContextLoader和Inject。

ContextLoader获得应用上下文文件的路径。该文件在XMLApplicationContext中加载。在应用初始化部分有:
private function init() : void
{
ContextLoader.contextPath = "application-context.xml";
}

ContextLoader在幕后实现对Spring ActionScript上下文的加载:
public static function set contextPath( value : String ) : void
{
_contextPath = value;
applicationContext = new XMLApplicationContext( _contextPath );
applicationContext.addEventListener( Event.COMPLETE, handleLoadComplete );
applicationContext.load();
}

接着在需要依赖的视图中,我创建一个Inject标签(受一个同事在Parsley中实现的启发)。利用这个标签,我可以很方便的声明我需将何种依赖添加到该视图。例如,在应用启动时,我有如下代码:
<springActionscript:Inject
property="pm"
objectId="{ ContextIds.MAIN_CONTAINER_PM }"/>


<springActionscript:Inject
property="controller"
objectId="{ ContextIds.CONTROLLER }"/>

这将向XMLApplicationContext请求一个ID为CONTROLLER的对象,并将它赋给视图中的成员变量controller。

这是在视图层获取对象的好办法。

说明:Christophe Herreman曾撰文说明如何用metadata实现上述类型的注入(类似Swiz框架),但这种方法有性能问题,因为要读取元数据,视图需序列化为XML。

使用道具 举报

回复
论坛徽章:
277
马上加薪
日期:2014-02-19 11:55:14马上有对象
日期:2014-02-19 11:55:14马上有钱
日期:2014-02-19 11:55:14马上有房
日期:2014-02-19 11:55:14马上有车
日期:2014-02-19 11:55:14马上有车
日期:2014-02-18 16:41:112014年新春福章
日期:2014-02-18 16:41:11版主9段
日期:2012-11-25 02:21:03ITPUB年度最佳版主
日期:2014-02-19 10:05:27现任管理团队成员
日期:2011-05-07 01:45:08
13#
 楼主| 发表于 2009-9-30 23:55 | 只看该作者
设置控制器

Spring ActionScript已计划发布一个支持MVCS的的扩展版。但在目前版本中,我将实现一个自有的控制器,并利用Spring ActionScript将处理程序挂接到事件源。

在最初的程序中,控制器会监听所有事件。当它截获到一个事件后,会查找所有处理器,筛选出能处理这个事件的对象。在经过修改后的例子中,不再监控整个显示列表,转而在application-context.xml中将事件源和事件处理器配对。

为此,我增加了一个新类ControllerPair,它负责事件源和处理器的配对。所有对被传递给SimpleController,并在它的init()函数中初始化每个对。
<object id="controller" class="com.adobe.login.control.SimpleController">
<method-invocation name="init"></method-invocation>
<property name="controllerItems">
<array><ref>controllerItem</ref></array>
</property>
</object>

<object id="controllerItem” class="com.adobe.login.control.ControllerPair">
<property name="dispatcher" ref="loginPM"/>
<property name="handler" ref="handler"/>
</object>

请注意其中的method-invocation标签,我们用它来指定对象被创建后马上自动调用的函数。在这里,被自动调用的函数是init(),它负责将事件派发者绑定到事件处理器。

使用道具 举报

回复
论坛徽章:
277
马上加薪
日期:2014-02-19 11:55:14马上有对象
日期:2014-02-19 11:55:14马上有钱
日期:2014-02-19 11:55:14马上有房
日期:2014-02-19 11:55:14马上有车
日期:2014-02-19 11:55:14马上有车
日期:2014-02-18 16:41:112014年新春福章
日期:2014-02-18 16:41:11版主9段
日期:2012-11-25 02:21:03ITPUB年度最佳版主
日期:2014-02-19 10:05:27现任管理团队成员
日期:2011-05-07 01:45:08
14#
 楼主| 发表于 2009-9-30 23:56 | 只看该作者
表现层模型的注入

在非IOC版的ProfileViewer中,为了实现对象的传递,表现层模型被配置为继承式结构。在IOC版中,我将删除此结构,以便每个表现层模型都能被配置为对应的视图。

尽管这样做,应用更易于配置和测试,但也有其缺点。在某些情况下,要实现离散的表现层模型之间的交互,需要做不少工作。

Spring ActionScript同时支持setter和constructor两种形式的注入。我更倾向于使用construtor注入,因为它可以完全暴露对象运作所需的全部依赖。如下是DashboardPM的配置:
<object id="dashboardPM" class="com.adobe.dashboard.presentationModel.DashboardPM">
<constructor-arg ref="user"/>
</object>

在XML中声明构造函数的参数时,顺序应和对象的构造函数所期望的参数顺序相同。上面代码中的ref表示引用在上下文中声明的另一个对象,在这里即User。

使用道具 举报

回复
论坛徽章:
277
马上加薪
日期:2014-02-19 11:55:14马上有对象
日期:2014-02-19 11:55:14马上有钱
日期:2014-02-19 11:55:14马上有房
日期:2014-02-19 11:55:14马上有车
日期:2014-02-19 11:55:14马上有车
日期:2014-02-18 16:41:112014年新春福章
日期:2014-02-18 16:41:11版主9段
日期:2012-11-25 02:21:03ITPUB年度最佳版主
日期:2014-02-19 10:05:27现任管理团队成员
日期:2011-05-07 01:45:08
15#
 楼主| 发表于 2009-9-30 23:56 | 只看该作者
配置服务层

LoginHandler引用了代理对象,代理对象又依赖于另一个远程对象,该远程对象可以调用后端系统的功能。

下面,我们用setter完成这些对象的配置。需通过setter传入的实例包括代理对象和AuthenticationClient(这是一个用于检查用户是否已登录的接口)。MainPM具体实现了AuthenticationClient。

在这里,我将代理设计为存根,依赖于一个远程对象。配置如下:
<object id="handler" class="com.adobe.login.control.handler.LoginHandler">
<property name="client" ref="mainPM"/>
<property name="user" ref="user"/>
<property name="friends" ref="friends"/>
<property name="delegate" ref="loginDelegate"/>
</object>

<object id="loginDelegate" class="com.adobe.login.service.LoginDelegate">
<property name="remoteObject" ref="remoteObject"/>
</object>

<object id="remoteObject" class="mx.rpc.remoting.RemoteObject">
<property name="destination" value="SPRING_ACTIONSCRIPT_DESTINATION"/>
</object>

使用道具 举报

回复
论坛徽章:
277
马上加薪
日期:2014-02-19 11:55:14马上有对象
日期:2014-02-19 11:55:14马上有钱
日期:2014-02-19 11:55:14马上有房
日期:2014-02-19 11:55:14马上有车
日期:2014-02-19 11:55:14马上有车
日期:2014-02-18 16:41:112014年新春福章
日期:2014-02-18 16:41:11版主9段
日期:2012-11-25 02:21:03ITPUB年度最佳版主
日期:2014-02-19 10:05:27现任管理团队成员
日期:2011-05-07 01:45:08
16#
 楼主| 发表于 2009-9-30 23:56 | 只看该作者
总结

Spring ActionScript是一个优秀的、成熟的、开发活动十分活跃的IOC框架。它使用的术语,应该说是任何用过Spring框架的人都熟悉的。

以XML形式声明对象存在一个问题,即在XML中声明一个类,并且这个类没被包含在SWF中(因为在你的应用中没有对它的直接引用)时,Flash Player会在运行时抛出异常。其解决办法是创建一个ActionScript类,声明它对上下文XML的依赖,并将此类包含在应用中。

使用道具 举报

回复
论坛徽章:
277
马上加薪
日期:2014-02-19 11:55:14马上有对象
日期:2014-02-19 11:55:14马上有钱
日期:2014-02-19 11:55:14马上有房
日期:2014-02-19 11:55:14马上有车
日期:2014-02-19 11:55:14马上有车
日期:2014-02-18 16:41:112014年新春福章
日期:2014-02-18 16:41:11版主9段
日期:2012-11-25 02:21:03ITPUB年度最佳版主
日期:2014-02-19 10:05:27现任管理团队成员
日期:2011-05-07 01:45:08
17#
 楼主| 发表于 2009-9-30 23:56 | 只看该作者
Parsley
框架:Parsley
网站:http://www.spicefactory.org/
开发者:Jens Halm
版本:2.0.0
许可:开源
配置:XML/MXML/ActionScript

Parsley也是一个成熟的IOC框架,最初灵感来源于Spring。它近期经历过一次较大规模的重写。新版本支持一些本地Flex特性,如绑定和元数据,使你在配置你的工程时有更多更好的选择。

使用道具 举报

回复
论坛徽章:
277
马上加薪
日期:2014-02-19 11:55:14马上有对象
日期:2014-02-19 11:55:14马上有钱
日期:2014-02-19 11:55:14马上有房
日期:2014-02-19 11:55:14马上有车
日期:2014-02-19 11:55:14马上有车
日期:2014-02-18 16:41:112014年新春福章
日期:2014-02-18 16:41:11版主9段
日期:2012-11-25 02:21:03ITPUB年度最佳版主
日期:2014-02-19 10:05:27现任管理团队成员
日期:2011-05-07 01:45:08
18#
 楼主| 发表于 2009-9-30 23:57 | 只看该作者
核心概念

Parsley的核心概念是源自于Spring的上下文,也即应用的依赖注入的配置。

Parsley的配置现在支持多种形式,其中包括XML和MXML。你可以使用本地的MXML标记或Parsley库提供的自定义MXML标签。Parsley使用元数据标签实现对注入机制的支持,这和Swiz框架是类似的。

Parsley还支持消息模式。基本不需代码干预,你就能将你的对象配置为事件源或事件处理器。在这个例子中,我会用这个特性替代Controller模式。
基本配置

Parsley的配置分三个基本步骤:
创建Config.mxml文件。
在应用的根部初始化一个Context对象。
在你的视图中,用Inject元数据实现依赖的注入。

准备配置文件的方法有多种,不过在这个例子中,我使用支持本地标记和Parsley标签的MXML文件。这种方法的好处是在编译时就将类引入,当然这样一来,也就不能直接修改已被编译的应用的配置了。

使用道具 举报

回复
论坛徽章:
277
马上加薪
日期:2014-02-19 11:55:14马上有对象
日期:2014-02-19 11:55:14马上有钱
日期:2014-02-19 11:55:14马上有房
日期:2014-02-19 11:55:14马上有车
日期:2014-02-19 11:55:14马上有车
日期:2014-02-18 16:41:112014年新春福章
日期:2014-02-18 16:41:11版主9段
日期:2012-11-25 02:21:03ITPUB年度最佳版主
日期:2014-02-19 10:05:27现任管理团队成员
日期:2011-05-07 01:45:08
19#
 楼主| 发表于 2009-9-30 23:57 | 只看该作者
对象工厂和对象配置

在Config.mxml中,你能看到应用中用到的从域模型到代理的所有对象。声明这些对象的方式有两种:
标准的MXML
使用Parsley的对象定义标签

在后面的内容中,我将详细介绍这种方法。
设置控制器和LoginHandler

我在这里不再使用自己编写的控制器,转而使用Parsley的消息系统(其设计决定了它对你编写的对象的影响很小)。具体是用元数据来实现。Parsley将事件源绑定到事件处理器,需要一个在上下文可见且具有元数据的对象。

在这个例子应用中,LoginPM是事件源,LoginAction(从LoginHandler重命名而来)是事件处理器。

如下代码摘自LoginPM:
[Event( name="LOGIN", type="com.adobe.login.control.event.LoginEvent")]
[ManagedEvents("LOGIN")]
public class LoginPM extends EventDispatcher
{
...
public function login() : void
{
var event : LoginEvent = new LoginEvent( username, password );
dispatchEvent( event );
}
}

让LoginPM成为事件源需要三个要素:Event元数据标签、ManagedEvents元数据标签,以及EventDispatcher#dispatchEvent。三者当中,只有ManagedEvents是Parsley扩展而来。Event元数据仅为习惯做法,事件的实际派发工作是由dispatchEvent完成的。Parsley将通过ManagedEvents决定它要处理哪个事件,并将该事件委托给事件处理器。
如下代码摘自LoginAction(已经被配置为事件处理器):
public class LoginAction implements IResponder
{

[MessageHandler]
public function execute( event : LoginEvent ) : void
{
...
}
}

因为我为这个函数补充了MessageHandler元数据,Parsley将把这个对象/函数当做类型为LoginEvent的所有事件的监听器。

要让这些对象对Parsley可见,可在传入FlexContextBuilder的配置文件内声明这些对象,或在视图中使用Configure对象。

使用道具 举报

回复
论坛徽章:
277
马上加薪
日期:2014-02-19 11:55:14马上有对象
日期:2014-02-19 11:55:14马上有钱
日期:2014-02-19 11:55:14马上有房
日期:2014-02-19 11:55:14马上有车
日期:2014-02-19 11:55:14马上有车
日期:2014-02-18 16:41:112014年新春福章
日期:2014-02-18 16:41:11版主9段
日期:2012-11-25 02:21:03ITPUB年度最佳版主
日期:2014-02-19 10:05:27现任管理团队成员
日期:2011-05-07 01:45:08
20#
 楼主| 发表于 2009-9-30 23:57 | 只看该作者
表现层模型的注入

和其他例子一样,我已将表现层模型的继承结构去除。相关原因请参看Spring ActionScript。

Parsley支持setter和constructor两种注入方法。像我在Spring ActionScript例子中提到的那样,我更倾向于使用constructor注入,因为它可以暴露出对象运作所需的全部依赖。如是是DashboardPM的配置:
<spicefactory:Object type="{ DashboardPM }"/>

如果你的对象构造函数需要参数,则应该用Object标签予以声明,因为这样的参数在本地MXML中是不支持的。

要完成此构造函数,你需向类中添加一些元数据:
[InjectConstructor]
public class DashboardPM
{
public var user : User;

public function DashboardPM( user : User )
{
this.user = user;
}
...
}

这里的元数据标签InjectConstructor,表示要求Parsley给DashboardPM的构造函数注入一个类型为User的、已声明过的对象。

若使用setter注入,你仅需在类中增加元数据标签Inject。例如,我在Config中用标准MXML声明SummaryPM:
<dashboard:SummaryPM/>

接着,在类文件中有如下代码:
public class SummaryPM
{
[Inject]
public var friends : Friends;
...
}

这里的Inject标签表示需将一个类型为Friends的实例注入到SummaryPM。

使用道具 举报

回复

您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

TOP技术积分榜 社区积分榜 徽章 团队 统计 知识索引树 积分竞拍 文本模式 帮助
  ITPUB首页 | ITPUB论坛 | 数据库技术 | 企业信息化 | 开发技术 | 微软技术 | 软件工程与项目管理 | IBM技术园地 | 行业纵向讨论 | IT招聘 | IT文档
  ChinaUnix | ChinaUnix博客 | ChinaUnix论坛
CopyRight 1999-2011 itpub.net All Right Reserved. 北京盛拓优讯信息技术有限公司版权所有 联系我们 未成年人举报专区 
京ICP备16024965号-8  北京市公安局海淀分局网监中心备案编号:11010802021510 广播电视节目制作经营许可证:编号(京)字第1149号
  
快速回复 返回顶部 返回列表