Cairngorm中Command、Delegate、Service交互详解

Command、Delegate、Service这三者的关系简单说来就是前者调用后者,后者为前者返回数据的关系!

原生的Cairngorm MVC 概览:

 

让我们以参考资料中的源码为例详细描述一下:

Command的源代码如下,类名叫:GetProductsCommand

package com.adobe.cairngorm.samples.store.command

{

        import mx.rpc.IResponder;

        import com.adobe.cairngorm.commands.ICommand;

        import com.adobe.cairngorm.control.CairngormEvent;

        import com.adobe.cairngorm.samples.store.business.ProductDelegate;

        import com.adobe.cairngorm.samples.store.model.ShopModelLocator;

        import com.adobe.cairngorm.samples.store.util.Comparator;

        import mx.rpc.events.ResultEvent;

        import mx.rpc.events.FaultEvent;

        import mx.controls.Alert;

        import mx.collections.ICollectionView;

        import mx.collections.Sort;

        import mx.collections.SortField;

        import mx.utils.ArrayUtil;

 

        public class GetProductsCommand implements ICommand, IResponder

        {

               public function execute( event : CairngormEvent ): void

               {

                  if( ShopModelLocator.getInstance().products == null )

                  {

                      var delegate : ProductDelegate = new ProductDelegate( this );

                      delegate.getProducts();

                   }

                   else

                   {

                        Alert.show( "Products already retrieved!" );

                        return;

                    }

               }

 

               public function result( event : Object ) : void

               {      

                 //result()方法从表现服务器结果的event

中提

                 //

products数组并将它们保存在

ShopModelLocator中,

                 //同时将

selectedItem

设置为产品列表中的第一个产品。

                 var products : ICollectionView = ICollectionView( event.result );

                 var model : ShopModelLocator = ShopModelLocator.getInstance();

                 // sort the data。

                 var sort :Sort = new Sort();

                 sort.fields = [ new SortField( "name", true ) ];

                 products.sort = sort;

                 products.refresh();

               // set the products on the model

                 model.selectedItem = products[ 0 ];

                 model.products = products;

                 model.workflowState = ShopModelLocator.VIEWING_PRODUCTS_IN_THUMBNAILS;

               }

 

               public function fault( event : Object ) : void

               {

                  var faultEvent : FaultEvent = FaultEvent( event );

                  Alert.show( "Products could not be retrieved!" );

               }

        }

 }

注意代码中的这一行:

var delegate : ProductDelegate = new ProductDelegate( this );

这里将GetProductsCommand类的实例作为参数(this)传递给ProductDelegate的构造函数,这样做就是告诉Delegate,该Command将处理所有通过Dlegate发送的请求所返回的结果。在Delgate中,这个接收所有返回结果的Delegate成为Responder。如下Delegate的源码(ProductDelegate.as)中可以看到:

package

com.adobe.cairngorm.samples.store.business

{

        import mx.rpc.IResponder;

        import com.adobe.cairngorm.business.ServiceLocator;

        import mx.rpc.events.FaultEvent;

        import mx.rpc.events.ResultEvent;

        import mx.rpc.AbstractOperation;

 

        public class ProductDelegate

        {

               //构造函数的参数为IResponder类,将接受所有请求的返回结果

               public function ProductDelegate( responder : IResponder )

               {             

                       this.service = ServiceLocator.getInstance().getRemoteObject( "productService" );

                       this.responder = responder;

               }

               public function getProducts() : void

               {                     

                       //调用Service,这两行代码可以作为模板,在其他Cairngorm应

                       //用中写法也是一样的!这两行代码确保了从服务器返回的

                       //results 和faults都能返回给Delegate。

                       var call : Object = service.getProducts();

                       call.addResponder( responder );

               }

               private var responder : IResponder;

               private var service : RemoteObject;

        }

}

从第一段代码中可以看到,Command类实现了ICommand和 IResponder两个接口,当它选择扮演responder的角色,Command实现FLEX定义的IResponder接口;这确保了开发者可以添加一个

result()犯法来处理任何

Delegate结果,一个

fault()方法来处理所有

Delegate的错误。

整个工作流程可以描述如下:

1、Command类产生一个业务代理(delegate)的实例;

2、Command类调用delegate中的业务处理方法,上例中为getProducts();

3、Delegate查找到service,并调用service;

4、过了一会,服务器返回结果给delegate;

5、Delegate立即将结果传给command的result()方法或者调用command的fault()方法。

通过这个机制的运用,一方面使得Delegate可以被重用,另一方面将服务器端调用进行了封装,delegate变成了客户端代码和服务器端代码的唯一接口。这样做也为客户端和服务器端代码分别开发提供了可能(我们在写客户端程序是不一定服务器端程序已经就绪,只需要它提供一些静态试验数据(或称为为数据:dummy data)即可)。

==================================================

参考资料:

1)Developing Flex RIAs with Cairngorm microarchitecture – Part 5: Server-side integration

你可能感兴趣的:(工作,mvc,Flex,Adobe)