spring+hibernate多层web开发eclipse下的开发模板(大愚弱智)
本人吐血奉献,内容包括:
1. xdoclet生成hbm配置文件和sql语句。
2. xdoclet生成spring的配置文件applicationContext.
3. 容器管理事务并解决延迟加载问题.
4. 解决国际化和中文问题.
5. bo、dao、business service、controller、view(jstl或jsp),一共五层结构。
6. 表单绑定、表单验证。能绑到bo的尽量使用bo来绑定,不能绑到bo就要自己做command(类似于struts 的ActionForm)。
7. Dao测试用例的设计技巧。Dao测试用例的设计要做到(我总结的经验,对错否还望指正):
(1) 独立性: 测试使用的数据记录由测试程序自己生成.
(2) 可移植性:通过CVS check out到另外机器也能通过。--公司领导在其机器上check out 后运行测试看到绿色也开心。
(3) 对数据库无入侵性:测试程序生成的数据记录最后也由自己删除。
(4) 测试完全性:尽力保证测完所有方法且当然希望是通过的,要做到这点有点难,特别是涉及多表查询时,所以我只说是“尽力”。
Service层的测试用例也应该这样设计的,但有时需要初始化的数据量太大,最后要删除的数据也太多,我感觉得不偿失,大家可量力而行。
8. 使用Hibernate映射解决树形数据结构的例子--Cat.java,在这里面也包含使用version乐观锁定的xdoclet tag的书写格式。
9. 使用继承策略简化Dao的编写.这点很重要,使用继承,有些dao接口和接口实现里面的方法是空的。
--------------------------------
^_^以下说明写地有点乱,我有时间再改。^_^
--------------------------------
环境说明:
spring+hibernate是轻量级的解决方案,在任何j2ee容器都能运行.
我使用的mysql4+tomcat5,配置文件也是针对mysql的.为了让初学者能很快上手,也建议你使用mysql4+tomcat5,这样你就不用修改里面配置啦!
项目说明:
从bo、dao interface、dao implement、daoTest、bussiness interface、bussiness implement、bussiness test、command(类似于ActionForm)、Controller(类似于Action)、Validator(类似于struts的validator)、jsp&jstl view都包含,自己慢慢看吧。2. hibernate的配置文件、sql语句生成使用xdoclet,spring bean的配置文件也用xdoclet生成,完全自动化呀,很酷!
3. 这是一个小型宠物管理,猫下面有孩子猫,所以是一个用ibernate实现树形的猫的数据结构,很强的hibernate啊!
运行说明:
安装eclipse3+myeclipse(配套eclipse3的版本的myeclipse)先,在eclipse里面建立一个名为pet的工程,把这个工程拷贝粘贴覆盖过去,再刷新工程绝对ok!
2. 下载spring、hibernate2.x、xdoclet2.1、ant、mysql_jdbc_driver(我用这个driver:mysql-connector-java-3.0.14-production-bin.jar),把它们lib目录下所有的jar文件都放入到WebRoot/WEB-INF/lib目录下--初学者这样最省事,哪些jar是需要的也别管。
3. 运行src/build.xml,然后右键点击工程pet,刷新,生成src/org/ggyy/bo/*.hbm.xml文件;再运行src/build.xml,再刷新工程,生成src/sql.ddl文件(这是eclipse3.0+myeclipse下的毛病,不能自动刷新,其它ide我没试过,我只喜欢eclipse!^_^).
4. 运行org.ggyy.util.DataBaseTask(这是我写的一个类),读取src/sql.ddl文件,然后往mysql数据库里面(使用mysql内置的test数据库)发送sql语句.
5. 运行springBuild.xml(也是ant脚本,eclipse不能自动刷新,我干脆把每个target分开!这样运行ant,郁闷ing),生成spring的配置文件WebRoot/WEB-INF/applicationContext.xml. 额外的bean声明写在src/spring-beans.xml,由spring的xdoclet将其合并到aplicationContext.xml里面去--自己看看就明白了.
6. 发布到tomcat5,启动tomcat服务器,浏览器键入:http://localhost:8080/pet/db/listCat.sf
国际化和中文问题解决说明:
1. messages_zh_CN.properties文件必须使用JDK提供的转码工具native2ascii.exe进行转换:
native2ascii messages_zh_CN.properties msg.txt
把生成的目标文件msg.txt拷贝粘贴替换掉Messages_zh_CN.properties里面的内容,这样就不会乱码了.
2. 中文问题我是这样解决的:
(1) src/spring-beans.xml有如下声明(spring-beans.xml实际上经sprng的xdcolet处理最终合并到applicationContext.xml里面):
java代码:
这里使用mysql里面的test数据库,请注意url中的useUnicode=true&characterEncoding=GB2312.这是mysql特有的,其它数据库环境我没试过,使用其它数据库的朋友要注意.
(2) WebRoot/WEB-INF/web.xml有编码的fillter声明,也是gb2312,自己看吧.
(3) 每一个jsp头文件, <%@ page contentType="text/html; charset=gb2312"%>,当然也gb2312也!
3.其它语言我没试过.
用容器管理事务来解决Hibernate的延迟加载问题:
1.在spring里面解决延迟加载的问题很简单,只要给方法配置一个事务,有事务上下文,就有HibernateSession上下文,就不存在延迟加载的问题.这个问题用AOP来解决可能好一些,可惜这样的拦截器我不会^_^.
2.容器管理的事务只在Dao层和Service层进行管理(在Controller层也管理事务我感觉很变态),由于没有使用OpenSessionInView 的filtter,在view层就有延迟加载的问题.为了避免这种现象,在Service或Dao层事先就把view层所需要的数据传递给Controller层,再由Controller层传给view层;同时约定,除非Controller层已经明确地把从表的数据加载,否则在view层不要试图取得从表类的数据!
使用基类实现Dao说明
bo包中:
Entity.java.
所有bo类的基类,只有一个属性id,这样你就可以统一控制主键的生成策略.
java代码:
package org.ggyy.bo;
public class Entity {
private long id = -1;
/**
* @hibernate.id
* generator-class="native"
* unsaved-value="-1"
*/
public long getId() {
return id;
}
public void setId(long i) {
id = i;
}
public boolean equals(Object arg0) {
return this.getId() == ((Entity) arg0).getId();
}
}
Cat.java.
bo 的一个例子,派生自Entity。使用version的锁定.
java代码:
package org.ggyy.bo;
/**
* 树形数据结构的例子:树猫,有parent和children,且映射到同一个字段:fk_parent_id
*/
/**
* @hibernate.class table="tbl_cat" dynamic-update="true" dynamic-insert="true"
* optimistic-lock="version"
*/
public class Cat extends Entity {
private Set children = new HashSet();
private Cat parent;
private Owner owner;
private String name;
private Integer version;
/**
* @hibernate.many-to-one
* column="fk_owner_id"
* class="org.ggyy.bo.Owner"
* cascade="save-update"
*/
public Owner getOwner() {
return owner;
}
public void setOwner(Owner owner) {
this.owner = owner;
}
/**
* @hibernate.version
*/
public Integer getVersion() {
return version;
}
public void setVersion(Integer version) {
this.version = version;
}
/**
* @hibernate.set
* cascade="all"
* inverse="true"
* lazy="true"
* @hibernate.collection-key
* column="fk_parent_id"
* @hibernate.collection-one-to-many
* class="org.ggyy.bo.Cat"
*/
public Set getChildren() {
return children;
}
public void setChildren(Set children) {
this.children = children;
}
/**
* @hibernate.many-to-one
* column="fk_parent_id"
* class="org.ggyy.bo.Cat"
* cascade="save-update"
*/
public Cat getParent() {
return parent;
}
public void setParent(Cat parent) {
this.parent = parent;
}
/**
* @hibernate.property
*/
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
Dao包中:
IEntityDao.java.
这个Dao接口是所有Dao接口的父接口.其它Dao接口只要继承这个接口,增加、删除、修改、load的方法就不用写了,只需要关注属于自己的finder(或filtter)方法就可以了.
增加和修改操作合为一个方法:public void store(Entity entity)。它实际调用saveOrUpdate的方法.
你也可以加入更多的共有方法,这样要看你的项目的需要和你自己对所建的系统的理解和抽象能力.
java代码:
package org.ggyy.dao;
public interface IEntityDao {
public Entity load(String id) throws DataAccessException;
public void store(Entity entity) throws DataAccessException;
public void delete(String id) throws DataAccessException;
public void delete(Entity entity) throws DataAccessException;
}
ICatDao.java.
这个Dao只需要很少的方法,因为基类已经有了增删改load的方法了.
java代码:
package org.ggyy.dao;
public interface ICatDao extends IEntityDao {
/**
* 树形遍历
*/
public void treeVisitCat(String catId, IVisit visit)
throws DataAccessException;
/**
* 查找所有孩子,要遍历树,使用HQL很难解决的^_^
*/
public List findAllChildren(String catId) throws DataAccessException;
/**
*查找直接子节点,直接使用HQL。
*/
public List findDirectChildren(String catId) throws DataAccessException;
public List findRootCats() throws DataAccessException;
}
IVisit.java
遍历接口,看一下CaoDaoImpl中的findAllChildren的方法就知道如何使用了这个接口了.
java代码:
package org.ggyy.dao;
public interface IVisit {
public void visit(Cat c);
}
Dao的实现:
EntityDaoImpl.java.这是一个抽象类,由于事先不知道bo的class,所以推后由每一个Dao实现来完成.
java代码:
abstract public class EntityDaoImpl extends HibernateDaoSupport implements
IEntityDao {
/**
*抽象方法,是留子类实现的。
*/
abstract protected Class getEntityClass();
public Entity load(String id) throws DataAccessException {
return (Entity) this.getHibernateTemplate().load(this.getEntityClass(),
new Long(id));
}
public void store(Entity entity) throws DataAccessException {
this.getHibernateTemplate().saveOrUpdate(entity);
}
public void delete(Entity entity) throws DataAccessException {
this.getHibernateTemplate().delete(entity);
}
public void delete(String id) throws DataAccessException {
Entity entity = (Entity) this.getHibernateTemplate().load(
this.getEntityClass(), new Long(id));
this.getHibernateTemplate().delete(entity);
}
}
CatDaoImpl.java.
继承于EntityDaoImpl.
我给treeVisitCat和findAllChildren这两个方法配置了一个readonly的事务,并不是因为这个方法需要事务,而是为了解决延迟加载的问题.
在本系统里,所有的一对多都是cascade="all" inverse="true" lazy="true"; 所有的的多对一都是cascade="save-update".主表类不负责从表类的加载,也不负责维护主从关系,这样的做我感觉性能是最好的.但由于lazy="true",下面这个方法的cat.iterator()再往下运行就出错了(是因为延迟加载的问题).但给它配置一个事务就不同了,有了事务上下文(我也不知道是不是这样称呼),就有HibernateSession的上下文,这样就不会有延迟加载的问题啦.
这样的做法有点无耻,为了弥补性能的损失,我只好把Transaction做成readonly.
对于这种query,使用OpenSessionInView是行,但这样HibernateSession不好控制. 我想过了,最好方法是使用AOP给它配置一个HibernateSession的上下文,也就是使用AOP来拦截,可惜我不会^_^.
java代码:
/**
* @spring.bean id ="catDaoTarget"
* @spring.property name="sessionFactory" ref="sessionFactory"
*/
public class CatDaoImpl extends EntityDaoImpl implements ICatDao {
/**
* 需要readonly的事务,是为了解决Hibernate的延迟加载问题.
*/
public List findAllChildren(String catId) throws DataAccessException {
final List children = new ArrayList();
this.treeVisitCat(catId, new IVisit() {
public void visit(Cat c) {
children.add(c);
}
});
return children;
}
/**
* 需要readonly的事务,是为了解决Hibernate的延迟加载问题.
*/
public void treeVisitCat(String catId, final IVisit visit)
throws DataAccessException {
Cat catt = (Cat) getHibernateTemplate()
.load(Cat.class, new Long(catId));
Stack s = new Stack();
s.push(catt);
while (s.empty() == false) {
Cat c = (Cat) s.pop();
visit.visit(c);
Set children = c.getChildren();
if (children != null && !children.isEmpty()) {
Iterator ci = children.iterator();
while (ci.hasNext()) {
Cat cc = (Cat) ci.next();
s.push(cc);
}
}
}
}
protected Class getEntityClass() {
return Cat.class;
}
public List findDirectChildren(String catId) throws DataAccessException {
return this.getHibernateTemplate().find(
"select cat from Cat as cat where cat.parent.id=?",
new Long(catId));
}
public List findRootCats() throws DataAccessException {
return this.getHibernateTemplate().find(
"select cat from Cat as cat where cat.parent=null");
}
}
IOwnerDao.java和OwnerDaoImpl.java
可以看出它们基本上等于空,OwnerDaoImpl只有一个方法:
protected Class getEntityClass() {
return Owner.class;
}
假如基类IEntityDao设计得很好的话,子类的很多方法很多方法都可以省了。
java代码:
package org.ggyy.dao;
/**
* @author jiangyubao
*OwnerDao的接口
*/
public interface IOwnerDao extends IEntityDao{
}
java代码:
package org.ggyy.dao.hibernate;
import org.ggyy.bo.Owner;
import org.ggyy.dao.IOwnerDao;
/**OwnerDao的实现
* @author jiangyubao
* @spring.bean id="ownerDao"
* @spring.property name="sessionFactory" ref="sessionFactory"
*/
public class OwnerDaoImpl extends EntityDaoImpl implements IOwnerDao {
protected Class getEntityClass() {
return Owner.class;
}
}
ps:
question: 谢谢楼主的共享,请问能不能给出比较详尽的ApplicationContext.xml文件内容?
answer:
这个文件其实是使用ant 运行 一下那个src/springBuild.xml文件就可以生成的了。运行方法:选中src/springBuild.xml,右键,选择run->ant build
本文来自互联网用户投稿,文章观点仅代表作者本人,不代表本站立场,不承担相关法律责任。如若转载,请注明出处。 如若内容造成侵权/违法违规/事实不符,请点击【内容举报】进行投诉反馈!
