搜档网
当前位置:搜档网 › Springside 3.3.2 技术参考手册

Springside 3.3.2 技术参考手册

Springside 3.3.2 技术参考手册
Springside 3.3.2 技术参考手册

Springside 3.3.2 技术参考手册

1. 总述

?架构风格简述

1.概述

技术框架选型以工业化大规模软件开发为原则---主流的选型、适宜团队分工的架构,同时照顾轻量级快速开发的需求,Java版的约定大于配置的风格。

)

在SpringSide的官方主页上,写着SpringSide 3 中包含的主流组件:Spring 2.5 + Hibernate 3 + Struts 2 + JSP 2.0 + JQuery + JAX-WS 2 (by Apache CXF 2) +SpringSecurity 2.0。不过,如果你认为和SpringSide 2 相比,升级的只是版本号,那么你就大错特错了。

像我这样从SpringSide 2过来的人,如果能够详细对比SpringSide 2 和SpringSide 3,对于我们理解SpringSide 3肯定更加有帮助。下面的内容是我自己的一些体会,不正确的地方欢迎大家指正。

从主要的架构来,依然是那么几层,先设计数据库,再写Entity层,再写Dao层,再写Action,然后使用JSP表现出来,其中的Entity和Dao层和Hibernate紧密相关,Action是Struts的主要部分,Spring作为粘合剂,把这些东西连在一起,并提供了一些机制简化编写这些层的工作。SpringSide就是这样一个大的粘合体,省去了我们自己很多的粘合工作。从SpringSide 2 到SpringSide 3,这些工作得到进一步的简化,我们开发起来就更爽了。

我认为,我们的开发模式,主要经历了这样的简化过程:

第一阶段,代码加一大堆的配置文件

第二阶段,代码加少量的配置文件

第三阶段,代码加基本上等于零的配置文件

这些进步是如何实现的呢?从第一阶段到第二阶段,主要靠的是Annotation,把一些大量的配置工作转移到了Java代码中,那么从第二阶段到第三阶段怎么实现呢?都已经Annotation了,你还有什么办法让配置文件更少?想不到吧!我也想不到,直到我看了SpringSide 3生成的项目,我才知道还有一个好东西叫自动扫描。

下面使用实际的例子进行说明。作为示例,Hibernate是个不错的选择。

第一阶段,如果单独使用Hibernate,我们每写一个Entity类,就得同时写一个.hbm.xml文件,如果有很多Entity,就得写很多.hbm.xml文件,每修改一次Entity,也要同时记得修改.hbm.xml文件,相当的不方便,其目录结构如下图:

第二阶段,使用Annotation,SpringSide 2 中就是这样的方式。具体内容可以看我的另一篇博文《打通持久层的任督二脉》。所以我用SpringSide 2 进行开发的时候,写Entity也是两个步骤,但是比第一阶段要简单。就是先写一个Entity,Entity里面使用Annotation,如下图:

然后修改配置文件,SpringSide 2 中专门为Hibernate的定义开辟了一个专门的xml文件dataAccessContext-hibernate.xml,里面关于SessionFactory的定义是我们很熟悉的。我只需要将我的Entity

类加入到里面就可以了,如下图中红笔圈出来的部分:

第二阶段和第一阶段相比,配置文件简单多了,不需要维护那么多的.hbm.xml。但是,每编写一个Entity,依然需要修改一下配置文件。

第三阶段就更了不起了,零配置文件,也就是说我们只管写Entity就行了,不需要管配置文件(当然,项目开始的时候还是要配置一次的)。打开SpringSide 3生成的项目,果然找不到

dataAccessContext-hibernate.xml文件。那么是如何实现零配置的呢?靠的就是自动扫描,如下图中红笔圈出来的部分:

上面讲Hibernate只是为了举例说明我的观点,事实上SpringSide 3 中的零配置可不仅仅只存在这一个地方。

Dao层也可以不用配置了。在SpringSide 2 中,我们每写一个Dao,都需要在serviceContext.xml中加入这个类,如下图:

SpringSide 3 中使用自动扫描了,如下图:

以后编写Service后,再也不用修改配置文件了,只要我们在Service中使用如下Annotation:

事务也不用配置了,在SpringSide 2 中,事务是这样配置的:

SpringSide 3 中,事务配置是这样的:

Action也不用配置了,不过使得Action零配置的代码却在web.xml中,如下图:

还记得SpringSide 2 中的Struts配置代码吗?编写完Action类的代码后,还需要修改两个配置文件,如下:

所以说,SpringSide 3 的一个优点就是让生活变得更加简单了。在以上的截图中,SpringSide 2 项目用的是Eclipse 3.2,而SpringSide 3项目用的是Eclipse 3.4 For JavaEE Developers。哪一个IDE更爽,那也是一目了然。

2.主要库栈

?JDK:JDK 6.0、 UTF-8.

?IOC container:Spring 3.

?ORM:Hibernate 3.3.

?Web:Struts 2.1、JSP 2.0、JQuery 1.4/Dojo Base 1.4.

?Web Service:基于Apache CXF 2.2的JAX-WS/JAXB 2.0规范, 基于Jersey的JAX-RS 规范.

?Security: Spring Security 3.0.

3.层次说明

3.1 entity - 领域模型层

使用Sql First的开发模式,先设计数据库,参考DBA的性能意见而不要太片面追求OO 化的表结构。

然后纯手工编写entity与极少量的JPA annotation(约定大于配置), 也可以用hibernate-tools从数据库逆向生成后再作修改。

3.2 access - 资源访问层

资源访问层包括对数据库、JMS、外部的WebService等的访问。

每个领域对象对应一个DAO类,继承于通用的HibernateDao,所有以该对象为查询主体的HQL语句统一定义于DAO内并提供查询函数.

在性能紧要而Hibernate又无法满足要求时,可混合使用JDBCTemplate。

3.3 service - 业务逻辑层

Service层有两类对象,

一类是领域对象管理类(Entity Manager),按领域对象划分,每个Manager类负责管理多个紧密关联的Entity的增删改查及其业务逻辑。

一类是业务服务类(Service),按业务脚本划分,可能会访问到多种领域对象与Manager 类。

用Spring的Transcation annotation定义事务。对于Hibernate Lazy load的关联对象,在性能要求不高时可使用OpenSessionInView Filter,否则在Service层完成对象的初始化操作。

3.4 web - Web MVC层

MVC框架使用Struts 2.1 这一老牌传统MVC框架 + Convention Plugin 实现零配置文件,每个Action实现一组页面操作。

View模板用JSP2.0 , 尽量使用纯html+JSP2.0 EL展示页面。

Javascript与Ajax使用JQuery或Dojo Base。

尽量采用CSS框架规范CSS的布局。

3.5 ws - WebService接口

使用Java first的开发模式,通过JSR181 annotation标注Web Service接口,用JAXB-2.0 annotation标注Java-XML Mapping。

用DTO类实现Entity与外系统的解耦,用WSResult包裹返回结果与返回码,不使用Exception返回错误。

对于Restful服务,同样采用JAX-RS annotation标注。

3.6 security - 安全控制

使用Spring Security的Filter 拦截URL,使用Spirng Security的taglib 拦截页面内容。

采用用户-角色-资源三层控制,角色-资源关系定义于XML的简化模式。

3.7 测试

单元测试尽量采用MockObject的方式屏幕所有依赖对象/数据的访问,对于dao层与特别查询与特别数据库操作,采用集成测试连接实际数据库,基于 Spring的集成测试Context。

功能测试使用selenium测试主要用户故事的主流程及Javascript的效果,使用Jetty 嵌入式Web服务器与H2嵌入式数据库,使用DBUnit预备数据,使得整个测试可以快速运行,对测试环境没有太多的依赖。

4.常见问题

4.1 什么时候使用基于接口编程?

基于接口编程、Fascade层等等抽象封装都是有开发和维护的代价的,是否使用归根结底还是看在团队人员的分工情况,在大家不得不依赖这几项技术来解决相互的接口、契约问题时,自然就用了。

4.2 Package是先分层还是先分模块

org.springside.模块A.web 还是 org.springside.web.模块A?同上,还是看团队人员的分工情况。如果是每人从头到尾负责一个独立模块的可以先分模块。反之,按层进行分工并鼓励层内重用的,可以考虑先分层。

目录结构描述

1.公共目录结构

完全采用maven的默认布局。

bin 命令脚本目录,详见后。

src main 主源码目录

main/java java源文件

main/resources 配置文件、属性文件

main/webapp

Web应用目录

test 测试目录,结构与主源码目录相同unit 单元测试

functional 功能测试

data 测试数据生成器

target maven编译目录

2.bin目录结构

文件

描述

db-export / db-init 将当前数据库数据通过dbunit导出到export-data.xml。

执行src/main/sql 中的脚本初始化数据库schema,再通过dbunit 执行default-data.xml初始化数据。

convert-h2 将真正数据库的sql转换为测试用的H2的sql。eclipse 生成Eclipse项目文件。

jetty / Jetty-debug 用jetty快速运行本项目。 jetty-debug支持eclipse以debug方式连入,端口为8000

hibernate/generate 从数据库逆向生成entity / dao / pojo

code

2. 基础框架

?Spring: Spring Framework

1. 参考资料

1.图书

参考手册中文版(英文版) ()

》 Apress

2.文章

?Introduction to the Spring Framework 2.5 (TSS) Rod 老大的Spring介绍 Spring

2.5版。

3.笔记

的笔记

part1part2、part3

)

》 Apress

2. 选型

有人说guice好,但我们用Spring不仅是为了IOC,也为了Spring对JEE所做的封装,以及整个JavaEE界众多项目 (SpringSecurity,CXF、Struts2等)对它的依赖。

3. in SpringSide 3 - Mini示例使用的基本特性

?属性配置及IOC容器

?@Transcational annotation标注的事务管理功能

?集成测试的基类

3.1 IOC部分特征使用

详见Spring 笔记(江南白衣博物馆)

3.1.1 零配置文件的使用场合

Service,Dao 层在无特殊配置项时,可使用IOC Annotation进行配置实现约定俗成大于配置的零配置文件.

而其他Database/SessionFactory ,SpringSecurity, CXF等需要进行参数配置时,仍然使用 XML 配置文件。

3.1.2 零配置文件的annotation

SpringSide3中使用@Component注释所有类别,在setter方法(不需要严格按setter 命名)上使用@Autowired 进行byType注入,如果需要byName注入则使用@Resource。

3.1.3 其他特征

使用@Required注释没有用@Autowired注释的属性,保证对象必然被注入,如果对象没有被注入则报错。

使用JSR250的@PostConstruct来定义在执行完所有setter注入后必须执行的函数。

@PreDestroy来定义JDK关闭时通过 shutdownhook调用的函数。

ApplcationContext中默认设置Lazyload 与Autowired by type 仍然是必然的选择。

3.1.4 属性文件管理

在配置文件中,很有一些属性会在团队的开发环境,个人的开发环境,集成测试环境,以及生产环境集群的每台服务器中都有所不同的。详见Spring 笔记(江南白衣博物馆)

3.2 事务管理

在Service层用@Transactional 定义事务,详见Spring 笔记(江南白衣博物馆)。注意Spring3.0 的@Transactional终于支持多个TransactionManager了, showcase里演示了多个数据源(不要求联合事务JTA)导致多个TransactionManager的情况。此时

@Transactional默认使用中指定的TransactionManager, 但也可以重新指定其他的TransactionManager的BeanName.

在AOP不到的地方,仍然有机会使用TransactionTemplate,(见Showcase的UserJdbcDao 与 LogJdbcWriter。)

3.3 集成测试的基类

详见测试文档

4. in SpringSide3

4.1 Showcase中使用的Spring功能索引

?JMX:JMX的Server端封装

?JMS:JMS Template封装

?Email:EMail客户端封装

?Schedule:Quartz的封装,ScheduleExecutor的封装

?Ehcache:Ehcache Server封装

?Jdbc:Spring JdbcTemplate

4.2 SpringContextHolder

在Spring ApplicationContext启动时,先把Context放入个该类的静态变量,当模块的代码无法通过正常渠道获得Spring Context及其中的Bean时,可直接从该静态变量中获取。

4.3 AOP Aspect的写法

最新的写法,一切都写在java文件里,只要在applicationContext.xml里面定义一把scan就可以了。

详见showcase中的TraceLogAspect。

?Database: ORM-Hibernate, Jdbc-Spring JdbcTemplate, 数据库 - H2、MySQL、Oracle, 数据库连接池-DBCP

1. 资料

2. 选型

因为最近一年来,JPA与EJB3的应用都没有想象中的多,在国内尤其突出。所以仍然使用纯Hibernate API+JPA1.0 annotation. 支持JPA2.0的Hibernate 3.5刚出来,要再继续观察。

3. in SpringSide 3

3.1 Entity类

1.Mini-Web中的User类是entity类的典型,集中演示了缓存,默认命名策略,多对多关系的定义和fetch策略,级联操作策略,子集合的排序和缓存、非持久化属性,见JPA 与Hibernate笔记

2.为了统一ID的列名与生成算法,编写了IdEntity基类,演示了自增序列,数据库Sequence(Oracle), UUID(多数据库)的情形。(如果要在Oracle中为每个Entity对应不同Sequence时- 重载getId()函数)。

3.2 更换MySQL或Oralce等数据库

详见数据库说明。

1. H2

1.1 选型

H2嵌入式模式是功能测试的首选。测试要求数据库运行要非常快速,而且为免测试间数据的耦合,要能在每个测试间很快速的重建数据,H2刚好满足这个需求。

H2的主要对手是Derby,对比显示H2比Derby更快,而且有着非常好用的Web管理界面(在Web 界面里写SQL时居然有表名列名的动态提示....)

1.2 in Springside 3

在/tools/h2目录里带了启动H2的命令(借助于maven管理的Jar包)。

另外,H2完全兼容hsqldb的语法,一般依赖数据库的开源项目都附带有hsqldb语法的初始化脚本。

SpringSide3中在使用嵌入式H2提供快速功能测试。url为jdbc:h2:mem:mini-service;DB_CLOSE_DELAY=-1

同时运行期的数据库也使用了H2的Server模式,url为jdbc:h2:tcp://localhost/~/mini-service。

2. MySQL 与Oracle

开发实际项目时,需要将H2更换为实际数据库,Hibernate支持大部分的数据库,其中Java程序员选用MySQL和Oracle的最多。

更换MySQL与Oracle 主要修改几个地方,SpringSide已在两个mini-example中带了示范配置,取消相应的注释符号即可。

1.pom.xml

一节jar包的groupid与artifactId,version用于下载依赖包。

注意因为版权问题,Oracle的Driver不在maven官网的仓库上,需要自行发布到团队的私服或拷贝到本机的maven仓库目录。

2.src/main/resource/applicationt.properties

用于连接池与hibernate的配置。

3. bin/build.xml

用于导出导入数据到数据库。

4. bin/hibernate/hibernate.cfg.xml

如果需要用hibernate-tools生成代码则修改此文件,否则可忽略。

5.修改entity的ID生成策略。

注意Oracle中一般会为每个主要的表创建一个Sequence,所以继承于IdEntity的子类需要重载getId()方法指定Sequence名。

当然也可以只修改IdEntity基类,所有entity共用一个sequence。

3.3 Event机制

在Showcase中的AuditListener,演示了利用Event机制,在save与update时自动为实体加入审计信息(创建人与创建时间,最后修改人与修改时间)

3.4 @Version字段

在Showcase中演示,@Version字段用在诸如两个管理员在差不多时间里打开了同一对象的修改页面可能引发的冲突,Struts2的 Prepareable机制对其使用有些许影响,需进行编码检查, 详见UserAction.

3.5 Clob字段

在Showcase中演示,Clob字段用String表示即可,但为了达到Lazy Load的效果,必需进行byte code enhancement,见showcase中bin/hibernate/bytecode-instrument中执行的Ant指令。

@Lob

@Basic(fetch = https://www.sodocs.net/doc/d68245275.html,ZY)

public String getContent() {

return content;

}

3.6 继承

在Showcase中演示,为性能考虑,一般采用同表继承,另外,在基类放一个

@ForceDiscriminator有时候很有用。

@Table(name = "SS_POST")

@Inheritance(strategy = InheritanceType.SINGLE_TABLE)

@ForceDiscriminator

public abstract class Post @Entity

@DiscriminatorValue("Subject")

public class Subject extends Post

3.7自定义ID生成器

在Showcase中演示一个16位的UID生成。

3.8 关联关系大全

在mini-web中演示@ManyToMany.

在shwocase中演示,单向,双向的@OneToMany, @ManyToOne .

4. in SpringSide II --性能相关

Hibernate 其实做了很多性能相关的优化,大家可以尽量使用后,再决定要不要直接跑JDBC。

4.1 二级缓存

在Entity类及其Collection属性中都可以声明Hibernate缓存策略。Entity缓存对象实体,而Collection则缓存对象间的关系。

注意并不是所有对象都适合缓存,需要进行精心选择,详见Hibernate笔记。

这里使用了ehcache cache方案,支持基于JGroups分布式二级缓存,在sessionFactory处配置了ehcache_hibernate.xml的配置文件。关于 Ehcache详见Ehcache 部分。

QueryCache本身就是吃力不讨好的事情,在集群环境下更加需要注意,因此我们一般不推荐使用使用查询缓存.

4.2 不使用OpenSessionInFilter

对于性能要求较高的Web网站与WebService应用,还是应该在Service层用Hibernate.init()主动初始化对象然后关闭 Session。详见mini-service的演示。注意Hibernate.init(user)初始化user对象,而如果要同时初始化user与 user的Lob字段与关联对象,调用的是Hibernate.init(user.getRoles()),

Hibernate.init(user.getDescription()).

4.3 预Fecth关联对象集合

HQL与Criteria都可以一次关联查询查出主对象与关联对象集合,需要注意的查出来的记录和我们平时用jdbc查一样,有三个子对象就会有三条重复的主对象记录,可以用CriteriaSpecification.DISTINCT_ROOT_ENTITY去除重复。

注意,如果关联对象集合和关联对象是被缓存的,就不一定要预Fetch对象了。

详见Showcase的UserDaoTest 。

4.3 Bulk Update HQL

嗯,HQL也有Update/Delete语句的, 详见Showcase的UserDaoTest 。

4.4 扩展HQL Dialect

如果有些数据库特定的功能而Hibernate又还没提供的话,可以简单的扩展HQL Dialect. Showcase中的H2ExtDialect,为H2Dialect添加了两个新函数, 测试函数见UserDaoTest。

4.5 直接JDBC访问

如果还是不行,就在需要的地方干脆的使用JdbcTemplate吧,不要折腾什么Hibernate Native SQL了。

5. SpringSide3 封装

5.1 DAO类

1.通用的HibernateDAO

SpringSide 3 封装的泛型SimpleHibernateDAO与HibernateDAO类,前者具有通用的CRUD函数,QBH,QBC函数,后者扩展了SpringSide的分页查询函数以及按属性过滤条件的查询函数。

因为HibernateDAO已满足一般的DAO需要,因此可以直接在Service层中直接使用,仍然定义DAO子类主要为了保存注入 sessionFactory对象(否则sessionFactory要注入到Service层) 以及可能存在的特殊DAO操作。

这里指的特殊处理并不是指一条HQL查询或一个按属性的查询都封装出一个查询函数,而是指RoleDAO中级联删除中间表这种真正特殊的 hibernate API调用和操作。

2. 不继承于HibernateDaoSupport/HibernateTemplate

参考Spring最新的Petlinc例子,DAO不再继承于Spring复杂的HibernateDaoSupport(Spring 1.x+ Hibernate 2.xd的产物),而是直接使用H3推荐的sessionFactorygetCurrentSession()方法与Hibernate的原生 API。

5.2 分页查询

SpringSide3封装的Page类配合Hibernate query by Criteria,可以迎合展示层Grid(表格控件)的种种需求(分页,排序,按属性查找)。

Page类分两部分:

一部分是请求参数:pageSize、pageNo、orderBy(多个排序字段以,分隔)、asc(是否升序,多个排序字段以,分隔)、 autoCount(是否自动执行count查询统计总结果数) 。

一部分是查询结果:result(分页的结果List),totalCount(查询的总结果数),totalPages(总页数), 以及相应的上页页号,下页页号,是否还有上一页,是否还有下一页等函数。

相关主题