知识积累(十五)——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框架,取决于你项目的状态和功能需求以及团队专家的意见。

 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值