知識積累(十五)——JSF+Spring+Hibernate的實例講解(三)

實現設計

        現在,讓我把每件事情都串起來,實現JCatalog項目。你可以衝資源列表中下載應用程序的完整源代碼。

數據庫設計

我們爲例子應用程序創建指定目錄的結構,它包含4個表,如圖5

 

 

 

 

5 數據結構圖

 

類設計

6圖解了JCatalog項目的類圖

 

 

 

6 類圖

 

面向接口編程貫穿於整個設計。在表現層,四個bean被使用:ProductBean, ProductListBean, UserBean MessageBean。業務邏輯層包含兩個服務(CatalogService and UserService)和三個業務對象(Product, Category, and User)。集成層包括兩個DAO接口和它們的Hibernate實現。Spring application contexts 包含和管理業務邏輯層和集成層的很多object beansServiceLocator使JSF和業務邏輯層結合到一起。

 

Wire everything up

因爲這篇文章篇幅的限制,我們只看一個用例。CreateProduct用例示範了怎樣將每件事情串起來建造應用程序。深入細節以前,讓我們使用一個序列圖(圖7)示範所有層端到端的整合:

 

 

7 CreateProduct用例的序列圖

 

現在,讓我們通過對每一層的介紹討論如何實現CreateProduct用例的更多細節。

 

表現層

表現層的實現包括創建JSP頁面,定義頁面導航,創建和配置backing beans,JSF與業務邏輯層結合。

 

JSP pagecreateProduct.jsp是創建新產品的頁面。它包括UI組件和捆綁這些組件的ProductBeanValidateItemsRange自定義標籤檢驗用戶選擇目錄的數目。每個新產品至少有一個目錄被選擇。

 

Page navigation應用程序的導航定義在應用程序的配置文件裏面,faces-navigation.xmlCreateProduct定義的導航規則是:

<navigation-rule>

   <from-view-id>*</from-view-id>

   <navigation-case>

      <from-outcome>createProduct</from-outcome>

      <to-view-id>/createProduct.jsp</to-view-id>

   </navigation-case>

</navigation-rule>

<navigation-rule>

   <from-view-id>/createProduct.jsp</from-view-id>

   <navigation-case>

      <from-outcome>success</from-outcome>

      <to-view-id>/uploadImage.jsp</to-view-id>

   </navigation-case>

   <navigation-case>

      <from-outcome>retry</from-outcome>

      <to-view-id>/createProduct.jsp</to-view-id>

   </navigation-case>

   <navigation-case>

      <from-outcome>cancel</from-outcome>

      <to-view-id>/productList.jsp</to-view-id>

   </navigation-case>

</navigation-rule>

 

Backing bean: ProductBean不僅包含了頁面中UI組件與數據映射的屬性,也包含三個actionscreateAction,editActiondeleteAction。這是createAction()方法的代碼:

 

public String createAction() {

   try {

      Product product = ProductBeanBuilder.createProduct(this);

 

      //Save the product.

      this.serviceLocator.getCatalogService().saveProduct(product);

 

      //Store the current product id inside the session bean.

      //For the use of image uploader.

      FacesUtils.getSessionBean().setCurrentProductId(this.id);

 

      //Remove the productList inside the cache.

      this.logger.debug("remove ProductListBean from cache");

      FacesUtils.resetManagedBean(BeanNames.PRODUCT_LIST_BEAN);

   } catch (DuplicateProductIdException de) {

      String msg = "Product id already exists";

      this.logger.info(msg);

      FacesUtils.addErrorMessage(msg);

 

      return NavigationResults.RETRY;

   } catch (Exception e) {

      String msg = "Could not save product";

      this.logger.error(msg, e);

      FacesUtils.addErrorMessage(msg + ": Internal Error");

 

      return NavigationResults.FAILURE;

   }

   String msg = "Product with id of " + this.id + " was created successfully.";

   this.logger.debug(msg);

   FacesUtils.addInfoMessage(msg);

 

   return NavigationResults.SUCCESS;

}

        在這個action裏面,基於ProductBean的一個Product業務對象被建立。ServiceLocator查詢CatalogService。最後,createProduct的請求被委派給業務邏輯層的CatalogService

 

Managed-bean declaration: ProductBean必須在JSF的配置資源文件faces-managed-bean.xml中配置:

 

<managed-bean>

   <description>

      Backing bean that contains product information.

   </description>

   <managed-bean-name>productBean</managed-bean-name>

   <managed-bean-class>catalog.view.bean.ProductBean</managed-bean-class>

   <managed-bean-scope>request</managed-bean-scope>   

   <managed-property>

      <property-name>id</property-name>

      <value>#{param.productId}</value>

   </managed-property>

   <managed-property>

      <property-name>serviceLocator</property-name>

      <value>#{serviceLocatorBean}</value>

   </managed-property>

</managed-bean>

        ProductBean有一個請求的範圍,這意味着如果ProductBeanJSP頁面內引用JSF執行爲每一個請求創建ProductBean實例的任務。被管理的ID屬性與productId這個請求參數組裝。JSF從請求得到參數,設置managed property

Integration between presentation and business-logic tiers: ServiceLocator抽象了查詢服務的邏輯。在例子應用程序中,ServiceLocator被定義成一個一個接口。接口被JSF managed bean實現爲ServiceLocatorBean,它從Spring application context查詢服務:

 

ServletContext context = FacesUtils.getServletContext();

this.appContext = WebApplicationContextUtils.getRequiredWebApplicationContext(context);

this.catalogService = (CatalogService)this.lookupService(CATALOG_SERVICE_BEAN_NAME);

this.userService = (UserService)this.lookupService(USER_SERVICE_BEAN_NAME);

ServiceLocator被定義爲BaseBean中的一個屬性。JSF managed bean容易連接ServiceLocator執行必須訪問ServiceLocator的那些managed beans。使用了Inversion of controlIOC,控制反轉)

 

業務邏輯層

        定義業務對象,創建服務接口和實現,在Spring中配置這些對象組成了這一層的任務。

Business objects: 因爲Hibernate提供了持久化,ProductCategory業務對象需要爲它們包含的所有屬性提供gettersetter方法。

Business services:CatalogService接口定義了所有與目錄管理有關的服務:

 

public interface CatalogService {

   public Product saveProduct(Product product) throws CatalogException;

   public void updateProduct(Product product) throws CatalogException;

   public void deleteProduct(Product product) throws CatalogException;

   public Product getProduct(String productId) throws CatalogException;

   public Category getCategory(String categoryId) throws CatalogException;

   public List getAllProducts() throws CatalogException;

   public List getAllCategories() throws CatalogException;

}

CachedCatalogServiceImpl服務的接口實現,它包含CatalogDao對象的一個setterSpringCachedCatalogServiceImpl CatalogDao連接在一起。因爲我們提供了接口,所以對實現的依賴不是很緊密。

Spring configuration: 下面是CatalogServiceSpring comfiguration

 

<!-- Hibernate Transaction Manager Definition -->

<bean id="transactionManager" class="org.springframework.orm.hibernate.HibernateTransactionManager">

   <property name="sessionFactory"><ref local="sessionFactory"/></property>

</bean>

 

<!-- Cached Catalog Service Definition -->

<bean id="catalogServiceTarget" class="catalog.model.service.impl.CachedCatalogServiceImpl" init-method="init">

   <property name="catalogDao"><ref local="catalogDao"/></property>

</bean>

 

<!-- Transactional proxy for the Catalog Service -->

<bean id="catalogService" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">

   <property name="transactionManager"><ref local="transactionManager"/></property>

   <property name="target"><ref local="catalogServiceTarget"/></property>

   <property name="transactionAttributes">

      <props>

         <prop key="get*">PROPAGATION_REQUIRED,readOnly</prop>

       <prop key="save*">PROPAGATION_REQUIRED</prop>

       <prop key="update*">PROPAGATION_REQUIRED</prop>

       <prop key="delete*">PROPAGATION_REQUIRED</prop>

      </props>

   </property>

</bean>

Spring聲明事務管理是在CatalogService. CatalogService 裏面設置,它能實現不同CatalogDaoSpring創建並管理單體實例Catalogservice,不需要工廠。

現在,業務邏輯層準備好了,讓我們將它與集成層整合。

Integration between Spring and Hibernate:下面是HibernateSessionFactory的配置:

 

<!-- Hibernate SessionFactory Definition -->

<bean id="sessionFactory" class="org.springframework.orm.hibernate.LocalSessionFactoryBean">

   <property name="mappingResources">

      <list>

         <value>catalog/model/businessobject/Product.hbm.xml</value>

         <value>catalog/model/businessobject/Category.hbm.xml</value>

         <value>catalog/model/businessobject/User.hbm.xml</value>

      </list>

   </property>

   <property name="hibernateProperties">

      <props>

         <prop key="hibernate.dialect">net.sf.hibernate.dialect.MySQLDialect</prop>

       <prop key="hibernate.show_sql">true</prop>

       <prop key="hibernate.cglib.use_reflection_optimizer">true</prop>

       <prop key="hibernate.cache.provider_class">net.sf.hibernate.cache.HashtableCacheProvider</prop>

      </props>

   </property>

   <property name="dataSource">

      <ref bean="dataSource"/>

   </property>

</bean>

CatalogDao使用HibernateTemplate集成HibernateSpring.下面是HibernateTemplate的配置:

 

<!-- Hibernate Template Defintion -->

<bean id="hibernateTemplate" class="org.springframework.orm.hibernate.HibernateTemplate">

   <property name="sessionFactory"><ref bean="sessionFactory"/></property>

   <property name="jdbcExceptionTranslator"><ref bean="jdbcExceptionTranslator"/></property>

</bean>

 

集成層

Hibernate使用一個XML配置文件去映射業務對象到關係型數據庫。在JCatalog項目中,Product.hbm.xml表示Product業務對象的映射。Category.hbm.xml用於業務對象Category。配置文件和相應的業務對象在同樣的目錄下。下面是Product.hbm.xml

 

<?xml version="1.0"?>

<!DOCTYPE hibernate-mapping PUBLIC

      "-//Hibernate/Hibernate Mapping DTD 2.0//EN"

      "http://hibernate.sourceforge.net/hibernate-mapping-2.0.dtd">

<hibernate-mapping package="catalog.model.businessobject">

   <class name="Product" table="product">

      <id name="id" column="ID" unsaved-value="null">

         <generator class="assigned"/>

      </id>

      <property name="name" column="NAME" unique="true" not-null="true"/>

      <property name="price" column="PRICE"/>    

      <property name="width" column="WIDTH"/>     

      <property name="height" column="height"/>     

      <property name="description" column="description"/>  

      <set name="categoryIds" table="product_category" cascade="all">

         <key column="PRODUCT_ID"/>

         <element column="CATEGORY_ID" type="string"/>

      </set>

   </class>

</hibernate-mapping>

CatalogDao通過Spring使用HibernateTemplate連接:

<!-- Catalog DAO Definition: Hibernate implementation -->

<bean id="catalogDao" class="catalog.model.dao.hibernate.CatalogDaoHibernateImpl">

   <property name="hibernateTemplate"><ref bean="hibernateTemplate"/></property>

</bean>

 

結論

這篇文章介紹了怎樣將JSF集成到Spring FrameworkHibernate,建立了一個真實的應用程序。這三種技術的聯合提供了一個可靠的Web應用程序開發框架。一個多層體系結構應該做爲Web應用程序的高級體系結構。JSF很適合MVC設計模式,能夠被用於實現表示層。Spring框架能被用於業務邏輯層去管理業務對象,提供聲明性事務管理和資源管理。SpringHibernate結合的很好。Hibernate是一個強有力的O/R映射框架,能夠提供集成層的服務。

通過將Web應用程序劃分成不同的層和麪向接口編程,每一層的技術可以被取代。例如, 在表示層Struts能取代JSF,在集成層JDO能取代Hibernate。應用程序層之間的整合不是沒有意義的,使用inversion of controlService Locator設計模式能使這個工作容易。JSF提供了其他框架,如Struts所缺少的功能。然而,這不意味着你應該立刻拋棄Struts而開始使用JSF 。無論怎樣,你的項目是否使用JSF作爲你的Web框架,取決於你項目的狀態和功能需求以及團隊專家的意見。

 
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章