一、SSH三大框架的概述
ssh为 struts+spring+hibernate的一个集成框架,是目前较流行的一种Web应用程序开源框架。
集成SSH框架的系统从职责上分为四层:表示层、业务逻辑层、数据持久层和域模块层(实体层),以帮助开发人员在短期内搭建结构清晰、可复用性好、维护方便的Web应用程序。
struts2:
1、什么是struts2:
Struts2是一个基于MVC设计模式的Web应用框架,它本质上相当于一个servlet,在MVC设计模式中,Struts2作为控制器(Controller)来建立模型与视图的数据交互。Struts 2是Struts的下一代产品,是在 struts 1和WebWork的技术基础上进行了合并的全新的Struts 2框架。其全新的Struts 2的体系结构与Struts 1的体系结构差别巨大。Struts 2以WebWork为核心,采用拦截器的机制来处理用户的请求,这样的设计也使得业务逻辑控制器能够与ServletAPI完全脱离开,所以Struts 2可以理解为WebWork的更新产品。虽然从Struts 1到Struts 2有着太大的变化,但是相对于WebWork,Struts 2的变化很小。
2、struts2框架的运行结构:
解析:客户端发送请求(HttpServletRequest)到服务器,服务器接收到请求就先进入web.xml配置文件看看有没有配置过滤器,发现有struts2的过滤器,然后就找到struts.xml配置文件,struts.xml配置文件里有定义一个action,然后就去找到类名叫IndexAction这个类(此action类必须是继承ActionSupport接口),并且实现了execute()方法,返回一个字符串为"success"给struts.xml配置文件,struts.xml配置文件的action会默认调用IndexAction类的execute()方法,result接收到了返回的字符串,然后查找结果字符串对应的(Result),result就会调用你指定的jsp页面将结果呈现,最后响应回给客户端。
spring:
1、什么是spring?
Spring是一个轻量级控制反转(IoC)和面向切面(AOP)的容器框架。
2、spring的流程图:
解析:上图是在struts结构图的基础上加入了spring流程图,在web.xml配置文件中加入了spring的监听器,在struts.xml配置文件中添加“<constant name="struts.objectFactory" value="spring" />”是告知Struts2运行时使用Spring来创建对象,spring在其中主要做的就是注入实例,将所有需要类的实例都由spring管理。
hibernate:
1、什么是hibernate?
Hibernate是一个开放源代码的对象关系映射框架,它对JDBC进行了非常轻量级的对象封装,它将POJO与数据库表建立映射关系,是一个全自动的orm框架,hibernate可以自动生成SQL语句,自动执行,使得Java程序员可以随心所欲的使用对象编程思维来操纵数据库。 Hibernate可以应用在任何使用JDBC的场合,既可以在Java的客户端程序使用,也可以在Servlet/JSP的Web应用中使用,最具革命意义的是,Hibernate可以在应用EJB的J2EE架构中取代CMP,完成数据持久化的重任。
2、hibernate的核心构成和执行流程图:
3、为什么使用Hibernate?
1). 对JDBC访问数据库的代码做了封装,大大简化了数据访问层繁琐的重复性代码。
2)、Hibernate是一个优秀的ORM实现。他很大程度的简化DAO层的编码工作,将软件开发人员从大量相同的数据持久层相关编程工作中解放出来,使开发更对象化了。
3)、移植性好,支持各种数据库,如果换个数据库只要在配置文件中变换配置就可以了,不用改变hibernate代码。
4)、支持透明持久化,因为hibernate操作的是纯粹的(pojo)java类,没有实现任何接口,没有侵入性。所以说它是一个轻量级框架。
二、搭建一个完整的SSH框架项目。
注意事项:
1、本文提纲:本文通过一个将所有图书借记卡信息查询出来并显示到JSP页面的项目实例讲解SSH的整合。创建Struts项目,整合Hibernate,整合Spring。
2、仅是创建SSH项目,对于其他的扩展例如Struts的国际化,Hibernate的缓存优化,Spring的AOP等,本博文涉及不到。想学习更多的东西请搜索其他博文。
3、本项目的搭建环境:
Windows 10-64位,Eclipse(开发工具),jdk1.8.0_91,Tomcat 8.0,struts-2.3.30-apps,spring-framework-4.2.2.RELEASE,hibernate-release-5.2.2.Final,mysql.
第一步:在eclipse(开发工具)里创建web项目(项目名称:ssh),并生成web.xml文件。
第二步:导入本次项目要使用到的jar包。
struts-2.3.30-apps官网下载地址:http://struts.apache.org/download.cgi#struts252
spring-framework-4.2.2.RELEASE官网下载地址:https://repo.spring.io/webapp/#/artifacts/browse/tree/General/libs-release-local/org/springframework/spring/4.2.2.RELEASE/spring-framework-4.2.2.RELEASE-dist.zip
hibernate-release-5.2.2.Final官网下载地址:http://hibernate.org/orm/
将所有的项目要用的jar包放入lib文件里:(导入的jar包需版本准确,否则可能会导致使用方法不一致)
第三步:在配置文件web.xml配置一个struts2的过滤器和spring监听器。
<?xml version="1.0" encoding="UTF-8"?> <web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://xmlns.jcp.org/xml/ns/javaee" xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd" id="WebApp_ID" version="3.1"> <display-name>ssh</display-name> <welcome-file-list> <welcome-file>index.action</welcome-file> </welcome-file-list> <!-- struts2的过滤器 --> <filter> <filter-name>struts2</filter-name> <filter-class>org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter</filter-class> <!-- struts2.5版本使用的过滤器 --> <filter-class>org.apache.struts2.dispatcher.filter.StrutsPrepareAndExecuteFilter</filter-class> --> </filter> <filter-mapping> <filter-name>struts2</filter-name> <url-pattern>/*</url-pattern> </filter-mapping> <!-- spring的监听器配置开始 --> <context-param> <param-name>contextConfigLocation</param-name> <param-value>classpath:applicationContext.xml</param-value> </context-param> <listener> <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> </listener> </web-app>
第四步:在Java Resources下的src目录下创建四个包(package)进行分层。
第五步:根据数据库表的字段编写BookCard(实体类)和BookCard.hbm.xml(映射文件)放到ssh.entity包里。
BookCard(实体类):
package ssh.entity; import java.math.BigDecimal; import java.util.Date; /* * 跟数据库表一致,作为一个java对象 * 1个对象代表的是数据库表中的一行记录 * 1个属性代表的是表中的一个字段 */ public class BookCard { private int cid ; private String name; private String sex ; private Date cardDate; private BigDecimal deposit; //定义get()、set()方法 public int getCid() { return cid; } public void setCid(int cid) { this.cid = cid; } public String getName() { return name; } public void setName(String name) { this.name = name; } public String getSex() { return sex; } public void setSex(String sex) { this.sex = sex; } public Date getCardDate() { return cardDate; } public void setCardDate(Date cardDate) { this.cardDate = cardDate; } public BigDecimal getDeposit() { return deposit; } public void setDeposit(BigDecimal deposit) { this.deposit = deposit; } }
BookCard.hbm.xml(映射文件):
<?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN" "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd"> <hibernate-mapping> <class name="ssh.entity.BookCard" table="BookCard"> <!-- 卡号 --> <id name="cid" column="cid"> <generator class="native"></generator> </id> <!-- 姓名 --> <property name="name" column="name"></property> <!-- 性别 --> <property name="sex" column="sex"></property> <!-- 办卡日期 --> <property name="cardDate" column="cardDate"></property> <!-- 押金 --> <property name="deposit" column="deposit"></property> </class> </hibernate-mapping>
第六步:在ssh.service包里编写IndexService(接口类)和IndexServiceImpl(实现类)。
IndexService(接口类):
package ssh.service; import java.util.List; import ssh.dao.IndexDao; import ssh.entity.BookCard; //创建一个IndexService接口类 public interface IndexService { public List<BookCard> getAllBookCard(); }
IndexServiceImpl(实现类):
package ssh.service; import java.util.List; import ssh.dao.IndexDao; import ssh.entity.BookCard; //创建IndexServiceImpl(实现类)实现IndexService接口 public class IndexServiceImpl implements IndexService { //dao实例使用注入方式 private IndexDao id; //用于注入使用 public void setId(IndexDao id) { this.id = id; } @Override public List<BookCard> getAllBookCard() { //本类应该编写业务逻辑的代码, //但本例没有业务逻辑,就不用写。 //访问数据库的代码,不会出现在service这一层 //交给dao来操作数据库 List<BookCard> myBookCardList = id.getAllBookCard(); //进行其它的业务逻辑操作,比如增加多一个选项,是否过期 //本例不需要 //.... return myBookCardList; } }
第七步:在ssh.dao包里编写IndexDao(接口类)和IndexDaoImpl(实现类)。
IndexDao(接口类):
package ssh.dao; import java.util.List; import ssh.entity.BookCard; //创建IndexDao(接口类) public interface IndexDao { public List<BookCard> getAllBookCard(); }
IndexDaoImpl(实现类):
package ssh.dao; import java.util.List; import org.hibernate.Session; import org.hibernate.SessionFactory; import org.hibernate.query.Query; import ssh.entity.BookCard; //创建IndexDaoImpl(实现类)实现IndexDao接口 public class IndexDaoImpl implements IndexDao { //在SSH的设计理念:要使用某个实例,那么就定义声明一个对象,然后 //给它添加set方法(用于spring注入进来) //实现不要关注这个实例来自于那里,以及怎么创建,或者它是谁 private SessionFactory sessionFactory; public void setSessionFactory(SessionFactory sessionFactory) { this.sessionFactory = sessionFactory; } @Override public List<BookCard> getAllBookCard() { //sessionFactory这个实例可以自己按常规的hibernate传统写法创建 //也可以交给spring去托管 /* Configuration cfg = new Configuration().configure(); sessionFactory = cfg.buildSessionFactory();*/ //获取session Session session = sessionFactory.openSession(); //后面当使用JPA的时候,EntityManager 类似于 Session Query query = session.createQuery("from BookCard"); //将所有的数据查询出来并放到List集合里 List<BookCard> list = query.getResultList(); //将集合遍历循环 for(BookCard bookCard:list){ //打印输出到控制台 System.out.println(bookCard); } //关闭session session.close(); //关闭sessionFactory sessionFactory.close(); //返回list集合 return list; } }
第八步:编写IndexAction(action类)。
package ssh.action; import java.text.DecimalFormat; import java.util.List; import com.opensymphony.xwork2.ActionContext; import com.opensymphony.xwork2.ActionSupport; import ssh.entity.BookCard; import ssh.service.IndexService; //创建IndexAction(action类)继承ActionSupport接口 public class IndexAction extends ActionSupport { private static final long serialVersionUID = 1L; //声明service,但不给它创建具体的实现类的实例, private IndexService is = null; //添加set()方法 public void setIs(IndexService is) { this.is = is; } //编写execute()方法 public String execute() { //获取IndexService实例,调用getAllBookCard()方法 //将结果保存到List集合里 List<BookCard> myBookCardList = is.getAllBookCard(); //将查询出来的结构集打印到控制台 System.out.println("结果集:"+myBookCardList.size()); //获取Context上下文对象 ActionContext ac = ActionContext.getContext(); //将myBookCardList集合添加到上下文对象里 ac.put("myBookCardList", myBookCardList); //返回一个字符串 return "success"; } //金额格式转换 public String formatDouble(double s){ DecimalFormat fmat=new DecimalFormat("\u00A4##.0"); return fmat.format(s); } }
第九步:编写struts.xml(struts配置文件)、applicationContext.xml(spring配置文件)、hibernate.cfg.xml(hibernate配置文件)。
注:将这些配置文件放到src里。
struts.xml:
<?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE struts PUBLIC "-//Apache Software Foundation//DTD Struts Configuration 2.3//EN" "http://struts.apache.org/dtds/struts-2.3.dtd"> <!-- 上面的头,注意版本,从样例里复制过来 showcase.war\WEB-INF\src\java\struts.xml --> <struts> <!-- 告知Struts2运行时使用Spring来创建对象 --> <constant name="struts.objectFactory" value="spring" /> <!-- 第1步:先定义一个包 --> <package name="mypck001" extends="struts-default"> <!-- 第2步:定义一个action,配置跳转信息 name 类似于Servlet @WebServlet("/IndexServlet") http://xxxx/xxx/Index.action http://xxxx/xxx/Index class 对应于自己写的Action类 当不写method属性时,默认调用的是execute class="ssh.action.IndexAction" ** new ssh.action.IndexAction() 设计思想:关心了具体的实现类必须改为不要关注那个实现类 加入spring后,struts的action节点的class属性意义发生变化,直接引用spring帮忙创建的实例 --> <action name="Index" class="myIndexAction"> <!-- 跳转是forward/WEB-INF/是防止jsp不经过action就可以访问--> <!-- result接收返回的字符串,然后做对应的事情 --> <result name="success">/WEB-INF/jsp/index.jsp</result> </action> </package> </struts>
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:context="http://www.springframework.org/schema/context" xmlns:jee="http://www.springframework.org/schema/jee" xmlns:tx="http://www.springframework.org/schema/tx" xsi:schemaLocation=" http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.2.xsd http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.2.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.2.xsd http://www.springframework.org/schema/jee http://www.springframework.org/schema/jee/spring-jee-4.2.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-4.2.xsd"> <!-- 类似于财务部门一样,类就是钱,所有需要类的实例都由srping去管理 --> <bean id="myIndexAction" class="ssh.action.IndexAction" scope="prototype"> <!-- setIs(myIndexService) --> <property name="is" ref="myIndexService"/> </bean> <!-- myIndexService = new ssh.service.IndexServiceImpl() --> <bean id="myIndexService" class="ssh.service.IndexServiceImpl" scope="prototype"> <property name="id" ref="myIndexDao"/> </bean> <bean id="myIndexDao" class="ssh.dao.IndexDaoImpl" scope="prototype"> <!-- 把sessionFactory 注入给IndexDao --> <property name="sessionFactory" ref="sessionFactory" /> </bean> <!-- 添加sessionFactory bane ,注意,该类是Spring提供的 --> <bean id="sessionFactory" class="org.springframework.orm.hibernate5.LocalSessionFactoryBean" scope="prototype"> <!-- 注入Hibernate 配置文件路径,前面要加上 classpath:--> <property name="configLocation" value="classpath:hibernate.cfg.xml"/> </bean> </beans>
hibernate.cfg.xml:
<?xml version="1.0" encoding="utf-8"?> <!DOCTYPE hibernate-configuration PUBLIC "-//Hibernate/Hibernate Configuration DTD 3.0//EN" "http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd"> <hibernate-configuration> <session-factory> <!-- 数据库连接配置 --> <property name="connection.driver_class">com.mysql.jdbc.Driver</property> <property name="connection.url">jdbc:mysql://localhost:3306/sshdb</property> <property name="connection.username">sshdb</property> <property name="connection.password">M87biLbmCmGnAeka</property> <!-- 每个数据库都有1个,针对特定的关系型数据库生成优化的SQL --> <property name="dialect">org.hibernate.dialect.MySQLDialect</property> <!-- 设置默认的数据库连接池 --> <property name="connection.pool_size">5</property> <!-- 显示SQL --> <property name="show_sql">true</property> <!-- 格式化SQL --> <property name="format_sql">true</property> <!-- 根据schema更新数据表的工具 --> <property name="hbm2ddl.auto">update</property> <!-- 数据表映射配置文件 --> <mapping resource="ssh/entity/BookCard.hbm.xml"/> </session-factory> </hibernate-configuration>
第十步:创建一个index.jsp页面将所有数据取出来显示到页面上。
注:跳转是forward,将jsp放到/WEB-INF/是防止jsp不经过action就可以访问。
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> <%@ taglib uri="/struts-tags" prefix="s" %> <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"> <html> <head> <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> <title>Insert title here</title> </head> <body> <table border="1"> <tr> <td>卡号</td> <td>姓名</td> <td>性别</td> <td>办卡日期</td> <td>押金</td> </tr> <!-- 使用struts2标签库中的iterator将所有数据遍历循环显示出来 --> <s:iterator value="#myBookCardList" status="bcs"> <tr> <td><s:property value="cid"></s:property></td> <td><s:property value="name"></s:property></td> <td><s:property value="sex"></s:property></td> <td><s:date name="cardDate" format="yyyy年MM月dd日"></s:date></td> <td><s:property value="%{formatDouble(deposit)}"></s:property></td> </tr> </s:iterator> <!-- 判断查询出来等于0,就显示“没有查找到数据” --> <s:if test="myBookCardList.size()==0"> <tr> <td colspan="7">没有查找到数据</td> </tr> </s:if> </table> </body> </html>
运行结果:
浏览器显示:
控制台输出:
总结:在SSH中使用Struts作为系统的整体基础架构,负责MVC的分离,在Struts框架的模型部分,控制业务跳转,利用Hibernate框架对持久层提供支持,Spring做支持,支持struts和hibernate。具体做法是:用面向对象的分析方法根据需求提出一些模型,将这些模型实现为基本的Java对象,然后编写基本的DAO(Data Access Objects)接口,并给出Hibernate的DAO实现,采用Hibernate架构实现的DAO类来实现Java类与数据库之间的转换和访问,最后由Spring做支持,支持struts和hibernate。其实ssh框架最主要的本质是:“高内聚、低耦合”。
本文由一叶发布,不代表一叶立场,转载联系作者并注明出处:https://yiyeo.com/java/195.html