6.1整合思路分析
Java Web应用开发经过多年的发展,已经形成了一套成熟的程序结构。一个典型的使用了SpringMVC和Hibernate框架的应用,其结构如下图所示。
SpringMVC+Hibernate应用的程序结构
SpringMVC的主控制器DispatcherServlet接到请求后会调用特定的Controller。在Controller中又会调用业务类(Service)来执行业务逻辑。如果需要访问数据库,业务类则会继续调用数据库访问对象(DAO)。而在数据库访问对象中,则需要调用SessionFactory提供的Session实例的方法执行具体操作。Session最终会通过Connection等JDBC API来实现增删该查等操作,而Connction可以通过配置的数据源(DataSource)来提供。
通过以上分析不难看出,程序执行过程中依赖的方向是Controller--->Service--->DAO--->Session(由SessionFactory提供)--->Connection(由DataSource提供),当使用Spring IOC进行依赖管理时,依赖注入的方向则正好与相反。下面通过对核心配置文件的讲解及功能点的展示来讲解Spring+SpringMVC+Hibernate框架的搭建过程。
6.2 泛型介绍
泛型,即“参数化类型”。一提到参数,最熟悉的就是定义方法时有形参,然后调用此方法时传递实参。那么参数化类型怎么理解呢?顾名思义,就是将类型由原来的具体的类型参数化,类似于方法中的变量参数,此时类型也定义成参数形式(可以称之为类型形参),然后在使用/调用时传入具体的类型(类型实参)。
使用泛型可以让我们简化大量类似代码。例如我们要写一个方法交换两个整数,我们可以定义方法:public void swap(int a,int b){},如果我们又要求交换两个double型数据,交换两个String型数据,按照我们一般的思路,我们就要定义方法public void swap(doublea,double b){},public void swap(String a,String b){}。但是如果我们使用泛型,我们定义一个方法即可:public void swap(T a,T b){}。同样道理,我们在做业务开发的时候,我们可能需要对每张表都编写方法getById, save, update,deleteById …,这样的话就会重复大量的代码,带来大量重复的工作。如果我们使用实体类结合泛型,就能很容易避免这样的问题,让项目开发如虎添翼。
6.3注解简介
注释配置相对于 XML 配置具有很多的优势:
(1)它可以充分利用 Java 的反射机制获取类结构信息,这些信息可以有效减少配置的工作。如使用 JPA 注释配置 ORM 映射时,我们就不需要指定 PO 的属性名、类型等信息,如果关系表字段和 PO 属性名、类型都一致,您甚至无需编写任务属性映射信息——因为这些信息都可以通过 Java 反射机制获取。
(2)注释和 Java 代码位于一个文件中,而 XML 配置采用独立的配置文件,大多数配置信息在程序开发完成后都不会调整,如果配置信息和 Java 代码放在一起,有助于增强程序的内聚性。而采用独立的 XML 配置文件,程序员在编写一个功能时,往往需要在程序文件和配置文件中不停切换,这种思维上的不连贯会降低开发效率。
因此在很多情况下,注释配置比 XML 配置更受欢迎,注释配置有进一步流行的趋势。但是大量使用注解会影响项目后期维护,因为只要改了一点点配置,就要修改源代码,整个项目需要重新编译再上线。那么,到底是使用注释配置还是XML 配置呢?这就要取决于整个项目的取舍。
5.3 开发环境
开发工具:myeclipse8.6.1
数据库:mysql5.5.23
服务器: tomcat6.0.37
框架版本: spring3.2.2+hibernate3.3.2
5.4 案例开发步骤
步骤一:在myeclipse8.6中新建web工程AnnoGenericSHMvc,拷贝如下包到lib目下:
代码的目录结构如下图所示:
步骤二:编写web.xml 配置文件,代码如下:
=================================web.xml========================
<?xml version="1.0"encoding="UTF-8"?>
<web-app version="2.5"
xmlns="http://java.sun.com/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee
http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd">
<!-- 配置统一错误处理页面start -->
<error-page>
<error-code>404</error-code>
<location>/Err404.html</location>
</error-page>
<error-page>
<error-code>500</error-code>
<location>/Err500.html</location>
</error-page>
<!-- 配置统一错误处理页面end -->
<!-- 配置登录验证过滤器start -->
<filter>
<filter-name>loginFilter</filter-name>
<filter-class>com.wx.filters.LoginFilter</filter-class>
<!-- 不做拦截的请求-->
<init-param>
<param-name>exclusions</param-name>
<param-value>AnnoGenericSHMvc/,.js,.gif,.jpg,.jpeg,.png,.css,.ico,
Login.jsp,Login.php,Register.jsp,register.php,CheckUser.php,
ShowOneUser.jsp,ShowResult.jsp,Trans.php,ShowOneUser.php
</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>loginFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
<!-- 配置登录验证过滤器end -->
<!--druid连接池监控 config start -->
<filter>
<filter-name>DruidWebStatFilter</filter-name>
<filter-class>com.alibaba.druid.support.http.WebStatFilter</filter-class>
<init-param>
<param-name>exclusions</param-name>
<param-value>*.js,*.gif,*.jpg,*.png,*.css,*.ico,/druid/*</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>DruidWebStatFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
<servlet>
<servlet-name>DruidStatView</servlet-name>
<servlet-class>com.alibaba.druid.support.http.StatViewServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>DruidStatView</servlet-name>
<url-pattern>/druid/*</url-pattern>
</servlet-mapping>
<!--http://localhost:8080/AnnoGenericSHMvc/druid/index.html end-->
<!-- 解决hibernate延迟加载问题的处理 start -->
<filter>
<filter-name>openSessionInViewFilter</filter-name>
<filter-class>org.springframework.orm.hibernate3.support.OpenSessionInViewFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>openSessionInViewFilter</filter-name>
<url-pattern>*.php</url-pattern>
</filter-mapping>
<!-- 解决hibernate延迟加载问题的处理 end -->
<!-- 配置spring mvc的字符集过滤 start-->
<filter>
<filter-name>encode</filter-name>
<filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
<init-param>
<param-name>encoding</param-name>
<param-value>UTF-8</param-value>
</init-param>
<init-param>
<param-name>forceEncoding</param-name>
<param-value>true</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>encode</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
<!-- 配置spring mvc的字符集过滤 end-->
<!--配置 springmvc 前端控制器 start-->
<servlet>
<servlet-name>springmvc</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:configs/mvc/springmvc-dispatcher.xml</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>springmvc</servlet-name>
<url-pattern>*.php</url-pattern>
</servlet-mapping>
<!--配置 springmvc 前端控制器 end-->
<!-- 加载log4jstart -->
<context-param>
<param-name>log4jConfigLocation</param-name>
<param-value>classpath:configs/log4j.properties</param-value>
</context-param>
<listener>
<listener-class>org.springframework.web.util.Log4jConfigListener</listener-class>
</listener>
<!-- 加载log4j end-->
<!-- 加载spring容器 start-->
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:configs/applicationContext.xml</param-value>
</context-param>
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<!-- 加载spring容器 end-->
<welcome-file-list>
<welcome-file>Login.jsp</welcome-file>
</welcome-file-list>
</web-app>
================================================================
web.xml是我们web工程的核心配置所在。在这里,我们配置了统一错误处理、登录验证过滤器、druid连接池监控、解决hibernate延迟加载问题的过滤器、spring mvc的字符集过滤、spring mvc前端控制器、加载log4j、加载spring容器等相关配置。相关代码请看web.xml的注释部分。
重点知识讲解:(1) Spring需要启动容器才能为其他框架提供服务,由于Web应用程序的人口是被Web服务器控制的,所以,无法在main()方法中通过创建ClassPallXmIApplicationContext对象来启动Spring容器。Spring提供了一个监听器类org.springframework.web.context.ContextLoaderListener来解决这个问题。该监听器实现了ServletContextListener接口,可以在Web容器启动的时候初始化Spring容器。当然,前提是需要在web.xml中配置好这个监听器。具体请参见上面配置代码及相关的注释部分。
(2) Spring为我们提供了一个名为OpenSessionInViewFilter的过滤器,可以和前面提到的事务管理器及HibernateDaoSupport很好地配合。其作用是把一个Hibernate Session和一次完整的请求过程相绑定,在请求开始时开启Session,请求结束时关闭Session。这使得在一次请求的完整周期中,所使用的HibernateSession是唯一的且一直保持开启的可用状态,比较简便地解决了诸如延迟加载等问题。具体请参见上面的配置代码及相关的注释部分。
步骤三:建立hibernate二级缓存的配置文件,如下所示:
=========================echache.xml==============================
<ehcache>
<diskStore path="java.io.tmpdir"/>
<defaultCache
maxElementsInMemory="100"
eternal="false"
timeToIdleSeconds="120"
timeToLiveSeconds="120"
overflowToDisk="true"
/>
</ehcache>
=====================================================================
步骤四:建立hibernate核心配置文件,如下所示:
======================hibernate.cfg.xml==============================
<!DOCTYPE hibernate-configuration PUBLIC
"-//Hibernate/Hibernate Configuration DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
<session-factory>
<!-- 配置数据库方言(即数据库的类型) -->
<property name="dialect">
org.hibernate.dialect.MySQLDialect
</property>
<!-- 把会话绑定到线程上 -->
<property name="current_session_context_class">thread</property>
<!-- 是否展示sql -->
<property name="show_sql">true</property>
<!-- 是否格式化sql -->
<property name="format_sql">true</property>
<!-- 二级缓存的提供类 -->
<property name="hibernate.cache.provider_class">
org.hibernate.cache.EhCacheProvider
</property>
<!-- 二级缓存配置文件的路径-->
<property name="hibernate.cache.provider_configuration_file_resource_path">configs/ehcache.xml</property>
<!-- 是否开启二级缓存 -->
<property name="hibernate.cache.use_second_level_cache">true</property>
<!-- 是否开启查询缓存 -->
<property name="hibernate.cache.use_query_cache">true</property>
</session-factory>
</hibernate-configuration>
=====================================================================
步骤五:建立druid数据源的属性文件,如下所示:
========================= druidConfig.properties=====================
driverClassName=com.mysql.jdbc.Driver
url=jdbc\:mysql\://localhost\:3306/schooldb
username=root
password=123456
#配置监控统计拦截的filters,去掉后监控界面sql无法统计
filters=stat
#配置初始化大小
initialSize=6
#配置初始化最大连接数
maxActive=20
#配置最小空闲连接数
minIdle=3
#配置获取连接等待超时的时间,1分钟
maxWait=60000
#检测连接是否有效的SQL
validationQuery=SELECT 'x'
#申请连接的时候检测,如果空闲时间大于timeBetweenEvictionRunsMillis,执行validationQuery检测连接是否有效
testWhileIdle=true
#申请连接时执行validationQuery检测连接是否有效,做了这个配置会降低性能
testOnBorrow=false
#归还连接时执行validationQuery检测连接是否有效,做了这个配置会降低性能
testOnReturn=false
#启用PSCache,必须配置大于0,当大于0时,poolPreparedStatements自动触发修改为true
maxPoolPreparedStatementPerConnectionSize=20
#对于长时间不使用的连接强制关闭
removeAbandoned=true
#超过30秒的空闲连接就可以被关闭了,单位是秒
removeAbandonedTimeout=30
#配置间隔多久才进行一次检测,检测需要关闭的空闲连接,单位是毫秒
timeBetweenEvictionRunsMillis=10000
#配置一个连接在池中最小生存的时间,单位是毫秒
minEvictableIdleTimeMillis=30000
=====================================================================
步骤六:建立log4j的属性文件,如下所示:
========================= log4j.properties=====================
log4j.rootLogger=INFO, Console
#Console
log4j.appender.Console=org.apache.log4j.ConsoleAppender
log4j.appender.Console.layout=org.apache.log4j.PatternLayout
log4j.appender.Console.layout.ConversionPattern=%d [%t] %-5p [%c] %m%n
=====================================================================
步骤七:建立spring的核心配置文件,如下所示:
========================= applicationContext.xml=====================
<?xml version="1.0"encoding="UTF-8"?>
<beans
xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:p="http://www.springframework.org/schema/p"
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:tx="http://www.springframework.org/schema/tx"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-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/tx
http://www.springframework.org/schema/tx/spring-tx-3.0.xsd">
<!-- 扫描包含注解类所在的包 -->
<context:component-scan base-package="com.wx.dao.*">
</context:component-scan>
<context:component-scan base-package="com.wx.service">
</context:component-scan>
<!-- 加载druid数据源的属性文件 -->
<bean id="config" class="org.springframework.beans.factory.config.PreferencesPlaceholderConfigurer">
<property name="location">
<value>classpath:configs/druidConfig.properties</value>
</property>
</bean>
<!-- druid数据源的配置 -->
<bean id="druidDataSource" class="com.alibaba.druid.pool.DruidDataSource"
init-method="init"destroy-method="close">
<property name="driverClassName"value="${driverClassName}" />
<property name="url"value="${url}" />
<property name="username"value="${username}" />
<property name="password"value="${password}" />
<property name="filters"value="${filters}" />
<property name="initialSize"value="${initialSize}" />
<property name="maxActive"value="${maxActive}" />
<property name="minIdle"value="${minIdle}" />
<property name="maxWait"value="${maxWait}" />
<property name="validationQuery"value="${validationQuery}" />
<property name="testWhileIdle"value="${testWhileIdle}" />
<property name="testOnBorrow"value="${testOnBorrow}" />
<property name="testOnReturn"value="${testOnReturn}" />
<property name="maxPoolPreparedStatementPerConnectionSize"value="${maxPoolPreparedStatementPerConnectionSize}"/>
<property name="removeAbandoned"value="${removeAbandoned}" />
<property name="removeAbandonedTimeout"value="${removeAbandonedTimeout}" />
<property name="timeBetweenEvictionRunsMillis"value="${timeBetweenEvictionRunsMillis}" />
<property name="minEvictableIdleTimeMillis"value="${minEvictableIdleTimeMillis}" />
</bean>
<!-- 配置sessionFactory ,即在这里整合hibernate-->
<bean id="sessionFactory" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
<property name="dataSource"ref="druidDataSource"></property>
<property name="configLocation">
<value>classpath:configs/hibernate.cfg.xml</value>
</property>
<property name="mappingLocations">
<list>
<value>classpath:com/wx/entitys/*.hbm.xml</value>
</list>
</property>
</bean>
<!-- 配置sessionFactory end-->
<!-- 统一事务处理的配置start -->
<bean id="txManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager">
<property name="sessionFactory"ref="sessionFactory"></property>
</bean>
<tx:advice id="txAdvise" transaction-manager="txManager">
<tx:attributes>
<tx:method name="find*"read-only="true"/>
<tx:method name="search*"read-only="true"/>
<tx:method name="query*"read-only="true"/>
<tx:method name="get*"read-only="true"/>
<tx:method name="add*"propagation="REQUIRED"/>
<tx:method name="save*"propagation="REQUIRED"/>
<tx:method name="do*"propagation="REQUIRED"/>
<tx:method name="update*"propagation="REQUIRED"/>
<tx:method name="del*" propagation="REQUIRED"/>
</tx:attributes>
</tx:advice>
<aop:config>
<aop:pointcut expression="execution(*com.wx.service.*.*(..))" id="myCut"/>
<aop:advisor advice-ref="txAdvise"pointcut-ref="myCut"/>
</aop:config>
<!-- 统一事务处理的配置end -->
</beans>
=====================================================================
步骤十:编写spring mvc的核心配置文件springmvc-dispatcher.xml,如下所示:
==================== springmvc-dispatcher.xml==================
<?xml version="1.0"encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-3.0.xsd">
<!-- 注意控制器的扫描不能放到 applicationContext.xml中进行配置-->
<context:component-scan base-package="com.wx.controler.*"></context:component-scan>
<!-- 视图解析器 -->
<bean id="irvResolver"
class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="prefix" value="/"></property>
<property name="suffix" value=".jsp"></property>
</bean>
</beans>
=====================================================================
上面列出了所有的配置文件及相关注释,接下来我们对整个框架使用泛型部分进行说明。泛型部分的代码即公共代码,我们结合一个用户模块对其进行说明。如下面的UML类图所示:实线空闲箭头代表继承,虚线空心箭头代表实现。IBaseDao接口里面的5个方法对所有的表都通用。
步骤十二: 编写公共模块的代码如下
==================== IBaseDao.java==============================
package com.wx.dao;
import java.util.List;
/**
* 用泛型编写的增删改查公共接口
* @author汪祥
* @param<T>实体类
* 2015-09-11
*/
public interface IBaseDao<T> {
/**
* 根据id查一个实体类的对象
* @param entityCla 实体类的类型
* @param id 编号
* @return实体类的对象
*/
T getById(Class<T> entityCla,Object id);
/**
* 查询所有
* @param t 实体类的对象
* @return实体类对象列表
*/
List<T> getAll(T t);
/**
* 添加数据
* @param t 实体类对象
* @return添加的数据条数
*/
Integer save(T t);
/**
* 根据编号删除
* @param entityCla 实体类的类型
* @param id 编号
*/
void deleteById(Class<T> entityCla,Object id);
/**
* 修改
* @param t 实体类的对象
*/
void update(T t);
}
=====================================================================
==================== MyDaoSupport.java==============================
package com.wx.dao;
import org.hibernate.SessionFactory;
import org.springframework.beans.factory.annotation.Autowired;
importorg.springframework.orm.hibernate3.support.HibernateDaoSupport;
public class MyDaoSupport extends HibernateDaoSupport {
@Autowired //对方法的入参进行标注
public void setMySessionFactory(SessionFactory sessionFactory){
super.setSessionFactory(sessionFactory);
}
}
=====================================================================
==================== BaseDaoImpl.java==============================
package com.wx.dao;
import java.util.List;
/**
* 用泛型编写的增删改查公共类
* @author汪祥
* @param<T>实体类
* 2015-09-11
*/
public class BaseDaoImpl<T> extends MyDaoSupport implementsIBaseDao<T>{
/**
* 根据编号删除
* @param entityCla 实体类的类型
* @param id 编号
*/
public void deleteById(Class<T> entityCla,Object id) {
this.getHibernateTemplate().delete(this.getById(entityCla,id));
}
/**
* 查询所有
* @param t 实体类的对象
* @return实体类对象列表
*/
@SuppressWarnings("unchecked")
public List<T> getAll(T t) {
return (List<T>)this.getHibernateTemplate().loadAll(t.getClass());
}
/**
* 根据id查一个实体类的对象
* @param entityCla 实体类的类型
* @param id 编号
* @return实体类的对象
*/
public T getById(Class<T> entityCla,Object id) {
T t1=null;
if(id instanceof String){
t1=(T)this.getHibernateTemplate().load(entityCla, (String)id);
}else if(id instanceof Integer){
t1=(T)this.getHibernateTemplate().load(entityCla, (Integer)id);
}else if(id instanceof Long){
t1=(T)this.getHibernateTemplate().load(entityCla, (Long)id);
}else{
throw new RuntimeException("id类型错误");
}
return t1;
}
/**
* 添加数据
* @param t 实体类对象
* @return添加的数据条数
*/
public Integer save(T t) {
return (Integer)this.getHibernateTemplate().save(t);
}
/**
* 修改
* @param t 实体类的对象
*/
public void update(T t) {
this.getHibernateTemplate().update(t);
}
}
=====================================================================
==================== IUserDao.java==============================
package com.wx.dao.user;
import com.wx.dao.IBaseDao;
import com.wx.entitys.UserEntity;
public interface IUserDao extends IBaseDao<UserEntity>{
//登录
boolean isLogin(UserEntity user);
//判断用户是否存在(trun-存在,false-不存在)
boolean isUserExist(String userName);
}
=====================================================================
==================== UserDaoImpl.java==============================
package com.wx.dao.user;
import java.util.List;
import org.springframework.orm.hibernate3.HibernateTemplate;
import org.springframework.stereotype.Repository;
import com.wx.dao.BaseDaoImpl;
import com.wx.entitys.UserEntity;
@Repository("userDao")
public class UserDaoImpl extends BaseDaoImpl<UserEntity> implements IUserDao {
public boolean isLogin(UserEntity user) {
HibernateTemplate hiberUtil= this.getHibernateTemplate();
hiberUtil.setCacheQueries(true);
boolean flag=false;
Object[] params=new Object[]{user.getUserName(),user.getPassWord()};
List<UserEntity> userList=
hiberUtil.find("from UserEntity where userName=? and passWord=?",params);
if(userList!=null && userList.size()==1){
flag=true;
}
return flag;
}
public boolean isUserExist(String userName) {
HibernateTemplate hiberUtil= this.getHibernateTemplate();
hiberUtil.setCacheQueries(true);
boolean flag=false;
Object[] params=new Object[]{userName};
List<UserEntity> userList=
hiberUtil.find("from UserEntity where userName=?",params);
if(userList!=null && userList.size()>=1){
flag=true;
}
return flag;
}
}
=====================================================================
说明:其它的java类和jsp代码不一一列出,因为实在太多了,有需要请联系我。
步骤十三: 启动tomcat,发布项目,在地址栏分别输入 http://localhost:8080/ AnnoGenericSHMvc
进行测试。 登录后的结果如图所示:
联系作者: