近年来,由于Struts2+Hibernate3+Spring3,这套SSH框架,Struts2屡次爆出安全漏洞,Hibernate就只会推行它HQL那套而越来越远离SQL查询关系数据库的本质,所以Spring+SpringMVC+Mybatis这套SSM框架悄然兴起,现阶段在Javaee领域,有种新项目必选SSM的趋势。本文将利用一个简单的例子,具体如下图所示,其实也就是把《【Struts2+Hibernate3+Spring3】利用SSH整合,完成打印用户表,用户登录、注册、修改密码系统》(点击打开链接)中的SSH例子再用SSM再做一次,同时视图尽可能地不杂糅其它技术,以最短的代码,给大家展示Spring、SpringMVC、Mybatis三者整合的例子。
一、SSM的下载和配置
在Eclipse新建一个JavaEE的Dynamic Web Project估计已经不用说了,接下来我就不用什么Maven了,自己配好所有必须的jar包,具体如下。
1、Spring和SpringMVC的下载和配置
其实Spring和SpringMVC所用的jar包都是相同的,毕竟Spring他希望自己一套吃遍Java的所有领域,但Spring做Java和xml交互、近年来SpringMVC做Java和JSP的交互做得最为突出,所以流行起来了。
Spring3.x的使用首先需要commons-logging-1.2.jar这玩意,这个commons-logging-1.2.jar是所有jar的前提,不然,你的java应用只有jar的话,spring根本启动不了,最直接的表现是配置了spring的javaee应用的tomcat启动不来。
所以,你要先下载commons-logging-1.2.jar。这玩意直接在Apache Tomcat的官网有,不是很明白为何不直接搞到Tomcat里面。
官网是:http://commons.apache.org/proper/commons-logging/download_logging.cgi
下载之后解压,直接取走里面的commons-logging-1.2.jar到你的java应用。
commons-logging-1.2-javadoc.jar是个API说明文档,并没有什么卵用。
接下来,才是下载Spring3.x。Spring的包全部放到http://repo.spring.io/里面了,在Spring的官网,http://spring.io你只会看到你并希望Maven下载和一大堆说明文档,顶多是最新版Github托管而已。
进入了spring的资源库之后,如下图,找到libs-release-local(这spring都不知道怎么想的,堂堂的javaee大厂,资源库里面的东西竟然不是按a-z排的)->org->springframework->spring->3.2.18 release然后选择右上角的download,下载Spring3.x的最终稳定版。
下载之后解压,在libs中,取走所有最终应用,关于API说明*-javadoc.jar、源代码*-sources.jar完全可以不需要。
2、JDBC和Mybatis的下载
JDBC请选择合适自己数据库的版本,本文用Mysql做例子,所以选择的是Mysql的JDBC。你首先需要两个jar,一个是连接Mysql必须的jar,这里以Mysql做例子,Orcale等数据库则换成相应的jar包即可,可以到Mysql的官网下载:https://dev.mysql.com/downloads/connector/j/,下载之后取走其中的mysql-connector-java-5.1.44-bin.jar到你的工程目录,一般在工程目录新建一个lib,放这些包。
然后下载Mybatis,最新的Mybatis是放在Github托管的,网址:https://github.com/mybatis/mybatis-3/releases,同样取走里面的核心jar,放到你的工程目录的WEB-INF\lib下。
3、Mybatis和Spring交互包
最后补上这个jar就OK了。
还需要在https://github.com/mybatis/spring/releases下载一个Spring连接Mybatis的jar包mybatis-spring-1.3.1.jar:
整合之后WEB-INF\lib文件夹如下所示:
二、目录结构和所使用的数据库表
按照上述过程配好在WEB-INF\lib配好jar包之后,先贴上我最终完工的工程目录结构:
还有我使用的数据库表,还是这张早已举个栗子举烂的Mysql中test数据库的User用户表,随便开个Mysql,建立张user表,数据库test弄出来的。
三、配置文件xml的编写
1、我们先从这个Javaee工程的总配置文件web.xml看起,所有Javaee工程必有这个文件。在这个文件中,主要是声明使用SpringMVC、Spring还有这个网站主要使用UTF-8编码,避免乱码。
SpringMVC
org.springframework.web.servlet.DispatcherServlet
SpringMVC
/
org.springframework.web.context.ContextLoaderListener
contextConfigLocation
/WEB-INF/applicationContext.xml
encodingFilter
org.springframework.web.filter.CharacterEncodingFilter
encoding
UTF-8
encodingFilter
/
这里的web.xml的初始化,不同于之前《【Spring+Mybatis】Spring整合Mybatis》(点击打开链接)中,指明在 Java工程Spring的核心配置文件applicationContext.xml所在目录时,在contextConfigLocation的值只写一个applicationContext.xml就行,必须写明/WEB-INF/applicationContext.xml,不完全写明,Javaee工程找applicationContext.xml会直接去网站根目录WebContent找,肯定找不到,同时会出现org.springframework.beans.factory.BeanCreationException之类的错误
2、接下来先看SpringMVC-servlet.xml,命名只能这样写,毕竟在web.xml,指明的servlet-name就是SpringMVC,这个文件我在《【SpringMVC】Helloworld》( 点击打开链接)已经详细写过了,还是那句话,这篇文章主要是介绍SSM整合的,要是每个技术都写一次,就显得没有重点了。这里除了要求SpringMVC在actios找servlet以外,同时要求Spring扫描Service层,是因为在action用到Service的类中的方法和数据库打交道,所以必须要求actions层注入Service。同时扫描test.actions和test.services。
这里/WEB-INF/configuration.xml也必须完全写明,不能只写configuration.xml,不然就会出现org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'org.mybatis.spring.mapper.MapperScannerConfigurer#0' defined in ServletContext resource [/WEB-INF/applicationContext.xml]: Cannot resolve reference to bean 'sqlSessionFactory' while setting bean property 'sqlSessionFactory'; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'sqlSessionFactory' defined in ServletContext resource [/WEB-INF/applicationContext.xml]: Invocation of init method failed; nested exception is java.io.FileNotFoundException: Could not open ServletContext resource [/configuration.xml]的报错,无法启动tomcat,具体如下图所示:
4、最后就是Mybatis的配置文件configuration.xml了,由于Spring的存在,让configuration.xml可以变得非常简单,不用像《【Mybatis】Helloworld》(点击打开链接)里面写得这么长,指出数据库和Java的映射关系Mapper的所在地即可。
四、工程核心文件的编写
经历了上面一大堆配置文件终于弄好整个Spring+SpringMVC+Mybatis的配置了,接下来开始分门别类地分层严格遵循MVC的思想编写整个Javaee工程的核心文件了。
1、首先完成刚才顺手建立的test.mapper.User.xml。该文件在《【Mybatis】Helloworld》(点击打开链接)详细说过了。主要是写关于该User表的主要操作和test.dao下的Java方法对应关系。这里分别有按照username查用户、增加一个用户、修改用户名对应密码的三个方法,也刚好展示Mybatis的增删改查了。
insert into
user(username,password) values(#{username},#{password})
update user set password=#{password} where username=#{username}
package test.dao;
import org.springframework.stereotype.Repository;
import test.entity.User;
@Repository
public interface UserDao {
public User findUsersByUsername(String username);
public int addUser(User user);
public int modifyPasswordByUsername(User user);
}
package test.entity;
public class User {
private Integer id;
private String username;
private String password;
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public String getUsername() {
return username;
}
public void setUsername(String username) {
this.username = username;
}
public String getPassword() {
return password;
}
public void setPassword(String password) {
this.password = password;
}
}
这里一样要严格注解一个@Service表示这是个事务层,不然Spring是直接识别不到报错的。
package test.services;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import test.dao.UserDao;
import test.entity.User;
@Service
public class UserServices {
@Autowired
private UserDao userDao;//自动注入的存在,无需getter和setter了
/*各个业务*/
//判断用户是否存在
public boolean isUserExist(String username) {
if (userDao.findUsersByUsername(username) == null) {
return false;
} else {
return true;
}
}
//根据提供的用户名拿密码
public String getPasswordByUsername(String username) {
return userDao.findUsersByUsername(username).getPassword();
}
//新增一个用户
public void addUser(String username, String password) {
User user = new User();
user.setUsername(username);
user.setPassword(password);
userDao.addUser(user);
}
//修改某一用户的密码
public void modifyPasswordByUsername(String username, String newpassword) {
User user = userDao.findUsersByUsername(username);
user.setUsername(username);
user.setPassword(newpassword);
userDao.modifyPasswordByUsername(user);
}
}
另外,大家也可以从这个文件看到Mybatis比Hibernate优秀的原因,Mybatis处理数据库根本就没有什么commit不commit的提交事务才能修改数据库事情!他就是一个简单的SQL过程,上面的调用的dao方法,你完全可以理解成,这其实就是mapper一条对数据库操作的sql语句,执行即更改。
5、之后是test.actions.WebAction.java,这里也就是网页的Action层了,这个文件也在《【SpringMVC】Helloworld》(点击打开链接)中详细表述过了。
这里在SSM同样要明确地加上@Controller注解表示这是个Action层。
这里也同时展示了SpringMVC中Java是如何拿前台的数据,和返回数据到前台的。拿前台的数据主要是靠@RequestParam("前台表单的name")+后台Java的变量名了这个注解,当然还有拿其它数据的方式,不过这个是拿Post方法提交的表单数据最常用了,当然在get提供也可以照样拿,其余几种拿前台数据的方式比较少用,大家有兴趣可以某度一下。@RequestMapping注解指明servlet的url和get/post的使用方式。
而返回数据到前台,主要压在model里面,model可以放任何类型的东西,也就是objects。model可以用jsp3.0简单的表达式${}打印,要是里面放的是个list可以配合c标签来打印。
至于各个方法如何处理用户登录、注册、修改密码的细节,相信大家配合上面的test.services.UserServices.java一下就看得懂了,一点都不难,非常简单的逻辑判断。
package test.actions;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.ui.ModelMap;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RequestParam;
import test.services.UserServices;
@Controller
public class WebAction {
@Autowired
private UserServices userServices;//自动注入的存在,getter和setter见鬼去吧!
//跳转到本工程主页的action
@RequestMapping(value = "/index")
public String index() {
return "index";
}
//关于用户登录的表单提交
@RequestMapping(value = "/login", method = RequestMethod.POST)
public String login(@RequestParam("username") String username,
@RequestParam("password") String password, ModelMap model) {
if (!userServices.isUserExist(username)) {
model.addAttribute("msg", "用户名不存在!");
} else {
if (password.equals(userServices.getPasswordByUsername(username))) {
model.addAttribute("msg", "登录成功!");
} else {
model.addAttribute("msg", "密码错误!");
}
}
return "index";
}
//关于用户注册的表单提交
@RequestMapping(value = "/register", method = RequestMethod.POST)
public String register(@RequestParam("username") String username,
@RequestParam("password") String password, ModelMap model) {
if (userServices.isUserExist(username)) {
model.addAttribute("msg", "用户名已存在!");
} else {
userServices.addUser(username, password);
model.addAttribute("msg", "注册成功!");
}
return "index";
}
//关于修改密码的表单提交
@RequestMapping(value = "/modifyPassword", method = RequestMethod.POST)
public String modifyPassword(@RequestParam("username") String username,
@RequestParam("password") String password,
@RequestParam("newpassword") String newpassword, ModelMap model) {
if (!userServices.isUserExist(username)) {
model.addAttribute("msg", "用户名不存在!");
} else {
if (password.equals(userServices.getPasswordByUsername(username))) {
userServices.modifyPasswordByUsername(username, newpassword);
model.addAttribute("msg", "修改密码成功!");
} else {
model.addAttribute("msg", "密码错误!");
}
}
return "index";
}
}
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
主页
${msg}
用户登录
用户注册
修改密码
大家看到这样一套下来,Spring+SpringMVC+Mybatis比Struts2+Hibernate3+Spring3更接地气,没有什么s标签,没有什么又要多开一个xml文件配action,也没有hibernate那套什么hql,完完全全就是你最常用的html语言和sql查询。所以SSM逐渐流行起来了,其实在2014年甚至更早的时候,大家都已经对Struts2、Hibernate3颇有微词,而随着Spring的普及,中文资料渐渐增多,它已经不再单纯是一个java~xml交互工具。简单的注解方法,避免以前大量的xml文件编写,SpringMVC的高安全性,更加让SSM成为业内的新宠!