日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

Flex与外部的数据通信

發布時間:2024/1/23 编程问答 29 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Flex与外部的数据通信 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

視頻課:https://edu.csdn.net/course/play/7621

本章簡介

第3章講解了視圖狀態、Flex頁面間的跳轉、Flex應用的模態窗體、數據綁定、使用拖放,圖表等知識。本章將學習Flex與外部的數據通信。在實際開發過程中,Flex應用的數據往往來自于業務邏輯數據提供的服務器端。雖然Flex是用于開發富客戶端界面的強大平臺,可以編寫業務邏輯,但從架構的角度看,仍然需要將核心的業務邏輯放在Flex程序之外。Flex與外部程序的數據通信主要包括HTTPService. WebService和Remoting 3種方式。

核心技能部分

?

?

從分層角度來看,企業應用系統主要分為三個層次,如表4-1-1所示。

表5-1-1企業應用主要分層

企業應用層次

?

職責簡介

?

展現層

主要負責信息展示和人機交互的相關邏輯

領域層

主要完成企業應用業務邏輯,是系統的核心部分

數據源層

負責數據的提取和持久化

展現層主要負責信息展示以及用戶與軟件之間的交互邏輯,“展現層”接受用戶輸入并將用戶的意圖轉換為對“領域層’或“數據源層’邏輯的調用。

領域層也被稱為“業務邏輯層”,主要解決應用所針對業務領域的問題。該層負責校驗來自“展現層”的輸人數據,根據“展現層’用戶指令進行業務邏輯處理,調用“數據源層’的邏輯實現數據的訪問和持久化。

數據源層主要負責數據訪問和持久化,數據可能來自于數據庫或者消息系統。對干絕大多數企業應用來說,數據源層主要負責同數據庫系統的交互。

Flex+Java企業應用中,“展現層’邏輯完全運行在客戶端的Flash虛擬機中,而“領域層”和“數據源層”邏輯則運行在服務器端的Java虛擬機中,如圖5.1.1所示。

?

5.1.1 Flex+java企業應用層次邏輯分布圖

???

?從圖5.1.1中可以看出,客戶端系統與服務端系統完全用不同的語言實現,因此系統是異構的。同時,客戶端代碼運行在客戶端的ActionScript虛擬機中。而服務器端代碼則運行在服務器上的Java虛擬機中,因此系統又是分布式的。這與我們開發傳統Web應用完全不同,

傳統Web應用中所有Java代碼,包括業務邏輯代碼和生成人機界面的代碼都在服務器Java虛擬機中執行,如圖5.1.2所示。

?

5.1.2 傳統web應用層次邏輯分布圖

????基于傳統Web技術進行開發,很多開發者已經習慣了“接受客戶端的請求,然后執行業務邏軾,最后輸出人機界面”這種工作模式。基于Jsp技術的MVC框架,比如Struts,Jsf等,都是基于這種工作模式開發的。

????因此、Flex+Java所開發的BS應用與傳統Web所開發的B/S系統最大的區就是:使用Flex+Java開發的B/S應用系統中,B系統(客戶端系統)S系統(服務器端系統)完全分離、各自獨立地運行在不同的CPU和虛擬機中。B系統主要負責“展現層”邏輯,而S系統主要負責“領域層”和“數據源層”邏輯。因此,Flex+J ava所開發的企業應用系統是異構的分布式系統,這種異構分布式系統給我們帶來了以下需要思考的問題:

??異構的客戶端系統和服務器端系統如何通信?

??如何保持分布式的客戶端系統和服務器端系統之間的狀態一致性?

????我們在進行架構設計時,必須要清楚并解決這些問題,才能順利進行企業應用開發,下面兩個小節主要針對以上兩個問題進行闡述,并給出解決方案。

1.1?確定Flex客戶端系統和Java服務器端通信框架

????開發異構系統時,如何進行通信和傳遞數據是我們比較關注的問題。使用Flex+Java開發基于B/S結構企業應用,客戶端和服務器端的通信協議是我們所熟知的HTTP協議。在Flex中,基于HTTP協議訪問服務器的通信組件有三個:

?????HttpService(mx.rpc.http.mxml.HTTPService)

?????WebService(mx.rpc.soap.mxml.WebService)

?????RemoteObject(mx.rpc.remoting.mxml.RemoteObject)

HttpService組件可以調用很多不同技術的服務端頁面,比如JSP,ASP ,PHP,Servlet等在大多數情況下,使用HttpService訪問服務器端頁面來完成Flex客戶端與服務器端的數據交互,服務器端返回的結果一般都是XML格式的數據。下面是Adobe官方關于HTTPService組件的例子應用:

<?xml version="1.0" encoding="utf-8"?>

<mx:Application xmlns:mx="http://www.adobe.com/2006/mxml"

???creationComplete="feedRequest.send()" layout="absolute">

???<mx:HTTPService id="feedRequest"

??????url="http://weblogs.macromedia.com/mchotin/index.xml"

??????useProxy="false" />

???<mx:Panel x="10" y="10" width="475" height="400"

????????title="{feedRequest.lastResult.rss.channel.title}">

???????<mx:DataGrid id="dgPosts" x="20" y="20" width="400"

???????????dataProvider="{feedRequest.lastResult.rss.channel.item}">

???????????<mx:columns>

??????????????<mx:DataGridColumn headerText="Posts" dataField="title"/>

??????????????<mx:DataGridColumn headerText="Date"

?????????????? ????dataField="pubDate" width="150" />

???????????</mx:columns>

???????</mx:DataGrid>

???????<mx:LinkButton x="20" y="225" label="Read Full Post"

???????????????????click="navigateToURL(new URLRequest(dgPosts.selectedItem.link));"/>

???????<mx:TextArea x="20" y="175" width="400"/>

???</mx:Panel>

</mx:Application>

這個例子的運行需要能夠訪問互聯網,在本例中通過調用URLhttp://weblogs.macromedia.com/mchotin/index.xmlHTTPService ,返回了一個XML文件,并將這個XML作為Datagrid控件的dataProvider,從而通過DatagridXML文件中的數據展示出來。XML中的數據主要是網站最近的發帖記錄。這個例子說明,HTTPService的工作方式主要通過請求URL獲取XML格式數據。

上例運行后 效果如圖5.1.3所示。

?

5.1.3 HttpService 示例

????HTTPService類似,Flex應用可以調用URL所表示WSDL 1.1服務,返回SOAP1.1格式的調用結果。SOAP也是基于XML格式規范,因此。使用HTTPServiceWebService組件同服務器之間的交互都是通過XML進行的。下面是Adobe官方關于WebService組件的例子應用。 ?

<?xml version="1.0" encoding="utf-8"?>

<mx:Application xmlns:mx="http://www.adobe.com/2006/mxml"

layout="absolute"

creationComplete="wsBlogAggr.getMostPopularPosts.send()">

<mx:WebService id="wsBlogAggr"

?????wsdl="http://weblogs.macromedia.com/mxna/webservices/

mxna2.cfc?wsdl"

?????useProxy="false">

??????<mx:operation name="getMostPopularPosts">

?????????<mx:request>

????????????<daysBack>30</daysBack>

?????????????<limit>{cbxNumPosts.value}</limit>

?????????</mx:request>

?????</mx:operation>

</mx:WebService>

<mx:Panel x="10" y="10" width="475" height="400" layout="absolute"

????title="Most Popular Posts">

????<mx:ComboBox x="30" y="25" id="cbxNumPosts"

???????change="wsBlogAggr.getMostPopularPosts.send()">

???????<mx:Object label="Top 5" data="5" />

???????<mx:Object label="Top 10" data="10" />

???????<mx:Object label="Top 15" data="15" />

????</mx:ComboBox>

????<mx:DataGrid x="30" y="75" id="dgTopPosts" width="400"

??????dataProvider="{wsBlogAggr.getMostPopularPosts.lastResult}">

<mx:columns>

<mx:DataGridColumn headerText="Top Posts"

dataField="postTitle"/>

<mx:DataGridColumn headerText="Clicks" dataField="clicks"

width="75"/>

</mx:columns>

</mx:DataGrid>

<mx:LinkButton x="30" y="250"

label="Select an item and click here for full post"/>

</mx:Panel>

</mx:Application>

?

??在這個例子中,通過WebService組件調用了服務器所提供的WebService服務,返回SOAP

格式的XML數據,根據請求參數,XML數據表示網站中最近30天的點擊率排名前5,10或者15的博客。

??通過這兩個例子我們可以看到,使用HTTPServiceWebService無需第三方框架,在服務器端直接編寫相應的服務即可,所以比較容易理解和使用。但是,無論使用HTTPService還是WebService訪問服務器,Flex客戶端和服務器之間傳遞的都是XML數據,客戶端和服務器端處理的也是XML數據。對于企業應用來說,客戶端和服務器端交互的數據量往往很大,因此使用XML作為數據交換格式會降低傳輸效率和轉換效率。同時,處理XML數據的代碼也遠比處理對象的代碼繁瑣,并且難以閱讀和調試。因此,在企業應用開發中,客戶端系統和服務器端系統之間采用HTTPServiceWebService進行通信的部分較少,即使使用這兩個組件。也應當用來傳遞少量、數據格式不易發生變化的數據。

????在企業應用開發中,Flex客戶端與后臺服務器之間的大量通信都是采用RemoteObject完成的。RemoteObject組件在“第三方軟件”的配合下,能夠調用后臺服務器對象上的方法,比如Java對象或者.net對象上的方法,從而實現客戶端與服務器端的通信。在客戶端使用RemoteObject可以直接將ActionScript對象作為調用的參數和返回結果。這一點聽起來似乎有些神奇,但其實也很容易理解:Adobe公司定義了一種二進制數據格式AMF(Action Message Format),用于客戶端與服務器端的數據交互。 ???其實,使用AMF格式交換數據與使用XML進行數據交換的主要區別在于: AMF二進制數據的轉換和傳輸效率更高,同時需要“第三方軟件”用于解釋AMF格式數據。Flex客戶端RemoteObject組件與服務器端通過HTTP協議傳遞AMF格式的二進制數據進行通信的大致過程如下:

????1)客戶端RemoteObject將調用參數中的ActionScript對象序列化為AMF數據格式,然后發出調用請求。

????2)服務器的“第三方軟件“獲取HTTP請求流。

????3)服務器的“第三方軟件”對HTTP請求流進行解析,并且建立響應消息。對HTTP請求流進行解析,解析過程包括解釋AMF格式數據,將ActionScript對象的AMF數據按照事先確定的協議“反序列化”為服務器端對象,比如Java對象,然后用這些參數調用客戶端指定的服務器對象上的方法。

????4)服務器的“第三方軟件,將調用的結果“序列化”為AMF格式的數據流。

????5)服務器發送HTTP響應給Fee客戶端。

????6) Flex客戶端解釋AMF格式數據流,將調用結果序列化為ActionScript對象。

下面我們給出一段Flex官方文檔代碼來展示Rernoteobject對象的使用:

<?xml version="1.0"?>

?

<mx:Application xmlns:mx="http://www.adobe.com/2006/mxml">

????<mx:Script>

????????<![CDATA[

????????????import mx.controls.Alert;

????????????import mx.rpc.remoting.RemoteObject;

????????????import mx.rpc.events.ResultEvent;

????????????import mx.rpc.events.FaultEvent;

????????????[Bindable]

????????????public var empList:Object; ?????????

????????????public var employeeRO:RemoteObject;

????????????public function useRemoteObject(intArg:int, strArg:String):void {

????????????????employeeRO = new RemoteObject();

????????????????employeeRO.destination = "SalaryManager";

????????????????employeeRO.getList.addEventListener("result",

???????????????????????????????????????????????????????????getListResultHandler);

????????????????employeeRO.addEventListener("fault", faultHandler);

????????????????employeeRO.getList(deptComboBox.selectedItem.data);

????????????}

????????????public function getListResultHandler(event:ResultEvent):void {

?????????????????// 遠程調用成功所要完成的處理。

????????????????empList=event.result;

????????????}

?

????????????public function faultHandler (event:FaultEvent):void {

?????????????// 調用失敗所要完成的處理。

????????????????Alert.show(event.fault.faultString, 'Error');

????????????}

????????]]>

????</mx:Script>

????<mx:ComboBox id="deptComboBox"/>

</mx:Application>

?

在上面的代碼中首先看一下函數useRemoteObject這個函數中首先使用語句:

??employeeRO = new RemoteObject();

創建了一個RemoteObject對象employeeRo。然后通過語句:

employeeRO.destination = "SalaryManager";

RemoteObject對象的destination屬性賦值為一個字符串"?SalaryManager?"destination屬性表示遠程對象調用的“目的地’,請求發送到服務器端后,服務器端的“第三方軟件’接收到請求后會檢查配置文件,找到destination值所映射的“服務器端組件”,從而可以調用該組件上的方法。接下來使用語句:

employeeRO.getList.addEventListener("result", getListResultHandler);????

設置遠程服務調用成功時的處理方法,使用語句:

employeeRO.addEventListener("fault", faultHandler);

設置遠程服務調用失敗時的處理方法。最后,使用語句:

employeeRO.getList(deptComboBox.selectedItem.data);

以deptComboBox.selectedItem.data為參數,調用destination屬性所映射的“服務器端組件”的getList方法,這里,“服務端組件”必須有一個名為getList的公開方法,調用“服務器端組件”的getList方法是異步調用,因此它不會阻塞線程來等待調用結果的返回,調用結果的返回時會在getListResultHandler方法中進行處理。在getListlRsultHandler方法中,我們到語句:

empList=event.result;

????該語句表示遠程調用所返回的結果event.result可以直接賦值給ActionScript對象。當然,

后端返回的對象類型與Flex客戶端的ActionScript對象類型要滿足“第三方軟件’所規定的對象類型之間的“映射規則,這樣,Flex就可以把后臺返回的AMF數據流自動地序列化為ActionScript對象。

????從上面的代碼分析中我們可以看出:同傳遞XML方式相比,RemoteObject調用方式直接將ActionScript對象作為調用的參數和返回結果,這對于開發者編程特別方便。同時,通過RemoteObject調用遠程方法需要多寫幾行代碼,但通過精巧的封裝可以很好地解決這個問題。

在上面的論述中多次提到了“第三方軟件”,要使用RemoteObject組件進行遠程方法調用,那么必須在服務器上部署和配置相應的“第三方軟件”。“第三方軟件”有兩個最基本的作用:

??服務器端對象序列化為AMF格式數據和將AMF格式數據反序列化為服務器端對象。

??將客戶端的請求映射為服務器端相應對象上的方法調用。

由于AMF規范已經公開,因此,有很多“第三方軟件”支持不同的后臺服務器端語言,在.net平臺下比較著名的“第三方軟件”為Midnight Coders WebORB

我們所關心的Java平臺“下的“第三方軟件”有Adobe官方商業收費軟件LifeCycle ?Data ?Service(LCDS)Adobe官方開源軟件BlazeDS

BlazeDSLCDS的開源版,只不過BlazeDS不具備LCDS的一些高級功能,比如:

??高級客戶端-服務器數據同步功能。

??沖突檢測/解決。

??Adobe AIR應用的離線數據管理服務。

??RIA生成PDF等。

使用BlazeDSLCDS進行企業應用開發的配置完全一樣,因此,在不需要LCDS高級功能的情況下,完全可以使用BlazeDS替換LCDS作為一種廉價方案,必須使用LCDS高級功能時用戶可以追加投資購買LCDS,因此基于BlazeDS企業應用可以很容易升級為基于LCDS的企業應用,這也是我們選擇BlazeDS作為配合RemoteObject遠程調用的“第三方軟件”的主要原因。

因此,使用Flex+Java開發企業應用,我們主要使用RemoteObject+BlazeDS實現Flex端與Java端的通信。

總之,使用RemoteObject+BlazeDS作為Flex端同Java服務器端的通信框架有如下優點:

??以二進制的AMF協議傳遞數據,轉換和傳輸數據的性能高于XML格式。

??使用RemoteObject+BlazeDS能夠實現Flex對象與Java對象之間的自動轉換,更加有利于開發者編程。

??使用開源框架BlazeDS所開發的企業應用可以更容易地升級為采用高端商業軟件LCDS作為數據通信框架的企業應用。

當然,還有一些其他開源框架,比如Hessian。和Granite也能夠完成與BlazeDS類似的功能,開發者可以根據實際情況加以選擇,但是由于它們的原理相同,所以可以使用相同架構方法和設計模式。

1.2?構建一個簡單的BlazeDS應用

????本節的任務是創建一個RemoteObject應用程序,這個應用程序包括兩部分:前端Flex應用和后端Java應用。兩者通過BlazeDS通信,?開發步驟如下

1.準備軟件環境

在Flex應用中使用Remoting技術時需要以下軟件:?

???MyEclipse 7.5及以上版本。?

??Adobe Flash Builder 4 Plug-in。?

??Tomcat 5.5及以上版本。?

??JDK l.6及以上版本。?

??BlazeDS 3.2及以上版本。?

2.安裝配置軟件

安裝配置軟件需要按照以下順序:?

??安裝MyEclipse 7.5。?

??安裝Adobe Flash Builder 4 Plug-in.

MyEclipse7.5安裝完成后將創建一個Common文件夾和啟動程序所在文件夾(默認命名為“MyEclipse 7.5”)。啟動程序所在文件夾中包含一個dropins文件夾,該文件夾是MyEclipse7.5中安裝Adobe Flash Builder 4 Plug-in時所需的文件夾。選擇“再插入一個Eclipse”項,單擊“選擇”按鈕,選擇MyEclipse啟動程序所在的文件夾進行安裝,如圖5.1.4所示。?

?

5.1.4 安裝Adobe Flash Bulider 4 Plug-in

??安裝BlazeDS 3.2。?

通過http://opensource.adobe.com/wiki/display/blazeds/Downloads地址下載Binary Distribution版本的BlazeDS軟件。該版本為最簡版本,解壓后只包含一個blazeds.war文件。在創建Flex項目時,為了在項目中通過BlazeDS使用Remoting技術,需要定位blazeds.war文件。?

??在MyEclipse 7.5中配置Tomcat和JDK。?

3.開發基于Remoting技術的Flex應用程序

開發基于Remoting技術的Flex應用程序步驟如下:?

??啟動MyEclipse 7.5,單擊右上角的【Flash】按鈕,切換至Flex應用開發視圖。選擇“file’

à“new”à“Flex項目”。彈出“新建Flex項目”對話框,按提示進行操作,如圖5.1.5所示。

?

5.1.5 新建Flex項目

??在圖5.1.5中單擊“Next”按報,彈出“配置J2EE服務器”對話框,按提示進行操作,?如圖5.1.6所示。?

?

5.1.6 配置J2EE服務器

??在圖5.1.6中單擊“Next”按鈕,彈出“新建Flex項目”對話框,為Flex項目設置捌徑,按提示進行操作,如圖5.1.7所示。?

?

5.1.7 Flex項目設置構建路徑

??在圖5.1.7中,單擊【Finish】按鈕,完成創建Flex項目的操作,生成的項目文件結構如圖5.1.8所示。

?

5.1.8 Flex項目文件結構圖

??編輯生成的remotingApp.mxml文件,向其中添加以下代碼:?

<?xml version="1.0" encoding="utf-8"?>

<s:Application xmlns:fx="http://ns.adobe.com/mxml/2009"

???xmlns:s="library://ns.adobe.com/flex/spark"

???xmlns:mx="library://ns.adobe.com/flex/mx" minWidth="955" minHeight="600">

<fx:Declarations>

<!-- 將非可視元素(例如服務、值對象)放在此處 -->

</fx:Declarations>

<s:Panel width="564" height="304" verticalCenter="74" horizontalCenter="17" title="Flex Remoting示例">

<s:Rect left="0" right="0" bottom="0" height="30">

<s:fill>

<s:SolidColor color="#E2EDF7"/>

</s:fill>

</s:Rect>

<s:Button id="btnLoad" x="465" y="242" label="加載遠程數據 "/>

<mx:AdvancedDataGrid id="adg1"

?designViewDataType="flat" width="562" height="241" x="0" y="0">

<mx:columns>

<mx:AdvancedDataGridColumn headerText="商品編號 " dataField="shopId"/>

<mx:AdvancedDataGridColumn headerText="商品名稱 " dataField="shopName"/>

<mx:AdvancedDataGridColumn headerText="商晶單價 " dataField="price"/>

<mx:AdvancedDataGridColumn headerText="商品類別 " dataField="catalog"/>

</mx:columns>

</mx:AdvancedDataGrid>

</s:Panel>

</s:Application>

??將remotingApp項目發布到Tomcat容器中,然后選中“remotingApp.mxml”,單擊右鍵選擇“Run As”à“Web應用程序”,運行Flex應用程序,結果如圖5.1.9所示。

?

5.1.9 運行Flex應用

?

??單擊MyEclipse右上角的【MyEclipse】按鈕,切換至Java應用開發視圖,創建POJO類,命名為“Shoplnfo.java”。?

package com.soft.flex.pojo;

public class ShopInfo

{

private String shopId;//商品編號

private String shopName;//商品名稱private double price;//商品單價

private String catalog;//商品類別

private double price;//商品價格

public ShopInfo (String shopId, String shopName, double price, String catalog) {

super();

this.shopId=shopId;

this.shopName= shopName;

this.price=price;

this.catalog=catalog;

}

public ShopInfo() {

super();

// TODO Auto-generated constructor stub

}

public String getShopId() {

return shopId;

}

public void setShopId(String shopId) {

this.shopId = shopId;

}

public String getShopName() {

return shopName;

}

public void setShopName(String shopName) {

this.shopName = shopName;

}

public String getCatalog() {

return catalog;

}

public void setCatalog(String catalog) {

this.catalog = catalog;

}

public double getPrice() {

return price;

}

public void setPrice(double price) {

this.price = price;

}

?

?

}

??繼續創建業務類,命名為“Service.java”,在其中定義getAllShop方法,用于從數據庫中獲取商品列表并返回商品信息。?

public class Service {

public List<ShopInfo> getAllShop(){

//訪問數據庫代碼

List<ShopInfo> shops = new ArrayList<ShopInfo>();

shops.add(new ShopInfo("S001", "彩電", 1200, "家電"));

shops.add(new ShopInfo("S002", "空調", 1300, "家電"));

shops.add(new ShopInfo("S003", "牛奶", 4.5, "飲料"));

shops.add(new ShopInfo("S004", "可口可樂", 3.8, "飲料"));

return shops;

}

}

??在remoting-config.xml文件中配置遠程調用類Service。使用記事本打開該文件,在根節點中添加一個子節點。?

<destination id="service">

<properties>

<source>com.soft.flex.service.Service</source>

</properties>

</destination>

在remoting-config.xml文件中可以配置多個destination 每個節點代表一個遠程調用類,使用id屬性加以標識?其值不可重復。?Source代表class文件路徑。

??修改remotingApp.mxml文件代碼,在其中創建RemoteObject對象,并使用該對象訪問遠程調用類Service.修改后的代碼如下:?

<?xml version="1.0" encoding="utf-8"?>

<s:Application xmlns:fx="http://ns.adobe.com/mxml/2009"

???xmlns:s="library://ns.adobe.com/flex/spark"

???xmlns:mx="library://ns.adobe.com/flex/mx" minWidth="955" minHeight="600">

<fx:Script>

<![CDATA[

import mx.controls.Alert;

import mx.rpc.events.FaultEvent;

import mx.rpc.events.ResultEvent;

//處理返回結果時間

protected function shopRemotingObject_resultHandler(event:ResultEvent):void

{

//獲取結果并顯示

this.adg1.dataProvider = event.result;

}

//處理訪問錯誤異常信息

protected function shopRemotingObject_faultHandler(event:FaultEvent):void

{

//顯示錯誤信息

Alert.show(event.fault.faultString,"錯誤");

}

?

?

protected function btnLoad_clickHandler(event:MouseEvent):void

{

this.shopRemotingObject.getAllShop();

}

?

]]>

</fx:Script>

<fx:Declarations>

<s:RemoteObject id="shopRemotingObject" destination="service" showBusyCursor="true"

endpoint="http://localhost:8080/remotingApp/messagebroker/amf"

result="shopRemotingObject_resultHandler(event)"

fault="shopRemotingObject_faultHandler(event)">

</s:RemoteObject>

</fx:Declarations>

<s:Panel width="564" height="304" verticalCenter="74" horizontalCenter="17" title="Flex Remoting示例">

<s:Rect left="0" right="0" bottom="0" height="30">

<s:fill>

<s:SolidColor color="#E2EDF7"/>

</s:fill>

</s:Rect>

<s:Button id="btnLoad" x="465" y="242" label="加載遠程數據 " click="btnLoad_clickHandler(event)"/>

<mx:AdvancedDataGrid id="adg1"

?designViewDataType="flat" width="562" height="241" x="0" y="0">

<mx:columns>

<mx:AdvancedDataGridColumn headerText="商品編號 " dataField="shopId"/>

<mx:AdvancedDataGridColumn headerText="商品名稱 " dataField="shopName"/>

<mx:AdvancedDataGridColumn headerText="商晶單價 " dataField="price"/>

<mx:AdvancedDataGridColumn headerText="商品類別 " dataField="catalog"/>

</mx:columns>

</mx:AdvancedDataGrid>

</s:Panel>

</s:Application>

??重新發布remotingApp,再次運行remotingApp.mxml文件,單擊“加載遠程數據”按鈕,?Flex會調用遠程Java類查詢數據庫的商品信息并顯示出來,如圖5.1.10所示。

?

5.1.10

1.3?BlazeDS的序列化機制

????為了傳輸對象,BlazeDS和Flex提供了客戶端ActionScript對象和服務器端Java對象之間的序列化功能;對于Web Service,也提供了客戶端ActionScript對象和SOAP對象之間的序列化功能。

????傳輸對象時,我們一般會選用AMF3來進行編碼。AMF3是Adobe在Flash Player 9之后引入的一種用于序列化ActionScript對象的壓縮的二進制格式。由于它非常緊湊,而且支持廣泛的數據類型和復雜的對象關系,所以在企業應用中被大量使用。

????本節以AMF3編碼規則為例,介紹ActionScript對象和Java對象之間的序列化機制。

????1.元標記RemoteClass和Transient

????Flex提供了兩個元標記來幫助序列化。RemoteClass是修飾類的編譯期的元標記,它[[RemoteClass(alias=””)]的形式定義在ActionScript類前,用于顯式映射其修飾的類和遠程類編譯器在遇到[RemoteClass]元標記時,會在其所在應用(Application)或模塊(Module)的初始化代碼中插入flash.net.registerClas sAlias( aliasName,classObject)調用以注冊ActionScript類及其遠程別名。而Transient則是運行期元標記,修飾類的成員變量,用于表明成員變量是瞬態變量,不參與序列化。

????有兩點必須注意:其一,使用[RemoteClass]修飾的ActionScript類必須在代碼中被引用或使用(也就是說,不是孤立的類),否則,從Java對象無法轉換到期望的ActionScript對象。編譯成library的類沒有此限制,因為不論其是否被引用,FlexBuilder都會編譯它;其二,除了[Transient]修飾的成員變量外,非公開變量或屬性、只讀屬性(只有get訪問函數)和靜態變量屬性也不參與序列化,這一點對于ActionScript和Java類是一致的。

??示例5.1給出了兩者的使用范例,代碼第4行通過[RemoteClass]元標記顯式地將自定義ActionScript靜態對象Employee映射到同名的Java對象。代碼第10行使用了元標記[Transient]修飾Employee的成員變量age。

??經過這兩個元標記修飾后,當Flex調用遠程方法傳遞Employee時,BlazeDS會將其轉換成同名的Java對象,但age變量會被Flex序列化機制忽略,反之亦然。

?

示例5.1 Employee.as

???package ?com. flexbook.blazeds

?????{

?????????????[ Bindable ]

???????????????[ Remo teClas s ( alias = " cam. flexbook.blazeds .Employee " ) ?]

????public class Employee

???????????{

???????public var name:String;

???????public var code:String;

???????public var birthday:Date;

??????[Transientl

?????public var age:uint;

????}

?}

?

??2.從ActionScript對象到Java對象

??當Flex應用程序通過RemoteObject調用遠程Java方法時,方法的參數會被自動從ActionScript對象轉換成Java對象。這個過程經歷了兩個階段,首先,Flash PlayerActionScript對象編碼成AMF3格式,然后,BlazeDS將AMF3格式的數據流轉換成Java對象。

??ActionScript中有些類型,如int、Boolean和String,與Java類型精確匹配,而uint和Number 則沒有相應的Java類型與之對應。表4-1-1列出了從ActionScript對象轉換到Java對象時的類型對應關系。

4-1-1 ?ActionScript對象轉換到Java對象時的類型對應關系

ActionScript 類型 (AMF 3)

反序列化為 Java

支持的 Java 類型綁定

Array(密集)

java.util.List

java.util.Collection, Object[ ] (本機數組)

如果類型是一個接口,則會映射到下面的接口實現:

List 變為 ArrayList

SortedSet 變為 TreeSet

Set 變為 HashSet

Collection 變為 ArrayList

自定義 Collection 實現的新實例會綁定到該類型。

Array(稀疏)

java.util.Map

java.util.Map

Boolean

字符串"true""false"

java.lang.Boolean

Booleanboolean String

Flash.utils.ByteArray

byte []

?

Flash.utils.IExternalizable

java.io.Externalizable

?

Date

java.util.Date

(已設置為協調世界時 (UTC) 格式)

java.util.Datejava.util.Calendarjava.sql.Timestampjava.sql.Time java.sql.Date

int/uint

java.lang.Integer

java.lang.Doublejava.lang.Longjava.lang.Floatjava.lang.Integerjava.lang.Shortjava.lang.Bytejava.math.BigDecimaljava.math.BigIntegerString,以及基元類型 doublelongfloatintshort byte

Null

null

基元

Number

java.lang.Double

java.lang.Doublejava.lang.Longjava.lang.Floatjava.lang.Integerjava.lang.Shortjava.lang.Bytejava.math.BigDecimaljava.math.BigIntegerString0(零)

如果發送了 null,則為基元類型 doublelongfloatintshort byte

Object(泛型)

java.util.Map

如果指定了 Map 接口,則為 java.util.Map 創建一個新的 java.util.HashMap,為 java.util.SortedMap 創建一個新的 java.util.TreeMap

String

java.lang.String

java.lang.Stringjava.lang.Booleanjava.lang.Numberjava.math.BigIntegerjava.math.BigDecimalchar[]、以及任何基元數字類型

有類型對象

有類型對象

在使用 [RemoteClass] 元數據標簽指定遠程類名稱時。Bean 類型必須具有公共的無參數構造函數。

有類型對象

undefined

null

null(對于對象)和默認值(對于基元)

XML

org.w3c.dom.Document

org.w3c.dom.Document

XMLDocument

(舊 XML 類型)

org.w3c.dom.Document

org.w3c.dom.Document

可以針對在 services-config.xml 文件中定義的任何通道啟用對于 XMLDocument 類型的舊 XML 支持。此設置僅在將數據從服務器發回到客戶端時很重要,它控制 org.w3c.dom.Document 實例如何發送到 ActionScript

?

??當然,BlazeDS在Java對象中尋找合適的方法簽名時會嘗試對Java類型做出兼容的轉換。比如,Flex應用在調用遠程方法時傳人一個int類型的參數,但遠程Java對象只有一個接受參數的方法,這時,BlazeDS將嘗試將這個int轉換成java.lang.String,然后再調用方法。

??ActionScript中的Array允許兩種方式索引元素,嚴格數組(strict array)使用數字作為索引,索引代表了元素的排列位置,關聯數組(associative array)使用字符串作為索引,索引代表了元素的名稱。一個數組中只要有一個元素使用字符串作為索引,那么它就是關聯數組,這時,數組實際上退化成了ActionScript的動態對象。在嚴格數組中,我們把索引不是從0開始或者索引不連續的數組稱為稀疏數組。關聯數組通過序列化將轉換成java.util.Map,稀疏數組也被轉換成java.util.Map以避免傳遞大量null元素。

對于ActionScriptString類型,由于可以匹配的JavaString類型,因此優先轉換成字符串,但如果遠程Java對象中沒有方法的簽名能夠匹配,BlazeDS將嘗試將字符串轉換成Boolean(如果字符串是true,false)或數值類型(如果字符串表示一個數值)

如果將ActionScriptnull 或者undefined傳給遠程Java方法,他將會被轉化成null(如果目標類型是java.lang.Object或其子類)或轉換成基本類型的默認值(如果目標類型是Java中的基本類型)

3 Java對象到ActionScript對象

當服務器需要返回Java對象時,BlazeDS會將Java對象編碼成AMF3格式,并序列化到Flex應用端,Flex應用解析AMF3格式的流數據生成ActionScript對象。表4-1-2列出了Java對象轉換成ActionScript對象的類型對應關系。

4-1-2 Java對象轉換成ActionScript對象的類型對應關系。

?

Java 類型

ActionScript 類型 (AMF 3)

java.lang.String

String

java.lang.Boolean, boolean

Boolean

java.lang.Integer, int

int

如果值小于 0xF0000000 且大于 0x0FFFFFFF,則會按照 AMF 編碼要求將值提升為 Number

java.lang.Short, short

int

如果 i 小于 0xF0000000 且大于 0x0FFFFFFF,則會將值提升為 Number

java.lang.Byte, byte[]

int

如果 i 小于 0xF0000000 且大于 0x0FFFFFFF,則會將值提升為 Number

java.lang.Byte[]

flash.utils.ByteArray

java.lang.Double, double

Number

java.lang.Long, long

Number

java.lang.Float, float

Number

java.lang.Character, char

String

java.lang.Character[], char[]

String

java. math.BigInteger

String

java.math.BigDecimal

String

java.util.Calendar

Date

日期按照協調世界時 (UTC) 時區的時間進行發送。客戶端和服務器必須根據時區相應地調整時間。

java.util.Date

Date

日期按照 UTC 時區的時間進行發送。客戶端和服務器必須根據時區相應地調整時間。

java.util.Collection(例如,java.util.ArrayList

mx.collections.ArrayCollection

java.lang.Object[]

Array

java.util.Map

Object(無類型)。例如,將 java.util.Map[] 轉換為對象的 Array

java.util.Dictionary

Object(無類型)

org.w3c.dom.Document

XML 對象

Null

null

java.lang.Object(以前列出的類型除外)

有類型 Object

通過使用 JavaBean 內部檢查規則將對象進行序列化,并且對象包括公共字段。不包括靜態字段、瞬態字段、非公共字段,以及非公共 bean 屬性或靜態 bean 屬性。

?

如果沒有使用[RemoteClass]標簽,則轉換成動態對象,否則轉換成自定義的靜態對象。

4.自定義序列化機制

?以上討論的是BlazeDS的標準序列化機制。如果標準規則不能滿足要求,BlazeDS還提供了擴展機制,允許編寫代碼自定義序列化規則。在Flex端,我們可以使目標類實現接口flash.net.IExternalizable,在Java端實現接口java.io.Externalizable。

??自定義序列化機制有很多應用場景,比如壓縮數據、隱藏敏感數據等。示例5.2定義了一個DataRow類,它是所有數據行對象的基類,每個數據行都有一個唯一標示符rowID,通過rowID客戶端和服務器端可以識別它們操作的對象,通常,我們期望rowID由服務器端負責生成,并且一旦分配給對象就不能被外部更改,因此它需要被定義成只讀屬性。而BlazeDS標準的序列化機制是不序列化只讀屬性的,但rowID是如此重要,以至于如果不傳遞給客戶端,那么在客戶端處理完DataRow后,服務器端就不知道是哪個DataRow對象被處理了。

??自定義序列化機制可以幫助實現我們的愿望:讓DataRow實現接口IExternalizable,然后在WriteExternal和readExternal中分別向序列化流寫入和從序列化流讀出rowID。這樣,即使我們將rowID定義成只讀屬性,絲毫不影響rowID的序列化。

示例5.2DataRow.as

package com.flexbook.blazeds

?{

???import flash.utils.IDataInput;

???import ?flash.utils .IDataOutput ;

???import ??flash.utils .IExternalizable ;

?

??[ RemoteClass ( alias = " cam. flexbook.blazeda .DataRow " ) ]

public class DataRow implemente IExternalizable

?{

??private var _rowID:Object;

??public function DataRow() {

?????????}

???public function get rowID():Object{

???????return _rowID;

?????????}

???public function writeExternal(output:IDataOutput)

???????????????output.writeObject (_rowID) ;

????????}

????public function readExternal(input:IDataInput) {

?????????????????rowID=input.readObject ( ) ,

?

?}

}

????flash.utils.IDataInput和flash.utils.IDataOutput代表了序列化的輸入流和輸出流,當Flex序列化對象時,會調用對象的writeExternal,并傳入IDataOutput以便對象輸出其屬性;當Flex反序列化對象時,則調用對象的readExternal,并傳入IDatalnput以便對象從流中讀取其屬性IDataInput和IDataOutput提供了讀取和寫入各種類型ActionScript對象的函數,來幫助我們序列化和反序列化。

????示例5.3DataRow對象在服務器端的定義,它與前端代碼基本相似,唯一不同的是它有兩個構造函數,默認構造函數用于反序列化(因為反序列化必須要有無參構造函數),:另一構造函數用于服務器端創建DataRow對象時為它指定rowID(這也是可以修改rowID的唯一的機會)。

?

示例5.3 ?DataRow.java


public class DataRow implements Externalizable{

?

private Object rowID,

????public ?DataRow() ?{

????super();

????) ???

????public DataRow(Object rowID) ?{

????super();

????this.rowID=rowID;

????)

????public ?Object ?getRowID()(

????return rowID;

????)

????public void readExternal(ObjectInput in) throws IOException,

????ClassNotFoundException {

????rowID=in.readObject();

????)

????public voicl writeExternal(ObjectOutput.ut) throws IOException ?{

????out.writeObiect(rowID);

????)

????)


????如此,我們就得到了一個安全的DataRow對象,除了構造時可以為它分配rowID,其他時毫無論在服務器端還是客戶端都無法對rowID進行修改。從它繼承的類都可以獲得這項好處,前提是服務器端的Java類和客戶端的ActionScript都需要繼承相應的DataRow。

1.4?BlazeDSSpring整合

如果需要BlazeDsSpring框架整合使用,是非常簡單的事情,因為,SpringSourceAdobe已經合作為BlazeDS提供了Spring支持,即Spring BlazeDS Integration,關于這個項目的更多信息可以訪問http://www.springsource.org/spring-flex

Spring BlazeDS Integration SpringSource 的開源項目,用于整合 Spring BlazeDS。不使用 Spring BlazeDS Integration 同樣可以整合 Spring BlazeDS。但這種整合方式不自然,需要額外維護一個 BlazeDS 配置文件,Spring BlazeDS Integration 會改善這種處境。

Spring BlazeDS Integration 需要的軟件環境:

??Java 5 或更高

??Spring 2.5.6 或更高

??BlazeDS 3.2 或更高

Spring BlazeDS Integration 特征

??MessageBrokerBlazeDS 的核心組件)被配置為 Spring 管理的 Bean

??Flex 客戶端發出的 HTTP 消息通過 Spring DispatcherServlet 路由給 MessageBroker

??Remote objects Spring 的方式配置在 Spring 配置文件內

?

下面我們演示BlazeDSSpring的整合。

1準備所需 jar

下載 Spring Framework dependenciesSpring BlazeDS Integration解壓備用,在項目中添加Spring支持,并將以下 2 部分 jar 包拷貝到項目的 lib 下:

??Spring Framework dependencies

org.aopalliance 內的 com.springsource.org.aopalliance-1.0.0.jar

edu.emory.mathcs.backport 內的 com.springsource.edu.emory.mathcs.backport-3.0.0.jar

net.sourceforge.cglib 內的 com.springsource.net.sf.cglib-2.2.0.jar

??Spring BlazeDS Integration

org.springframework.flex-1.0.3.RELEASE.jar

2):修改 web.xml 文件

web.xml 內所有 Flex 相關配置刪除掉,添加以下內容(改用 Spring web 應用的前端控制器處理所有應用請求)

?

<servlet>

??<servlet-name>Spring MVC Dispatcher Servlet</servlet-name>

??<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>

??<init-param>

????<param-name>contextConfigLocation</param-name>

????<param-value>/WEB-INF/web-application-config.xml</param-value>

??</init-param>

??<load-on-startup>1</load-on-startup>

</servlet>

?

<servlet-mapping>

??<servlet-name>Spring MVC Dispatcher Servlet</servlet-name>

??<url-pattern>/messagebroker/*</url-pattern>

</servlet-mapping>

3):配置 web-application-config.xml

1)創建應用上下文配置文件 web-application-config.xml

?

<?xml version="1.0" encoding="UTF-8"?>

<beans xmlns="http://www.springframework.org/schema/beans"

???????xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

???????xsi:schemaLocation="http://www.springframework.org/schema/beans ??????http://www.springframework.org/schema/beans/spring-beans-3.0.xsd">

?

</beans>

2)為了使用 Spring BlazeDS Integration tag,增加命名空間

?

<?xml version="1.0" encoding="UTF-8"?>

<beans xmlns="http://www.springframework.org/schema/beans"

???????xmlns:flex="http://www.springframework.org/schema/flex"

???????xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

???????xsi:schemaLocation="http://www.springframework.org/schema/beans ??????http://www.springframework.org/schema/beans/spring-beans-3.0.xsd ??????http://www.springframework.org/schema/flex ??????http://www.springframework.org/schema/flex/spring-flex-1.0.xsd">

?

</beans>

3)為了把請求路由給 MessageBroker,添加以下 tag

?

<flex:message-broker />

4)定義 Bean,并用 remoting-destination tag 把它暴露給 Flex

?

<bean id="employeeServiceDest" class="com.sample.EmployeeService">

??<flex:remoting-destination />

</bean>

?

至此BlazeDSSpring的整合就完成了。

?

1.5?BlazeDS的消息服務

BlazeDS消息服務(Message Service )提供發布(publish)/訂閱(subscribe)機制,允許 Flex 應用程序發布消息、訂閱消息終端(messaging destination),從而實現實時數據的推動和協作傳送。

?

1Message Service

?

Message Service 提供發布(publish)/訂閱(subscribe)機制允許Flex 應用 程序發布消息、訂閱消息終端(messaging destination),從而實現數據的實時 推動和協作傳送。

消息終端在messaging-config.xml配置,其中頻道(channel)是其關鍵元素, 它用來實現客戶端和服務器端交換數據。使用BlazeDS,消息終端通常用作 streaming頻道或者polling頻道。

使用streaming頻道,服務器端會一直響應HTTP請求直到該頻道連接被關閉, 它允許服務器向客戶端不斷傳送大量的數據。因為HTTP連接是獨一無二的,這實 現數據的雙向傳送,每個streaming AMF或者HTTP頻道事實上需要兩個瀏覽器 HTTP連接, 一個連接需要不斷處理服務器端與頻道緊密相關的客戶端的響應。 另外需要一個短暫連接,只有當數據需要傳送到服務器時,它才脫離瀏覽器連接 池;當短暫連接不再需要時,它立即被釋放回瀏覽器連接池。

polling頻道可以通過簡單的時間間隔或者使用服務器等待來配置,如果數據 不馬上可用 (長輪循)的話。另外,每次輪循響應完成請求。默認下瀏覽器HTTP 1.1的連接是持續的,瀏覽器輪循已有的連接,發送并發的輪循請求,以此來減 輕輪循的開銷。

當需要準實時通信時,streaming 頻道是最好選擇。

?

2IE Firefox瀏覽器下的不同

瀏覽器對每個session都有連接數限制。不同的瀏覽器,連接最大數以及對 session的處理方式都不一樣。

IE中每個session的最大連接數為2。 但如果從開始菜單或快捷方式打開多個 IE實例,每個IE實例開啟不同的進程并擁有各自session。另外,如果我們通過 CTRL+N 開啟對已有的IE實例一個新的IE窗口,該窗口將與創建它的IE實例共用 一個session 。也就是說,如果程序實例開啟不同的進程,我們可以通過HTTP streaming建立不限量應用取得服務器端數據;如果通過CTRL+N開啟多個窗口, 每個session最多建立2個連接。

Firefox中每個session最多建立8個連接。如果從開始菜單或快捷方式打開多 個Firefox實例,所有實例開啟使用同一進程并共用一個session。既然瀏覽器對 普通的HTTP請求通常只需要一個連接, 理論上我們可以最多可以建立7HTTP streaming連接。

?

3messaging-config.xml

另外,如果每個session到達最大連接數,使用streaming channel連接到服務器的下一次嘗試將失敗并拋出以下異常:

Endpoint with id 'my-streaming-amf' cannot grant streaming connection to FlexClient with id 'D640B86F-6B1D-92DF- 8288-1B737A371AFE' because max-streaming-connections-per-session limit of '1' has been reached

不過,BlazeDS提供一種優雅的退后機制來處理這種情況:

客戶端始終會嘗試使用頻道表(messaging-config.xml中為服務終端定義) 中的第一個頻道來連接。如果該連接失敗, 客戶端將自動退后到頻道表中的下一頻道。我們可以為所有的服務終端定義了如下默認的ChannelSet

?

<default-channels>

<channel ref="my-streaming-amf"/>

<channel ref="my-polling-amf"/>

</default-channels>

?

也就是說,客戶端應用會首先嘗試使用streaming channel連接,如果連接失 敗會使用polling channel

?

在客戶端,Flex提供了 ProducerConsumer這兩個組件,讓你用來向目標地址發送或訂閱消息。如果要訂閱消息,你就使用Consumer類的 subscribe()方法。當有消息發送到你訂閱了的目標地址時,Consumer上就會觸發message事件。

示例5.4

客戶端代碼:

<?xml version="1.0" encoding="utf-8"?>

<s:Application xmlns:fx="http://ns.adobe.com/mxml/2009"

???xmlns:s="library://ns.adobe.com/flex/spark"

???creationComplete="consumer.subscribe();"

???xmlns:mx="library://ns.adobe.com/flex/mx" minWidth="955" minHeight="600">

<fx:Script>

<![CDATA[

import mx.controls.Alert;

import mx.messaging.events.MessageFaultEvent;

import mx.messaging.messages.AsyncMessage;

import mx.messaging.messages.IMessage;

private function send():void ?

{ ??

???var message:IMessage?= new AsyncMessage(); ??

???message.body.chatMessage = msg.text+consumer.clientId; ??

????producer.send(message); ??

????msg.text = ""; ??

???????

??} ??

??????????

???private function messageHandler(message:IMessage):void ?

???{ ??

????log.text += message.body.chatMessage + "\n"; ???

???} ??

?

?

protected function consumer_faultHandler(event:MessageFaultEvent):void

{

Alert.show(event.faultDetail);

}

?

?

protected function producer_faultHandler(event:MessageFaultEvent):void

{

Alert.show(event.faultDetail);

}

?

]]>

</fx:Script>

<fx:Declarations>

<mx:ChannelSet id="cs"> ??

??<mx:StreamingAMFChannel url="http://localhost:8400/MsgService/messagebroker/streamingamf"/> ??

??</mx:ChannelSet> ??

??<mx:Producer id="producer" ?fault="producer_faultHandler(event)" destination="chat" channelSet="{cs}"/> ??

??<mx:Consumer id="consumer" destination="chat" channelSet="{cs}" message="messageHandler(event.message)" fault="consumer_faultHandler(event)"/> ??

?

</fx:Declarations>

<s:Panel title="Chat Test!" ??x="20" y="19" width="518" height="295"> ??

??<s:TextArea id="log" x="19" y="11" width="473" height="166"/> ??

?????

??<s:TextInput id="msg" x="19" y="191" width="377" height="46" enter="send()"/> ??

??<s:Button x="411" y="192" label="發送消息" height="43" width="77" click="send()"/> ??

????

?</s:Panel> ??

?

</s:Application>

?

服務器端services-config.xml定義Streaming通道:

<?xml version="1.0" encoding="UTF-8"?>

<service id="message-service"

????class="flex.messaging.services.MessageService">

?

????<adapters>

????????<adapter-definition id="actionscript" class="flex.messaging.services.messaging.adapters.ActionScriptAdapter" default="true" />

???????<adapter-definition id="jms" class="flex.messaging.services.messaging.adapters.JMSAdapter"/>

????</adapters>

?

??? <!-- ??<default-channels>

????????<channel ref="my-polling-amf"/>

????</default-channels>-->

<destination id="chat"> ??

??????????

????????<properties> ??

???????????<network> ??

????????????????<session-timeout>0</session-timeout> ??

????????????</network> ??

????????????<server> ??

???????????????<max-cache-size>1000</max-cache-size> ??

????????????????<message-time-to-live>0</message-time-to-live> ??

????????????????<durable>false</durable> ??

???????????</server> ??

????????</properties> ??

????????<channels> ??

????????<channel ref="my-streaming-amf" /> ??

????????</channels> ??

????</destination> ??

?

</service>

?

服務器端messaging-config.xm中定義目標并指定通道:

<?xml version="1.0" encoding="UTF-8"?>

<services-config>

?

????<services>

????????<service-include file-path="remoting-config.xml" />

????????<service-include file-path="proxy-config.xml" />

????????<service-include file-path="messaging-config.xml" /> ???????

???

?

?????

??????</services>

????<security>

????????<login-command class="flex.messaging.security.TomcatLoginCommand" server="Tomcat"/>

????????<!-- Uncomment the correct app server

????????<login-command class="flex.messaging.security.TomcatLoginCommand" server="JBoss">

<login-command class="flex.messaging.security.JRunLoginCommand" server="JRun"/> ???????

????????<login-command class="flex.messaging.security.WeblogicLoginCommand" server="Weblogic"/>

????????<login-command class="flex.messaging.security.WebSphereLoginCommand" server="WebSphere"/>

????????-->

?

????????<!--

????????<security-constraint id="basic-read-access">

????????????<auth-method>Basic</auth-method>

????????????<roles>

????????????????<role>guests</role>

????????????????<role>accountants</role>

????????????????<role>employees</role>

????????????????<role>managers</role>

????????????</roles>

????????</security-constraint>

?????????-->

????</security>

?

????<channels>

?

?<channel-definition id="my-streaming-amf" class="mx.messaging.channels.StreamingAMFChannel"> ??

????????????<endpoint url="http://localhost:8400/MsgService/messagebroker/streamingamf" class="flex.messaging.endpoints.StreamingAMFEndpoint"/> ??

????????????<properties> ??

???????????????<idle-timeout-minutes>0</idle-timeout-minutes> ??

????????????????<max-streaming-clients>50</max-streaming-clients> ??

????????????????<server-to-client-heartbeat-millis>5000</server-to-client-heartbeat-millis> ??

???????????????<user-agent-settings> ??

????????????????????<user-agent match-on="MSIE" kickstart-bytes="2048" max-streaming-connections-per-session="1"/> ??

???????????????????<user-agent match-on="Firefox" kickstart-bytes="2048" max-streaming-connections-per-session="1"/> ??

????????????????</user-agent-settings> ??

???????????</properties> ??

????????</channel-definition>

????????<channel-definition id="my-amf" class="mx.messaging.channels.AMFChannel">

????????????<endpoint url="http://{server.name}:{server.port}/{context.root}/messagebroker/amf" class="flex.messaging.endpoints.AMFEndpoint"/>

????????</channel-definition>

?

????????<channel-definition id="my-secure-amf" class="mx.messaging.channels.SecureAMFChannel">

????????????<endpoint url="https://{server.name}:{server.port}/{context.root}/messagebroker/amfsecure" class="flex.messaging.endpoints.SecureAMFEndpoint"/>

????????????<properties>

????????????????<add-no-cache-headers>false</add-no-cache-headers>

????????????</properties>

????????</channel-definition>

?

????????<channel-definition id="my-polling-amf" class="mx.messaging.channels.AMFChannel">

????????????<endpoint url="http://{server.name}:{server.port}/{context.root}/messagebroker/amfpolling" class="flex.messaging.endpoints.AMFEndpoint"/>

????????????<properties>

????????????????<polling-enabled>true</polling-enabled>

????????????????<polling-interval-seconds>4</polling-interval-seconds>

????????????</properties>

????????</channel-definition>

?

??????

????????<channel-definition id="my-http" class="mx.messaging.channels.HTTPChannel">

????????????<endpoint url="http://{server.name}:{server.port}/{context.root}/messagebroker/http" class="flex.messaging.endpoints.HTTPEndpoint"/>

????????</channel-definition>

??<!--

????????<channel-definition id="my-secure-http" class="mx.messaging.channels.SecureHTTPChannel">

????????????<endpoint url="https://{server.name}:{server.port}/{context.root}/messagebroker/httpsecure" class="flex.messaging.endpoints.SecureHTTPEndpoint"/>

????????????<properties>

????????????????<add-no-cache-headers>false</add-no-cache-headers>

????????????</properties>

????????</channel-definition>

????????-->

????</channels>

?

????<logging>

????????<target class="flex.messaging.log.ConsoleTarget" level="Error">

????????????<properties>

????????????????<prefix>[BlazeDS] </prefix>

????????????????<includeDate>false</includeDate>

????????????????<includeTime>false</includeTime>

????????????????<includeLevel>false</includeLevel>

????????????????<includeCategory>false</includeCategory>

????????????</properties>

????????????<filters>

????????????????<pattern>Endpoint.*</pattern>

????????????????<pattern>Service.*</pattern>

????????????????<pattern>Configuration</pattern>

????????????</filters>

????????</target>

????</logging>

?

????<system>

????????<redeploy>

????????????<enabled>false</enabled>

????????????<!--

????????????<watch-interval>20</watch-interval>

????????????<watch-file>{context.root}/WEB-INF/flex/services-config.xml</watch-file>

????????????<watch-file>{context.root}/WEB-INF/flex/proxy-config.xml</watch-file>

????????????<watch-file>{context.root}/WEB-INF/flex/remoting-config.xml</watch-file>

????????????<watch-file>{context.root}/WEB-INF/flex/messaging-config.xml</watch-file>

????????????<watch-file>{context.root}/WEB-INF/flex/data-management-config.xml</watch-file>

????????????<touch-file>{context.root}/WEB-INF/web.xml</touch-file>

?????????????-->

????????</redeploy>

????</system>

?

</services-config>

運行應用效果如圖5.1.11所示,輕松實現了數據的推送:

?

?

5.1.11 Flex數據推送


任務實訓部分?

?

實訓任務1:使用HttpServcie方式與后臺通信

訓練技能點

HttpServcie

需求說明

使用HttpService對象開發Flex應用程序 按條件查詢數據庫中某張表的數據并顯示在表格中。

實現思路

1)創建Flex項目,將此項目的服務器技術選擇為J2EE服務器(無需使用遠程對象訪問服務)。

2)切換到MyEclipese java 開發視圖,創建一個POJO類用于描述奧運會各個國家獲得的獎牌情況。

package com.soft.flex.flex4sj.pojo;

?

public class Cup {

private int id;

//國家代號

private String countryId;

//國家名稱

private String countryName;

//金牌數

private int goldMedal;

//銀牌數

private int silverMedal;

//銅牌數

private int bronzeMedal;

public int getId() {

return id;

}

public void setId(int id) {

this.id = id;

}

public String getCountryId() {

return countryId;

}

public void setCountryId(String countryId) {

this.countryId = countryId;

}

public String getCountryName() {

return countryName;

}

public void setCountryName(String countryName) {

this.countryName = countryName;

}

public int getGoldMedal() {

return goldMedal;

}

public void setGoldMedal(int goldMedal) {

this.goldMedal = goldMedal;

}

public int getSilverMedal() {

return silverMedal;

}

public void setSilverMedal(int silverMedal) {

this.silverMedal = silverMedal;

}

public int getBronzeMedal() {

return bronzeMedal;

}

public void setBronzeMedal(int bronzeMedal) {

this.bronzeMedal = bronzeMedal;

}

public Cup(String countryId, String countryName, int goldMedal,

int silverMedal, int bronzeMedal) {

super();

this.countryId = countryId;

this.countryName = countryName;

this.goldMedal = goldMedal;

this.silverMedal = silverMedal;

this.bronzeMedal = bronzeMedal;

}

public Cup() {

super();

// TODO Auto-generated constructor stub

}

}

(3)創建業務類,在該類 中定義根據國家查詢獲取獎牌情況的方法。

package com.soft.flex.flex4sj.service;

?

?

import java.util.*;

?

import com.soft.flex.flex4sj.pojo.Cup;

?

?public class CupService {

private List<Cup> cupList;

//模擬數據庫數據

public CupService(){

cupList ?= new ArrayList();

Cup cup1 = new Cup("china", "中國",30 ,20 ,10 );

Cup cup2 = new Cup("america", "美國",20 ,23 ,12 );

Cup cup3 = new Cup("japan", "日本",25 ,27 ,15 );

Cup cup4 = new Cup("france", "法國",10 ,18 ,20 );

Cup cup5 = new Cup("russia", "俄羅斯",16 ,30,25 );

Cup cup6 = new Cup("singapore", "新加坡",12 ,25 ,18 );

cupList.add( cup1);

cupList.add( cup2);

cupList.add( cup3);

cupList.add( cup4);

cupList.add( cup5);

cupList.add( cup6);

}

public List getAll(){

return cupList;

}

public Cup getCupByCountryId(String countryId){

Cup c = null;

for(Cup cup : cupList){

if(countryId.equals(cup.getCountryId())){

c = cup;

break;

}

}

return c;

?

}

?

?

}

?

(4)創建servlet 路徑為/query, servlet根據傳入的cid 調用業務類獲取結果,并轉化為xml格式返回。

public class QueryServlet extends HttpServlet {

?

public void doGet(HttpServletRequest request, HttpServletResponse response)

throws ServletException, IOException {

doPost(request, response);

}

public void doPost(HttpServletRequest request, HttpServletResponse response)

throws ServletException, IOException {

?

response.setContentType("text/html;charset=utf-8");

PrintWriter out = response.getWriter();

//構造xml頭標記

String xml="<?xml version=\"1.0\" encoding=\"utf-8\"?><cups>";

???? String method = request.getParameter("method");

???? CupService service = new CupService();

???? //如果是查詢所有

???? if("all".equals(method)){

???? List<Cup> list = service.getAll();

????????

????????

????????

???????? for(Cup cup : list){

????????

???????? String cid = cup.getCountryId();

???????? String cname = cup.getCountryName();

???????? int gm = cup.getGoldMedal();

???????? int sm = cup.getSilverMedal();

???????? int bm = cup.getBronzeMedal();

???????? //構造xml節點

???????? xml += "<cup countryId='"+cid+"' countryName='"+cname+"' goldMedal='"+gm+"' silverMedal='"+sm+"' bronzeMedal='"+bm+"'/>";

????????

????????

???????? }

????

???? }else{

???? String countryId = ?request.getParameter("cid");

???? Cup cup = service.getCupByCountryId(countryId);

???? String cid = cup.getCountryId();

???? String cname = cup.getCountryName();

???? int gm = cup.getGoldMedal();

???? int sm = cup.getSilverMedal();

???? int bm = cup.getBronzeMedal();

????

???? xml += "<cup countryId='"+cid+"' countryName='"+cname+"' goldMedal='"+gm+"' silverMedal='"+sm+"' bronzeMedal='"+bm+"'/>";

????

????

???? }

????

????

????

????

????

???? xml+="</cups>";

???? //輸出結尾標志

???? out.print(xml);

out.flush();

out.close();

}

?

}

?

(5)創建MXML界面,通過HttpService對象訪問servlet 并獲取查詢結果。

<?xml version="1.0" encoding="utf-8"?>

<s:Application xmlns:fx="http://ns.adobe.com/mxml/2009"

???xmlns:s="library://ns.adobe.com/flex/spark"

???xmlns:mx="library://ns.adobe.com/flex/mx" minWidth="955" minHeight="600"

???initialize="application1_initializeHandler(event)"

???>

<fx:Script>

<![CDATA[

import mx.controls.Alert;

import mx.events.FlexEvent;

import mx.rpc.events.FaultEvent;

import mx.rpc.events.ResultEvent;

import spark.events.IndexChangeEvent;

private var flag:int;

protected function application1_initializeHandler(event:FlexEvent):void

{

this.flag=0;

this.myhttp.url = "/sj41/query?method=all&cid=all";

this.myhttp.send();

}

?

?

protected function myhttp_resultHandler(event:ResultEvent):void

{

//如果是初始化時返回的結果

if(flag==0){

this.ddl.dataProvider=event.result.cups.cup;

this.ddl.labelField = "countryName";

this.adg1.dataProvider=event.result.cups.cup;

}

else{//如果是選擇下拉列表返回的結果

this.adg1.dataProvider=event.result.cups.cup;

}

}

?

?

protected function myhttp_faultHandler(event:FaultEvent):void

{

Alert.show(event.fault.faultString);

}

?

?

protected function ddl_changeHandler(event:IndexChangeEvent):void

{

var cid:String = this.ddl.selectedItem.countryId;

this.flag=1;

this.myhttp.url = "/sj41/query?method=getById&cid="+cid;

this.myhttp.send();

}

?

]]>

</fx:Script>

<fx:Declarations>

<s:HTTPService id="myhttp" ?showBusyCursor="true" result="myhttp_resultHandler(event)" fault="myhttp_faultHandler(event)"/>

</fx:Declarations>

<s:Panel width="557" height="376" title="使用HttpService 與后臺通信" horizontalCenter="0" verticalCenter="0">

<mx:AdvancedDataGrid x="2" y="9" id="adg1" designViewDataType="flat" width="545" height="274">

<mx:columns>

<mx:AdvancedDataGridColumn headerText="代號" dataField="countryId"/>

<mx:AdvancedDataGridColumn headerText="國家" dataField="countryName"/>

<mx:AdvancedDataGridColumn headerText="金牌數" dataField="goldMedal"/>

<mx:AdvancedDataGridColumn headerText="銀牌數" dataField="silverMedal"/>

?

<mx:AdvancedDataGridColumn headerText="銅牌數" dataField="bronzeMedal"/>

?

</mx:columns>

</mx:AdvancedDataGrid>

<s:Label x="216" y="305" text="選擇國家:" height="23" verticalAlign="middle"/>

?

<s:Button x="475" y="305" label="顯示全部" click="application1_initializeHandler(event as FlexEvent)"/>

<s:DropDownList x="306" y="305" id="ddl" prompt="請選擇國家" change="ddl_changeHandler(event)"></s:DropDownList>

</s:Panel>

</s:Application>

?

(6)運行應用程序,效果如圖5.2.1所示。

?

5.2.1 HttpService 示例

實訓任務2:使用RemotingObject 與后臺通信

訓練技能點

RemotingObject

需求說明

使用RemotingObject 重構任務1

實現思路:

(1)創建Flex項目,將此項目的服務器技術選擇為J2EE服務器(使用遠程對象訪問服務)。 如圖5.2.2所示。

?

5.2.2 創建Flex項目

2)修改WebRoot/WEB-INFO/flex/remoting-config.xml,配置業務類

<?xml version="1.0" encoding="UTF-8"?>

<service id="remoting-service"

????class="flex.messaging.services.RemotingService">

?

????<adapters>

????????<adapter-definition id="java-object" class="flex.messaging.services.remoting.adapters.JavaAdapter" default="true"/>

????</adapters>

?

????<default-channels>

????????<channel ref="my-amf"/>

????</default-channels>

<destination id="service">

<properties>

<source>com.soft.flex.flex4sj.service.CupService</source>

</properties>

</destination>

</service>?

(1)修改MXML應用程序?添加RemotingObject對象

<?xml version="1.0" encoding="utf-8"?>

<s:Application xmlns:fx="http://ns.adobe.com/mxml/2009"

???xmlns:s="library://ns.adobe.com/flex/spark"

???xmlns:mx="library://ns.adobe.com/flex/mx" minWidth="955" minHeight="600"

???initialize="application1_initializeHandler(event)"

???>

<fx:Script>

<![CDATA[

import mx.collections.IList;

import mx.controls.Alert;

import mx.events.FlexEvent;

import mx.rpc.events.FaultEvent;

import mx.rpc.events.ResultEvent;

import spark.events.IndexChangeEvent;

private var flag:int;

protected function application1_initializeHandler(event:FlexEvent):void

{

this.flag=0;

this.myremoting.getAll();

}

?

?

?

?

protected function ddl_changeHandler(event:IndexChangeEvent):void

{

var cid:String = this.ddl.selectedItem.countryId;

this.flag=1;

this.myremoting.getCupByCountryId(cid);

}

?

?

protected function myremoting_resultHandler1(event:ResultEvent):void

{

this.ddl.dataProvider=event.result as IList;

this.ddl.labelField = "countryName";

this.adg1.dataProvider=event.result;

}

?

protected function myremoting_resultHandler2(event:ResultEvent):void

{

this.adg1.dataProvider=event.result;

}

protected function myremoting_faultHandler(event:FaultEvent):void

{

Alert.show(event.fault.faultString);

}

?

]]>

</fx:Script>

<fx:Declarations>

<s:RemoteObject id="myremoting" showBusyCursor="true" destination="service"

endpoint="http://localhost:8080/sj42/messagebroker/amf"

fault="myremoting_faultHandler(event)"

>

<s:method name="getAll" result="myremoting_resultHandler1(event)" />

<s:method name="getCupByCountryId" result="myremoting_resultHandler2(event)"/>

</s:RemoteObject>

</fx:Declarations>

<s:Panel width="557" height="376" title="使用HttpService 與后臺通信" horizontalCenter="0" verticalCenter="0">

<mx:AdvancedDataGrid x="2" y="9" id="adg1" designViewDataType="flat" width="545" height="274">

<mx:columns>

<mx:AdvancedDataGridColumn headerText="代號" dataField="countryId"/>

<mx:AdvancedDataGridColumn headerText="國家" dataField="countryName"/>

<mx:AdvancedDataGridColumn headerText="金牌數" dataField="goldMedal"/>

<mx:AdvancedDataGridColumn headerText="銀牌數" dataField="silverMedal"/>

?

<mx:AdvancedDataGridColumn headerText="銅牌數" dataField="bronzeMedal"/>

?

</mx:columns>

</mx:AdvancedDataGrid>

<s:Label x="216" y="305" text="選擇國家:" height="23" verticalAlign="middle"/>

?

<s:Button x="475" y="305" label="顯示全部" click="application1_initializeHandler(event as FlexEvent)"/>

<s:DropDownList x="306" y="305" id="ddl" prompt="請選擇國家" change="ddl_changeHandler(event)"></s:DropDownList>

</s:Panel>

</s:Application>

?

?

運行應用,效果如圖5.2.1所示。

?

?

實訓任務3RemotingObject 整合Hibernate Spring

訓練技能點

??RemotingObject

??整合Spring框架。

需求說明

使用RemotingObject 整合Hibernate Spring 重構任務1

實現思路:

1創建Flex項目,將此項目的服務器技術選擇為J2EE服務器(使用遠程對象訪問服務)

2)創建數據庫表tb_cup 表字段與實體類屬性對應 如圖5.2.3所示。

?

5.2.2 tb_Cup

3)切換到MyEclipes視圖,分別添加hibernate支持和spring支持,并使用逆向工程生成視圖類,映射文件等。

4創建業務類,在該類 中定義根據國家查詢獲取獎牌情況的方法。

package com.soft.flex.flex4sj.service;

?

import java.util.List;

?

import com.soft.flex.flex4sj.dao.Cup;

import com.soft.flex.flex4sj.dao.CupDao;

?

public class CupService {

private CupDao dao;

?

public List<Cup> getAll(){

return dao.findAll();

}

public List<Cup>?getCupByCountryId(String cid){

return dao.findByCountryId(cid);

}

public CupDao getDao() {

return dao;

}

?

public void setDao(CupDao dao) {

this.dao = dao;

}

}

?

(5)添加整合Spring框架的響應jar包 并在applicationContext.xml中配置業務類。

<?xml version="1.0" encoding="UTF-8"?>

<beans xmlns="http://www.springframework.org/schema/beans"

???????xmlns:flex="http://www.springframework.org/schema/flex"

???????xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

???????xsi:schemaLocation="http://www.springframework.org/schema/beans

?????????http://www.springframework.org/schema/beans/spring-beans-3.0.xsd

?????????http://www.springframework.org/schema/flex ??????

?????????http://www.springframework.org/schema/flex/spring-flex-1.0.xsd">

?????????<flex:message-broker />

<bean id="service" class="com.soft.flex.flex4sj.service.CupService">

??<flex:remoting-destination />

??<property name="dao" ref="CupDAO"></property>

</bean>

<!--其他配置省略-->

</beans>

(6)修改工程的web.xml文件如下:

<?xml version="1.0" encoding="UTF-8"?>

<web-app id="WebApp_ID" version="2.4" xmlns="http://java.sun.com/xml/ns/j2ee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd">

<display-name>sj43</display-name>

?

<context-param>

<param-name>flex.class.path</param-name>

<param-value>/WEB-INF/flex/hotfixes,/WEB-INF/flex/jars</param-value>

</context-param>

?

<!-- Http Flex Session attribute and binding listener support -->

<listener>

<listener-class>flex.messaging.HttpFlexSession</listener-class>

</listener>

?

<!-- MessageBroker Servlet -->

<servlet>

??<servlet-name>Spring MVC Dispatcher Servlet</servlet-name>

??<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>

??<init-param>

????<param-name>contextConfigLocation</param-name>

????<param-value>classpath:applicationContext.xml</param-value>

??</init-param>

??<load-on-startup>1</load-on-startup>

</servlet>

?

<servlet-mapping>

??<servlet-name>Spring MVC Dispatcher Servlet</servlet-name>

??<url-pattern>/messagebroker/*</url-pattern>

</servlet-mapping>

<welcome-file-list>

<welcome-file>index.html</welcome-file>

<welcome-file>index.htm</welcome-file>

<welcome-file>index.jsp</welcome-file>

<welcome-file>default.html</welcome-file>

<welcome-file>default.htm</welcome-file>

<welcome-file>default.jsp</welcome-file>

</welcome-file-list>

?

</web-app>

(7)修改MXML代碼如下:

<s:RemoteObject id="myremoting" showBusyCursor="true" destination="service"

endpoint="http://localhost:8080/sj43/messagebroker/amf"

fault="myremoting_faultHandler(event)"

>

其中destination=serviceservice要與applicationContext.xml中業務beanid一致。

運行應用程序,效果如圖5.2.1.所示(注意要在服務器中將重復的cglib.jar刪去)

實訓任務4:實現分頁

訓練技能點

??RemotingObject

??整合Spring框架。

需求說明

在任務3的基礎上實現分頁功能。

實現步驟:

(1)切換到MyEclipse視圖 創建Page.java用來封裝分頁數據。

import java.util.List;

?

public class Page {

private List data;

private int currentPage;

private int totalPage;

private int pageSize;

private int totalClum;

public List getData() {

return data;

}

public void setData(List data) {

this.data = data;

}

public int getCurrentPage() {

return currentPage;

}

public void setCurrentPage(int currentPage) {

this.currentPage = currentPage;

}

public int getTotalPage() {

return totalPage;

}

public void setTotalPage(int totalPage) {

this.totalPage = totalPage;

}

public int getPageSize() {

return pageSize;

}

public void setPageSize(int pageSize) {

this.pageSize = pageSize;

}

public int getTotalClum() {

return totalClum;

}

public void setTotalClum(int totalClum) {

this.totalClum = totalClum;

this.totalPage=( totalClum%pageSize==0 ??totalClum/pageSize:totalClum/pageSize+1);

?

}

}

?

(2)dao類中添加分頁查詢的方法

public List findByPage(Page page) {

??????

????????String hql = "from Cup";

????????try {

???????? Session session = getSession();

???????? Query query = session.createQuery(hql);

???????? query.setFirstResult((page.getCurrentPage()-1)*page.getPageSize());

???????? query.setMaxResults(page.getPageSize());

???????? List list = query.list();

???????? releaseSession(session);

???????? return list;

???????????

????????} catch (RuntimeException re) {

????????????log.error("delete failed", re);

????????????throw re;

????????}

????}

public int count() {

????????log.debug("deleting User instance");

????????String hql = "select count(cup) from Cup as cup";

????????try {

???????? Session session = getSession();

???????? Query query = session.createQuery(hql);

????????

???????? long l = (Long)query.uniqueResult();

???????? return (int)l;

???????????

????????} catch (RuntimeException re) {

????????????log.error("delete failed", re);

????????????throw re;

????????}

????}

(3)service類中添加相應的方法。

public Page findByPage(Page page){

List data= dao.findByPage(page);

int count = dao.count();

page.setData(data);

page.setTotalClum(count);

return page;

}

?

(4)切換回Flash視圖 開發與JavaPage類對應的as類。

package com.oa.vo

{

import mx.collections.ArrayCollection;

[RemoteClass(alias="com.soft.flex.flex4sj.dao.Page")]

public class Page

{

public function Page()

{

}

public var data :ArrayCollection;

public var ?currentPage:int;

public var ?totalPage:int;

public var ?pageSize:int;

public var ?totalClum:int;

}

}

?

?

(5)修改主程序的MXML文件 添加分頁欄 。

<?xml version="1.0" encoding="utf-8"?>

<s:Application xmlns:fx="http://ns.adobe.com/mxml/2009"

???xmlns:s="library://ns.adobe.com/flex/spark"

???xmlns:mx="library://ns.adobe.com/flex/mx" minWidth="955" minHeight="600"

???initialize="application1_initializeHandler(event)"

????xmlns:ns1="*">

<fx:Script>

<![CDATA[

import com.oa.vo.Page;

import mx.collections.IList;

import mx.controls.Alert;

import mx.events.FlexEvent;

import mx.rpc.events.FaultEvent;

import mx.rpc.events.ResultEvent;

import spark.events.IndexChangeEvent;

[Bindable]

private var pageData:Page;

public ?function loadData(currentPage:int):void{

var page:Page = new Page();

page.currentPage=currentPage;

page.pageSize=this.setPageSize.selectedItem.data;

this.myremoting.findByPage(page);

}

protected function application1_initializeHandler(event:FlexEvent):void

{

loadData(1);

}

protected function ddl_changeHandler(event:IndexChangeEvent):void

{

var cid:String = this.ddl.selectedItem.countryId;

this.myremoting.getCupByCountryId(cid);

}

protected function myremoting_resultHandler1(event:ResultEvent):void

{

?pageData =Page( event.result) ;

Alert.show(pageData.currentPage+'c');

Alert.show(pageData.totalPage+'t');

this.ddl.dataProvider=pageData.data;

this.ddl.labelField = "countryName";

this.adg1.dataProvider=pageData.data;

}

protected function myremoting_resultHandler2(event:ResultEvent):void

{

this.adg1.dataProvider=event.result;

}

protected function myremoting_faultHandler(event:FaultEvent):void

{

Alert.show(event.fault.toString());

}

]]>

</fx:Script>

<fx:Declarations>

<s:RemoteObject id="myremoting" showBusyCursor="true" destination="service"

endpoint="http://localhost:8080/sj43/messagebroker/amf"

fault="myremoting_faultHandler(event)"

>

<s:method name="findByPage" result="myremoting_resultHandler1(event)" />

<s:method name="getCupByCountryId" result="myremoting_resultHandler2(event)"/>

</s:RemoteObject>

</fx:Declarations>

<s:Panel width="741" height="427" title="使用HttpService 與后臺通信" horizontalCenter="0" verticalCenter="25">

<mx:AdvancedDataGrid x="2" y="9" id="adg1" designViewDataType="flat" width="727" height="274">

<mx:columns>

<mx:AdvancedDataGridColumn headerText="代號" dataField="countryId"/>

<mx:AdvancedDataGridColumn headerText="國家" dataField="countryName"/>

<mx:AdvancedDataGridColumn headerText="金牌數" dataField="goldMedal"/>

<mx:AdvancedDataGridColumn headerText="銀牌數" dataField="silverMedal"/>

<mx:AdvancedDataGridColumn headerText="銅牌數" dataField="bronzeMedal"/>

</mx:columns>

</mx:AdvancedDataGrid>

<s:Label x="215" y="361" text="選擇國家:" height="23" verticalAlign="middle"/>

<s:Button x="475" y="361" label="顯示全部" click="application1_initializeHandler(event as FlexEvent)"/>

<s:DropDownList x="306" y="361" id="ddl" prompt="請選擇國家" change="ddl_changeHandler(event)"></s:DropDownList>

<!--以下為分頁欄-->

<mx:HBox cornerRadius="0" borderStyle="solid" horizontalAlign="left" verticalAlign="middle" width="722" x="7" y="305" height="27">

<mx:Text fontSize="12" text="{' '+pageData.currentPage+'/'+pageData.totalPage+''+' '+pageData.totalClum+'條記錄'}"/>

<mx:LinkButton id="lbtnFirst" label="首頁" ?click="loadData(1)" ?enabled="{lbtnPrevious.enabled}" fontSize="12"/>

<mx:LinkButton id="lbtnPrevious" label="上一頁" click="loadData(pageData.currentPage-1)" ??enabled="{pageData.currentPage!=1?true:false}" ?fontSize="12"/>

<mx:LinkButton id="lbtnNext" label="下一頁" ??click="loadData(pageData.currentPage+1)" enabled="{pageData.totalPage>=(pageData.currentPage+1)?true:false}" ?fontSize="12"/>

<mx:LinkButton id="lbtnLast" label="尾頁" click="loadData(pageData.totalPage)" enabled="{lbtnNext.enabled}" fontSize="12"/>

<mx:Label ??text="每頁顯示:"/>

<mx:ComboBox id="setPageSize" ?width="71" change="loadData(1)">

<mx:dataProvider>

<mx:ArrayList ?>

<fx:Object label="5" data="5" /> ????????????????????????????????????????????????????????????????????????????????????????????????????????

<fx:Object label="10" data="10" />

<fx:Object label="20" data="20" /> ???

</mx:ArrayList>

</mx:dataProvider>

</mx:ComboBox>

<mx:Label ??text=""/>

<mx:NumericStepper id="nsPageNum" stepSize="1" minimum="1" maximum="{pageData.totalPage}" ?enabled="{lbtnJump.enabled}" cornerRadius="0" width="54"/>

<mx:LinkButton id="lbtnJump" label="跳轉" ?click="loadData(nsPageNum.value)" enabled="{pageData.totalPage>1?true:false}" ?fontSize="12"/>

</mx:HBox>

</s:Panel>

</s:Application>

?

6)運行應用程序,效果如圖5.2.4所示。

?

5.2.4 ?分頁

?

?

?

?

?

?

?

?

?

?

?

?

?

?

?

?

?

?

鞏固練習

選擇題

1. ?Flex與外部進行數據通信的方式有()

A. ?HTTPService

B. ?WebService

C. ?Remoting

D. ?HttpRequest

2. ?以下關于FlexRemoting數據通信方式的說法,正確的是()

A. ?Remoting使用AMF二進制信息格式化傳遞數據。

B. ?Flex應用中使用Remoting 技術需要有第三方軟件支持。

C. ?數據量越大,Remoting方式傳輸效率越高。

D. ?Remoting不支持序列化與反序列化。

3. ?以下關于remoting-config.xml文件配置信息的描述 正確的是()

A. ?使用destination節點配置遠程調用類的標示。

B. ?Source節點代表遠程調用類的class文件路徑。

C. ?一個remoting-config.xml文件只允許配置一個destination節點。

D. ?一個remoting-config.xml文件只允許配置多個destination節點。

簡答題

1)什么Remoting數據通信技術?

操作題

開發一個WebService ,用于查詢Oracle數據庫中的商品信息表,并返回結果。然后通過FlexWebService組件調用webservice,將獲取到的結果顯示在表格組件中。

創作挑戰賽新人創作獎勵來咯,堅持創作打卡瓜分現金大獎

總結

以上是生活随笔為你收集整理的Flex与外部的数据通信的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。