java+se+ee+spring_【JavaEE】Springmvc+Spring+Hibernate整合及example

前面两篇文章,分别介绍了Springmvc和Spring的搭建方法,本文再搭建hibernate,并建立SSH最基本的代码结构。

Hibernate和前面两个比就比较复杂了,Hibernate是一个orm的框架,也就是负责面向对象中的对象(Object)和关系型数据库这个关系(Relation)之间的映射(Mapping)。因为关系型数据库的思维方式,和编程的时候对于对象的理解是有偏差的,所以也有一些面向对象的数据库,但是随着这些orm框架的完善,面向对象的数据库就销声匿迹了。

当然,我这篇文章也只是搭建环境,并不多讲annotation,这个会专门写一篇来讲。

闲话少说,第一步是安装好数据库,我用的是mysql,安装在本地,本开启了mysql服务。之后看程序怎么改进。

1. pom.xml

第一步加入hibernate的相关jar包,首先在properties标签下面定义一个版本:

4.0.4.RELEASE

4.3.5.Final

然后加入hibernate的核心包,因为要让Spring管理一些对象,还要引入spring-orm的包:

org.springframework

spring-orm

${spring.version}

org.hibernate

hibernate-core

${hibernate.version}

除此之外,因为我们要用mysql数据库,并使用c3p0连接池,因此需要引入这两个对应的包:

c3p0

c3p0

0.9.1.2

mysql

mysql-connector-java

5.1.31

2. web.xml

hibernate有两种使用方法,一种是通过一个SessionFactory来获取Session,还有一种是通过EntityManagerFactory来获取entityManager,我忘了什么原因了,刚开始用的EntityManager后来换回了SessionFactory(hibernate3以前用的是SessionFactory),我后面的描述也都是基于sessionFactory了。而不管是Session还是EntityManager,我们都要去创建和管理这个对象,当然,有了Spring,这个事情就不用我们来操心了,只需要在Spring的配置文件里面配置一下我需要的SessionFactory就行了,不过为了每个文件专注一个事情,我又单独配了一个配置文件,而不是用之前Spring的那一个applicationContext.xml,修改之前的contextConfigLocation,加一个新的文件位置:

contextConfigLocation

classpath:/META-INF/applicationContext.xml,

classpath:/META-INF/infrastructure.xml

除此之外,hibernate本质上不需要在web.xml里面配置什么,但是有一个属性,我觉得很重要,hibernate对对象的管理是基于session的,如果开启了延迟加载,对于关联对象的查询可能会在渲染jsp的时候才发生,但是这个时候hibernate的当前session默认已经关闭了,就会抛异常,所以我们经常需要hibernate在渲染jsp页面的时候还开着session,这就需要在web.xml中配置一个拦截所有请求的filter:

hibernateFilter

org.springframework.orm.hibernate4.support.OpenSessionInViewFilter

hibernateFilter

/*

3. infrastructure.xml

在src/main/resources这个资源包的META-INF目录下创建infrastructure.xml,在这里配置hibernate的SessionFactory:

http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-3.0.xsd

http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.0.xsd

http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.0.xsd

">

${hibernate.hbm2ddl.auto}

${hibernate.dialect}

${hibernate.show_sql}

${hibernate.format_sql}

首先要主意我标称玫红的部分,这是实体类的存放package,hibernate就是用这下面的实体类和数据表对应,需要修改成自己的package,可以看到这个文件还是比较长,不利于改参数,所以继续抽取一个属性文件出来,property-placeholder就是这个文件的位置,之后配置一个数据源,用c3p0连接池连接数据库,之后再根据这个数据源配置SessionFactory,进而配置事务,配置SessionFactory的时候有一个属性hibernate.hbm2dll.auto,是配置根据我们对类的定义自动生成数据表及主外键关系的,刚创建工程的时候设置成create,以后可以改成update或者删了这个属性(刚建工程的时候就设置成update也可以自己创建数据表,但是不会打印建表的sql语句),下面来看这个抽取出来的属性文件,在同目录下创建子目录properties,再创建hibernate.properties:

hibernate.dialect=org.hibernate.dialect.MySQLDialect

driverClassName=com.mysql.jdbc.Driver

validationQuery=SELECT 1

url=jdbc:mysql://localhost:3306/test?useUnicode=true&characterEncoding=UTF-8

username=root

password=

hibernate.hbm2ddl.auto=create

hibernate.show_sql=true

hibernate.format_sql=true

4. applicationContext.xml

还需要改这个文件吗?需要改一点,这个文件之前只有一行:

SSH最基本的代码结构呢,需要有实体类,也就是持久化对象,这是一些和数据表一一对应的类,这些类我们放在domain这个子package下,然后访问数据库要使用DAO,即数据访问对象,我们为它们创建一个repo包,再加上之前已经有的serivce和controller,基本架子就全了,所以,不光service要交给spring来管理,repo和domain都要让spring来扫描:

5. domain.User/repo.UserDao/service.UserService

首先在domain包下创建实体类User,我们就用最简单的只有两个字段,id和username:

@Entity

@Tablepublic classUser implements Serializable {

private static final long serialVersionUID = 172643386440351811L;private intid;privateString username;

@Id

@GeneratedValuepublic intgetId() {returnid;

}public void setId(intid) {this.id =id;

}

@Size(min=6)publicString getUsername() {returnusername;

}public voidsetUsername(String username) {this.username =username;

}

}

这里的annotation用的都是JPA的接口,所以都是javax.persistence包下的,annotation加在变量上和getter方法上都可以,只不过加在public域更好一些,这儿为User对象定义了两个成员,id是主键、自增长,还有一个username,并对username做了一个限制,比如长度超过6才可以。

之后定义一个dao来访问数据库,创建一个新的package叫做repo(org.zhangfc.demo4ssh.repo),创建UserDao:

public interfaceUserDao {public intsave(User u);public ListfindAll();

}

创建实现类UserDaoImpl:

@Repositorypublic class UserDaoImpl implementsUserDao {

@AutowiredprivateSessionFactory sessionFactory;public ListfindAll() {

Criteria criteria= sessionFactory.getCurrentSession().createCriteria(User.class);returncriteria.list();

}

}

首先一个annotation注明这是一个repository,需要被Spring管理,然后自动装载之前在配置文件中配置的SessionFactory,获取当前session后获取所有的用户,下面修改之前的UserService代码,service封装业务逻辑层代码,我把每个service方法封装为一个事务,注:上面的SessionFactory获取当前Session是依赖于事务的,如果不在某个事务之内,会报错:No Session found for current thread。

UserService:

public interfaceUserService {public void saveUsers(Listus);public ListgetAllUsers();

}

UserServiceImpl:

@Service

@Transactionalpublic class UserServiceImpl implementsUserService {

@AutowiredprivateUserDao userDao;public void saveUsers(Listus) {for(User u : us) {

userDao.save(u);

}

}public ListgetAllUsers() {returnuserDao.findAll();

}

}

这个Transactional的annotation就是指明每个方法是一个事务。最后修改HomeController:

@RequestMapping("")publicString home(){

List us = new ArrayList();

User u= newUser();

u.setUsername("zhangsan");

us.add(u);

u= newUser();

u.setUsername("wangwu");

us.add(u);

userService.saveUsers(us);return "index";

}

@RequestMapping("/json")

@ResponseBodypublic Listjson(){returnuserService.getAllUsers();

}

之后运行程序,数据库会自动创建user表(注意,hibernate.hbm2dll.auto=create的话,每次运行程序都会删了之前的表重建,设置为update也会自动建表,且如果数据表已存在但和类的设置不同,会更新数据表,把类中声明的新的东西加上,但是数据表里有、类里面没有的不会在现有的数据表中删除),运行起来之后,访问http://localhost:8080/demo4ssh/会向user表中插入两条记录,因为userService的方法设置了事务,如果其中的任意一条sql执行失败,都将回滚,比如把wangwu改成lisi,因为长度不够,所以第二次插入时失败,那么zhangsan也不会被插入到数据表中。之后再访问http://localhost:8080/demo4ssh/json:

[{"id":1,"username":"zhangsan"},{"id":2,"username":"wangwu"}]

到此为止,SSH的基本框架就搭建出来了。源码下载。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值