实例介绍Struts+Spring+Hibernate开发 :: 〖启飞飞扬〗

来源: BlogBus 原始链接: http://qifei.blogbus.com:80/logs/2006/01/1839988.html 存档链接: https://web.archive.org/web/20060218024643id_/http://qifei.blogbus.com:80/logs/2006/01/1839988.html


介绍 一 本文并不想介绍 Struts , Spring , Hibernate 的原理系统架构等,本文地目的是通过一个较复杂地实例介绍如何整合 Struts , Spring , Hibernate ,网上现有的例子虽然也能达到目的,但功能都比较单一,复杂的例子时会有意想不到的麻烦。本文对读者假设已经具备了以上框架的基础知识。以及那些已经了解 Struts , Spring , Hibernate 的基本概念,但是还没有亲身在较复杂的项目中体验 Struts + Spring + Hibernate 的开发人员。 1 Struts 虽然不打算过多介绍 Struts 的原理,但是大概介绍一下还是有必要的。 Struts 本身就是 MVC 在这里负责将用户数据传人业务层,以及 将业务层处理的结果返回给用户,此系统属于较简单 WEB 应用,采用了 OpenSessionInView 模式处理 LazyLoad 问题,这样我们可以在用户视图中使用 get , set 方法来方便地获取关联对象。为了处理庞大的 Action 和 ActionForm 问题,在此我门准备使用 DynaActionForm (DynaValidatorForm) 和 DispatchAction 以及 动态验证框架 来解决。及使用 Tile 来解决框架问题 。使用自定义标签处理分页和身份验证问题。 2 Spring Spring Framework 最得以出名的是与 Hibernate 的无缝链接,虽然 Spring 对 Hibernate 提供了 90 %以上的封装,使我们不必去关心 Session 的建立,关闭,以及事务使我们能够专心的关注业务逻辑。但是一些特殊情况如 有时需要 Query 以及 Criteria 对象,分页等, Spring 不能给我们提供支持,总不能每次都在你的 DAO 上写个 HibernateCallBackup() 吧? Spring 的作用不是把 Hibernate 再封装一层,而是让你接触不到 Hibernate 的 API ,而是帮助你管理好 Session 和 Transaction 。 在这里解决方法是:首先 写一个 IBase 的接口,和一个 BaseDao 的实现。在实现中仿照 HibernateTemplate ,将其功能一一实现,同时考虑到 Spring 未能支持的地方,我们不得已只好自己来管理 Session ,因此加入 public Session openSession() , public Query getQuery(String sql) , public Criteria getCriteria(Class clazz) ,以及分页的方法。 然后为每一个 Entity 都建立继承于以上类的 IEntity ,与 EntityDao 。这里可以根据需求对 Entity 加入特殊的方法实现,如 在 StudentsDao.java 中加入类似用户身份验证等。以上就是数据访问层。接下来在 Service 层中通过对 dao 的引用完成业务逻辑方法。在下面的例子中我们分别为学生模块,教师模块,管理员模块构建 Service 层, StudentsServiceImpl , TeachersServiceImpl , AdminServiceImpl 。 3 Hibernate 有了 Spring 的封装,我们要对 Hibernate 做的就是正确实现对象关系的映射。由于此处处于系统的最底层,准确无误的实现对象之间的关联关系映射将起着至关重要的作用。 总之,理解了 Struts , Spring , Hibernate 地原理以及之间的关系之后,剩下的工作就如同在以 Spring 为核心的 Struts 为表现的框架中 堆积木。 下图可以更好的帮助我们理解 Struts , Spring , Hibernate 之间的关系。 二 案例简述 : 设计思路主要源于 大学选修课,该系统可以方便处理学生在课程选报,学分查询,成绩查询,以及 成绩发布等。 系统以班级为核心,一门课程可以对应多个班级,一名教师也可以带不同的班级,学生可以选报不同课程所对应的班级,班级自身有目前人数,和最大人数,以及上课时间,上课地点的属性。 学生在选报班级之后,班级的人数会自动加一,直到等于最大人数时,其他学生将会有人数已满的错误提示。同理如果学生选择了同一课程的不同班级,也将收到错误提示。学生有密码,系别,学分,地址,电话等属性。 教师在系统中主要负责成绩发布,教师可以对其所带的班级的学生的成绩修改,系统会以成绩是否大于等于 60 来判断学生是否通过考试,如果通过会将该课程的学分累加到学生学分,同样如果教师二次修改了成绩,而且小于 60 ,系统会在学生学分上扣掉该课程的分数。 课程在系统中具体体现为班级,自身带有学分属性。 系有编号,名称的属性,同时可以作为联系教师,课程,学生的桥梁。 功能模块 l 身份验证模块: 根据用户名,密码,用户类别 转发用户到不同的模块。 l 学生模块: 查看课程,查看班级,选报课程,查看己选课程,成绩查询。 l 教师模块: 录入成绩 l 管理员模块:对学生,教师,课程,班级,系 增,删,查,改。 三 具体实践 代码下载 http://www.blogjava.net/Files/limq/StudentManger.rar 1 对象关系映射: 首先,将库表映射为数据模型( SQL 在源码中查看),转换后的数据模型如下图: 由此我们可以看出一下关联关系: 1 Students 和 Contact (联系方式)一对一关系。 2 Students 和 History (选课历史) 一对多关系 3 Students 和 Classes 多对多关系。 4 Classes 和 Classes_info 一对多关系。 5 Classes 和 Teachers 多对一关系。 6 Classes 和 Courses 多对一关系。 7 Course 和 Department (系) 多对一关系。 8 Teachers 和 Department 多对一关系。 9 Students 和 Department 多对一关系。 在 Hibernate 中将以上关系一一映射,如 Students 和 History 一对多关系 Students.cfg.xm. : 1 < set name ="history" 2 table ="history" 3 cascade ="all" 4 inverse ="true" 5 lazy ="true"

6 < key column ="student_id" /> 7 < one-to-many class ="limq.hibernate.vo.History" 8 /> 9 set

10 用过 MyEclipse 开发 Hibernate 的就知道, MyEclipse 会帮助我们生成持久对象和抽象对象,我们要在 Students.java 中加入对 History 的引用 private Set history=new HashSet(); public Set getHistory() { return history; } public void setHistory(Set history) { this.history = history; } 同时,在 AbstractHistory.java 中删除 student_id 以及对应的 get , set 方法, History.java 中加入 private Students student; public Students getStudent() { return student; } public void setStudent(Students student) { this.student = student; } 具体内容请查看 源代码。 1 public interface IDepartment extends IBaseDao {} 2 3 public class DepartmentDao extends BaseDao implements IBaseDao {} 4 3 Service 层 在这里需要认真思考每个业务逻辑所能用到的持久层对象和 DAO ,还要完成配置 Spring 框架, 首先我一起看看 applications-service.xml 1 xml version="1.0" encoding="UTF-8" ?> 2 DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN" 3 "http://www.springframework.org/dtd/spring-beans.dtd"

4 < beans

5 < bean id ="dataSource" class ="org.apache.commons.dbcp.BasicDataSource" destroy-method ="close"

6 < property name ="driverClassName"

7 < value

com.mysql.jdbc.Driver value

8 property

9 < property name ="url"

10 < value

jdbc:mysql://localhost:3306/Student value

11 property

12 < property name ="username"

13 < value

root value

14 property

15 < property name ="password"

16 < value

value

17 property

18 bean

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

20 < property name ="dataSource"

21 < ref local ="dataSource" /> 22 property

23 < property name ="mappingResources"

24 < list

25 < value

limq/hibernate/vo/Admins.hbm.xml value

26 < value

limq/hibernate/vo/Classes.hbm.xml value

27 < value

limq/hibernate/vo/Courses.hbm.xml value

28 < value

limq/hibernate/vo/Students.hbm.xml value

29 < value

limq/hibernate/vo/ClassesInfo.hbm.xml value

30 < value

limq/hibernate/vo/Contact.hbm.xml value

31 < value

limq/hibernate/vo/Department.hbm.xml value

32 < value

limq/hibernate/vo/History.hbm.xml value

33 < value

limq/hibernate/vo/Teachers.hbm.xml value

34 list

35 property

36 < property name ="hibernateProperties"

37 < props

38 < prop key ="hibernate.dialect"

net.sf.hibernate.dialect.MySQLDialect prop

39 < prop key ="hibernate.show_sql"

true prop

40 props

41 property

42 bean

43 < bean id ="myTransactionManager" class ="org.springframework.orm.hibernate.HibernateTransactionManager"

44 < property name ="sessionFactory"