【SpringMVC】学习总结(二)

SpringMVC(第二天)

1.学习目标

在这里插入图片描述

2. 拦截器

2.1.基本概念

​ SpringMVC中的Interceptor拦截器也是相当重要和相当有用的,它的主要作用是拦截用户的请求并进行相应的处理。比如通过它来进行权限验证,或者是来判断用户是否登陆等操作。对于SpringMVC拦截器的定义方式有两种:

​ 实现接口:org.springframework.web.servlet.Handlerlnterceptor

​ 继承适配器:org.springframework…web.servlet.handler.HandlerInterceptorAdapter

2.2.拦截器实现

2.2.1.实现HandlerInterceptor接口
  • 接口实现类
package com.zgz.springmvc.interceptor;
import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

/**
 *
 * 拦截器实现
 *      实现接口:HandlerInterceptor
 */
public class MyInterceptor01 implements HandlerInterceptor {


    /**
     * 在目标方法(Handler)执行前执行
     *      如果返回true,表示执行目标方法
     *      如果返回false,表示阻止目标方法执行
     **/
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {

        System.out.println("MyInterceptor01 -> preHandle方法.,在目标方法(Handler)执行前执行.,");
        return true;
    }

    /**
     * 在目标方法(handLer)执行后,视图生成前执行
     **/
    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
        System.out.println("MyInterceptor01->postHandle方法.·在目标方法(handler)执行后,视图生成前执行..");
    }

    
    /**
     * 在目标方法(handler)执行后,视图生成后执行
     **/
    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        System.out.println("MyInterceptor01->afterCompletion方法,.·在目标方法(handler)执行后,视图生成后执行..");
    }
}

  • 拦截器配置
<!--拦截器设置-->
    <!--方式一:设置拦截器-->
    <mvc:interceptors>
        <!--
        在mvc:interceptors标签中直接定义bean标签
        bean标签的class执行拦截器的类路径,表示拦截所有的请求
        -->
        <bean class="com.zgz.springmvc.interceptor.MyInterceptor01"/>
    </mvc:interceptors>

<!--拦截器配置 方式二(推荐使用)-->
    <mvc:interceptors>
        <!--
            在mvc:interceptors标签下定义子标签mvc:interceptor,设置具体的拦截器
                在mvc:interceptor标签中:
                    设置mvc:mapping的path属性,设置需要被拦截的路径,可以使用通配符"*",可以配置多个。"/**"表示拦截所有请求
                    设置mvc:exclude-mapping的path属性,设置需要放行的资源,可以使用通配符"*",可以配置多个。
					设置bean标签的class属性,设置拦截器的类路径。

        -->
        <mvc:interceptor>
            <!--通过mvc:mapping配置需要拦截的资源。支持通配符。可配置多个。-->
            <mvc:mapping path="/**"/><!--"/**"表示拦截所有的请求。-->
            <!--通过mvc:mapping配置不需要被拦截的资源。支持通配符。可配置多个。-->
            <mvc:exclude-mapping path="/url/*"/><!--"/url/*"表示放行url路径下的请求。-->
            <bean class="com.zgz.springmvc.interceptor.MyInterceptor01"/>
        </mvc:interceptor>
    </mvc:interceptors>
2.2.2.继承HandlerInterceptorAdapter

​ 实际上最终还是Handlerlnterceptor接口实现。

  • 子类实现类
public class MyInterceptor02 extends HandlerInterceptorAdapter {

    /**
     * 在目标方法(Handler)执行前执行
     *      如果返回true,表示执行目标方法
     *      如果返回false,表示阻止目标方法执行
     **/
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {

        System.out.println("MyInterceptor02 -> preHandle方法.,在目标方法(Handler)执行前执行.,");
        return true;
    }
}

  • 拦截器xml配置
<mvc:interceptors>
        <mvc:interceptor>
            <mvc:mapping path="/**"></mvc:mapping>
            <mvc:exclude-mapping path="/hello"></mvc:exclude-mapping>
            <bean class="com.zgz.springmvc.interceptor.MyInterceptor02"></bean>
        </mvc:interceptor>
</mvc:interceptors>

2.3.拦截器应用-非法请求拦截

​ 使用拦截器完成用户是否登录请求验证功能

2.3.1.用户控制器

UserlnfoController定义

@Controller
@RequestMapping("users")
public class UserController {

    /**
     * 用户登录
     */
    @RequestMapping("login")
    public String login(HttpServletRequest request , HttpSession session){
        //设置请求域
        request.setAttribute("msg","用户登录...");
        //如果用户执行了登录状态,则设置用户信息到session作用域中(如果session作用域中的用信息不为空,则表示登录状态:否则为未登录状态)
        User user = new User();
        user.setUserId(1);
        user.setUserName("zhangsan");
        //设置到作用域
        session.setAttribute("user",user);

        return "success";
    }

    /**
     * 用户添加
     */
    @RequestMapping("add")
    public String add(HttpServletRequest request){
        //设置请求域
        request.setAttribute("msg","用户添加..");
        return "success";
    }

    /**
     * 用户修改
     */
    @RequestMapping("update")
    public String update(HttpServletRequest request){
        //设置请求域
        request.setAttribute("msg","用户修改..");
        return "success";
    }

    /**
     * 用户删除
     */
    @RequestMapping("delete")
    public String delete(HttpServletRequest request){
        //设置请求域
        request.setAttribute("msg","用户删除..");
        return "success";
    }
}

拦截器定义

/**
 *
 * 拦截器
 *  非法访问请求拦截器
 *      用户登录(无需登录,不需要拦截)
 *      用户添加(需要登录,需要拦截判断)
 *      用户修改(需要登录,需要拦截判断)
 *      用户刷别除(需要登录,需要拦截判断)
 */
public class LoginAccessInterceptor extends HandlerInterceptorAdapter {

    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {

        //判断用户是否登录
        //通过request对象获取session对象
        HttpSession session = request.getSession();
        //获取session作用域中的用户信息
        User user = (User) session.getAttribute("user");
        //判断user对象是否为空(不为空表示登录状态,否则为未登录状态)
        if(user == null){
            //未登录状态,跳转到登录页面
            response.sendRedirect(request.getContextPath() +"/login.jsp");
            System.out.println(request.getContextPath());
            return false;
        }
        return true;
    }
}

拦截器xml配置

        <mvc:interceptor>
            <mvc:mapping path="/**"></mvc:mapping>
            <mvc:exclude-mapping path="/users/login"></mvc:exclude-mapping>
            <bean class="com.zgz.springmvc.interceptor.LoginAccessInterceptor"></bean>
        </mvc:interceptor>

登录页面

<%--
  Created by IntelliJ IDEA.
  User: ZHANG
  Date: 2023/2/9
  Time: 18:42
  To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
    <h3>用户登录</h3>
    <form action="users/login" method="post">
    <button>登录</button>
</form>
</body>
</html>

3.文件上传

3.1.环境配置

3.1.1.pom.xml文件修改
        <!--添加commons-fileupload依赖-->
        <dependency>
            <groupId>commons-fileupload</groupId>
            <artifactId>commons-fileupload</artifactId>
            <version>1.3.2</version>
        </dependency>
3.1.2.servlet-context.xml修改
    <!--文件上传-->
    <bean id="multipartResolver"
          class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
        <!--允许文件上传的最大尺寸-->
        <property name="maxUploadSize">
            <value>104857600</value>
        </property>
        <!--
            设置文件放入临时文件夹的最大大小限制。
            此值是阈值,低于此值,则保存在内存中,如高于此值,则生成硬盘上的临时文件。
        -->
        <property name="maxInMemorySize">
            <value>4096</value>
        </property>
    </bean>

3.2.代码实现

3.2.1.单文件上传
3.2.1.1.页面表单
  • input的type设置为file
  • form表单的method设为post,
  • form表单的enctype设置为multipart/form-data,以二进制的形式传输数据
    <form action="uploadFile" method="post" enctype="multipart/form-data">
        文件<input name="myfile" type="file">
       <button>提交</button>
    </form>
3.2.1.2. 代码实现
package com.zgz.springmvc.controller;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.multipart.MultipartFile;

import javax.servlet.http.HttpServletRequest;
import java.io.File;
import java.io.IOException;

/**
 * 单文件上传
 * 使用MultipartFile对象作为参数,接收前端发送过来的文件
 */
@Controller
public class FileController {

    /**
     * 文件上传
     *  在形参上使用@RequestParam("myfile")MultipartFile file接收上传的文件
     *      @RequestParam("myfile“)中的"myfiLe"代表的是表单元素的name属性值
     **/
    @RequestMapping("uploadFile")
    public String uploadFile(@RequestParam("myfile") MultipartFile file, HttpServletRequest request) {
        //判断文件是否为空,如果不为空进行对应的文件上传操作
        if (!file.isEmpty()) {
            try {
                //获取项目的所在的路径(绝对路径)
                String path = request.getServletContext().getRealPath("/");
                System.out.println(path);
                //设置上传的文件存放的目录
                File uploadFile = new File(path + "/upload");
                //判断文件目录是否存在,不存在则新建目录
                if (!uploadFile.exists())
                    //新建目录
                    uploadFile.mkdir();
                //获取上传文件的原文件名
                String originalName = file.getOriginalFilename();
                //获取上传的文件的后缀
                String suffix = originalName.substring(originalName.lastIndexOf("."));
                //通过系统当前时间的毫秒数,生成随机文件名(避免上传的文件名重复)
                String fileName = System.currentTimeMillis() + suffix;
                //上传文件(转存文件到指定目录)
                file.transferTo(new File(uploadFile, fileName));

                //设置成功的域对象
                request.setAttribute("msg", "文件上传成功!");
            } catch (IOException e) {
                e.printStackTrace();
                //如果报错,设置的域对象
                request.setAttribute("msg", "文件上传失败!");
            }
        } else
            //上传文件不存在,设置的域对象
            request.setAttribute("msg", "文件不存在,上传失败!");

        return "success";
    }
}

4.SSM框架集成与测试

4.1.环境配置

4.1.1.IDEA下创建项目

创建Maven对应的Web项目

4.1.2.配置pom.xml
4.1.2.1.修改JDK版本
    <properties>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <maven.compiler.source>1.8</maven.compiler.source>
        <maven.compiler.target>1.8</maven.compiler.target>
    </properties>
4.1.2.2.添加坐标依赖
    <dependencies>
        <!-- junit测试 -->
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>4.12</version>
            <scope>test</scope>
        </dependency>

        <!--spring核心jar-->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context</artifactId>
            <version>5.2.4.RELEASE</version>
        </dependency>

        <!--spring测试jar-->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-test</artifactId>
            <version>5.2.4.RELEASE</version>
        </dependency>

        <!--spring jdbc-->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-jdbc</artifactId>
            <version>5.2.4.RELEASE</version>
        </dependency>

        <!--spring事务-->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-tx</artifactId>
            <version>5.2.4.RELEASE</version>
        </dependency>

        <!--aspectj切面编程的jar-->
        <dependency>
            <groupId>org.aspectj</groupId>
            <artifactId>aspectjweaver</artifactId>
            <version>1.9.5</version>
        </dependency>

        <!--c3p0连接池-->
        <dependency>
            <groupId>com.mchange</groupId>
            <artifactId>c3p0</artifactId>
            <version>0.9.5.2</version>
        </dependency>

        <!--mybatis -->
        <dependency>
            <groupId>org.mybatis</groupId>
            <artifactId>mybatis</artifactId>
            <version>3.5.3</version>
        </dependency>

        <!--添加nybatis.与Spring整合的核心包-->
        <dependency>
            <groupId>org.mybatis</groupId>
            <artifactId>mybatis-spring</artifactId>
            <version>2.0.3</version>
        </dependency>

        <!--mysql驱动包-->
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>8.0.19</version>
        </dependency>

        <!--日志打印相关的jar-->
        <dependency>
            <groupId>org.slf4j</groupId>
            <artifactId>slf4j-log4j12</artifactId>
            <version>1.7.2</version>
        </dependency>
        <dependency>
            <groupId>org.slf4j</groupId>
            <artifactId>slf4j-api</artifactId>
            <version>1.7.2</version>
        </dependency>

        <!--分页插件-->
        <dependency>
            <groupId>com.github.pagehelper</groupId>
            <artifactId>pagehelper</artifactId>
            <version>5.1.10</version>
        </dependency>

        <!--spring web -->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-web</artifactId>
            <version>5.2.4.RELEASE</version>
        </dependency>

        <!--spring mvc-->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-webmvc</artifactId>
            <version>5.2.4.RELEASE</version>
        </dependency>

        <!--web servlet -->
        <dependency>
            <groupId>javax.servlet</groupId>
            <artifactId>javax.servlet-api</artifactId>
            <version>3.0.1</version>
            <scope>provided</scope>
        </dependency>

        <!--添加json依赖jar包(注意版本问题)-->
        <dependency>
            <groupId>com.fasterxml.jackson.core</groupId>
            <artifactId>jackson-core</artifactId>
            <version>2.10.0</version>
        </dependency>
        <dependency>
            <groupId>com.fasterxml.jackson.core</groupId>
            <artifactId>jackson-databind</artifactId>
            <version>2.10.0</version>
        </dependency>
        <dependency>
            <groupId>com.fasterxml.jackson.core</groupId>
            <artifactId>jackson-annotations</artifactId>
            <version>2.10.0</version>
        </dependency>
        <!--commons-fileupload -->
        <dependency>
            <groupId>commons-fileupload</groupId>
            <artifactId>commons-fileupload</artifactId>
            <version>1.3.2</version>
        </dependency>
    </dependencies>
4.1.2.3.设置资源目录和插件
    <build>
        <finalName>ssm</finalName>
        <!--
            Maven项目:如果源代码(src/main/java)存在xml、properties、tld等文件
            Maven默认不会自动编译该文件到输出目录,如果要编译源代码中xml properties tld等文件
            需要显式配置resources标签
        -->
        <resources>
            <resource>
                <directory>src/main/resources</directory>
            </resource>
            <resource>
                <directory>src/main/java</directory>
                <includes>
                    <include>**/*.xml</include>
                    <include>**/*.properties</include>
                    <include>**/*.tld</include>
                </includes>
                <filtering>false</filtering>
            </resource>
        </resources>
    </build>
4.1.3.配置web.xml
<?xml version="1.0" encoding="UTF-8"?>
<web-app id="WebApp_ID" version="3.0"
         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_3_0.xsd">
    <!--启动spring容器-->
    <context-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>classpath:spring.xml</param-value>
    </context-param>
    <!--设置监听器-->
    <listener>
        <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
    </listener>

    <!--编码过滤utf-8-->
    <filter>
        <description>char encoding filter</description>
        <filter-name>encodingFilter</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>
    </filter>
    <filter-mapping>
        <filter-name>encodingFilter</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping>
    <!--serv1et请求分发器-->
    <servlet>
        <servlet-name>springMvc</servlet-name>
        <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
        <!--初始化参数,在servlet类加载时加载的参数-->
        <init-param>
            <param-name>contextConfigLocation</param-name>
            <param-value>classpath:servlet-context.xml</param-value>
        </init-param>
        <!--表示启动容器时初始化该Servlet-->
        <load-on-startup>1</load-on-startup>

    </servlet>
    <servlet-mapping>
        <servlet-name>springMvc</servlet-name>
        <!--这是拦截请求,"/"代表拦截所有请求,"*.do"拦截所有.do请求-->
        <url-pattern>/</url-pattern>
        <!--        <url-pattern>*.do</url-pattern>-->
    </servlet-mapping>
</web-app>
4.1.4配置servlet-context.xml
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:mvc="http://www.springframework.org/schema/mvc"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="
        http://www.springframework.org/schema/mvc
        http://www.springframework.org/schema/mvc/spring-mvc.xsd
        http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context
        http://www.springframework.org/schema/context/spring-context.xsd">
    
    <!--开启扫描器 -->
    <context:component-scan base-package="com.zgz.springmvc.controller"/>
    
    <!--使用默认的Servlet来响应静态文件-->
    <mvc:default-servlet-handler/>

    <!--开启注解驱动-->
    <mvc:annotation-driven/>

    <!-- mvc注解驱动井添加json支持-->
    <mvc:annotation-driven>
        <mvc:message-converters>
            <!-- 返回信息为字符串时处理 -->
            <bean class="org.springframework.http.converter.StringHttpMessageConverter"/>
            <!-- 将对象转换为json对象 -->
            <bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter"/>
        </mvc:message-converters>
    </mvc:annotation-driven>

    <!--配置视图解析器-->
    <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver"
          id="internalResourceviewResolver">
        <!--前缀:在WEB-INF目录下的jsp目录下-->
        <property name="prefix" value="/WEB-INF/jsp/"/>
        <!--后缀:以.jsp结尾的资源-->
        <property name="suffix" value=".jsp"/>
    </bean>

    <!--文件上传-->
    <bean id="multipartResolver"
          class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
        <!--允许文件上传的最大尺寸-->
        <property name="maxUploadSize">
            <value>104857600</value>
        </property>
        <!--
            设置文件放入临时文件夹的最大大小限制。
            此值是阈值,低于此值,则保存在内存中,如高于此值,则生成硬盘上的临时文件。
        -->
        <property name="maxInMemorySize">
            <value>4096</value>
        </property>
    </bean>

</beans>
4.1.5.配置spring.xml

​ 在项目的src/main/resources下创建spring.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"
       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.xsd
       http://www.springframework.org/schema/context
       http://www.springframework.org/schema/context/spring-context.xsd
       http://www.springframework.org/schema/aop
       http://www.springframework.org/schema/aop/spring-aop.xsd
       http://www.springframework.org/schema/tx
       http://www.springframework.org/schema/tx/spring-tx.xsd">


    <!--扫描基本包-->
    <context:component-scan base-package="com.zgz.ssm">
        <!--context:exclude-filter标签:排除对某个注解的扫描 (过滤Controller层)-->
        <context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
    </context:component-scan>

    <!--加载properties配置文件-->
    <context:property-placeholder location="classpath:db.properties"/>
    <!--aop-->
    <aop:aspectj-autoproxy/>
    <!--配置c3p0数据源-->
    <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
        <property name="driverClass" value="${jdbc.driver}"></property>
        <property name="jdbcUrl" value="${jdbc.url}"></property>
        <property name="user" value="${jdbc.username}"></property>
        <property name="password" value="${jdbc.password}"></property>
    </bean>

    <!--配置事务管理器-->
    <bean id="txManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <property name="dataSource" ref="dataSource"></property>
    </bean>

    <!--设置事务增强-->
    <tx:advice id="txAdvice" transaction-manager="txManager">
        <tx:attributes>
            <tx:method name="add*" propagation="REQUIRED"/>
            <tx:method name="insert*" propagation="REQUIRED"/>
            <tx:method name="update*" propagation="REQUIRED"/>
            <tx:method name="delete*" propagation="REQUIRED"/>
        </tx:attributes>
    </tx:advice>

    <!--aop切面配置-->
    <aop:config>
        <aop:pointcut id="servicePointcut" expression="execution(* com.zgz.ssm.service..*.*(..))"/>
        <aop:advisor advice-ref="txAdvice" pointcut-ref="servicePointcut"/>
    </aop:config>

    <!--配置sqlSessionFactory-->
    <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
        <property name="dataSource" ref="dataSource"></property>
        <property name="configLocation" value="classpath:mybatis.xml"/>
        <property name="mapperLocations" value="classpath:com/zgz/ssm/mapper/*.xml"/>
    </bean>

    <!--配置扫描器-->
    <bean id="mapperScanner" class="org.mybatis.spring.mapper.MapperScannerConfigurer">
        <!--扫描com.zgz.ssm.dao这个包以及它的子包下的所有映射接口类-->
        <property name="basePackage" value="com.zgz.ssm.dao"/>
        <property name="sqlSessionFactoryBeanName" value="sqlSessionFactory"/>
    </bean>
</beans>
4.1.6.配置mybatis.xml

在项目的src/main/resources下创建mybatis.xml文件,内容如下

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE configuration
        PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-config.dtd">

<configuration>
    <typeAliases>
        <package name="com.zgz.ssm.po"/>
    </typeAliases>
    <plugins>
        <plugin interceptor="com.github.pagehelper.PageInterceptor"></plugin>
    </plugins>
</configuration>
4.1.7.配置db.properties

在项目的src/main/resources下创建db.properties文件,内容如下(mysql创建数据库ssm)

jdbc.driver=com.mysql.cj.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/ssm?useUnicode=true&characterEncoding=utf8&serverTimezone=GMT%2B8&useSSL=false
jdbc.username=root
jdbc.password=zhangguzong
4.1.8.添加log4j.properties

在项目的src/main/resources下创建log4j.properties文件,内容如下

1og4j.rootLogger=DEBUG,Console
Console
1og4j.appender.Console=org.apache.10g4j.ConsoleAppender
log4j.appender.Console.layout=org.apache.l0g4j.PatternLayout
1og4j.appender.Console.layout.ConversionPattern=%d [%t]%-5p [%c]-%m%n
log4j.logger.java.sql.ResultSet=INFO
1og4j.logger.org.apache=INFO
1og4j.logger.java.sql.Connection=DEBUG
log4j.logger.java.sql.Statement=DEBUG
log4j.logger.java.sq1.PreparedStatement=DEBUG

4.2.添加源代码

4.2.1.添加包

​ 在项目的src/main/java下创建对应的包结构
com.xxxx.ssm.controller
com.xXxx.ssm.service
com.xxxx.ssm.mapper
com.XxXX.ssm.dao
com.xxxx.ssm.po

4.2.2.添加User,java

在com.xxxx.ssm.po包下创建JavaBean文件User.java(数据库字段对应如下)

在这里插入图片描述

public class User {
    private Integer userId;
    private String userName;
    private String userPwd;
    private String userEmail;
    private Date createDate;
    private Date updateDate;
}
4.2.3.添加UserDao.java接口

com.xxxx.ssm.dao包下创建UserDao.java文件,提供对应的用户详情查询功能

public interface UserDao {

    //通过用户ID查询用户对象
    public User queryUserById(Integer userId);
}
4.2.4.添加UserMapper.xml映射文件

com.xxxx.ssm.mapper包下创建UserMapper.xml文件,提供select查询标签配置

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper
        PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
    <mapper namespace="com.xxxx.ssm.dao.UserDao">
        <select id="queryUserByUserId" parameterType="int" resultType="com.zgz.ssm.po.User">
            select user_id as userId,user_name as userName,user_pwd as userPwd
            from tb_user
            where user_id = #{userId}
    </select>
</mapper>
4.2.5.添加UserService.java

com.xxxx.ssm.service包下创建UserService.java文件,提供用户详情查询方法

@Service
public class UserService {

    @Autowired
    private UserDao userDao ;
    /**
     * 通过用户ID查询用户对象
     **/
    public User queryUser(Integer userId){
        User user = userDao.queryUserById(userId);
        return user;
    }
}
4.2.6.添加UserController.java
@Controller
@RequestMapping("user")
public class UserController {

    @Autowired
    private UserService userService;

    @RequestMapping("query")
    @ResponseBody
    public User queryUser(Integer userId) {
        return userService.queryUser(userId);
    }
}

4.3.执行测试

4.3.1.Idea下配置tomcat启动命令

在这里插入图片描述

4.3.2.启动jetty浏览器访问http://localhost:8080/ssm/user/query?userId=1查看结果

6.全局异常统一处理

6.1.全局异常概念

​ 在JavaEE项目的开发中,不管是对底层的数据库操作过程,还是业务层的处理过程,还是控制层的处理过程,都不可避免会遇到各种可预知的、不可预知的异常需要处理。每个过程都单独处理异常,系统的代码耦合度高,工作量大且不好统一,维护的工作量也很大。

​ SpringMVC对于异常处理这块提供了支持,通过SpringMVC提供的全局异常处理机制,能够将所有类型的异常处理从各处理过程解耦出来,既保证了相关处理过程的功能较单一,也实现了异常信息的统一处理和维护。

全局异常实现方式Spring MVC处理异常有3种方式:

  1. 使用Spring MVC提供的简单异常处理器SimpleMappingException Resolver
  2. 实现Spring的异常处理接口HandlerExceptionResolver自定义自己的异常处理器
  3. 使用@ExceptionHandler注解实现异常处理

6.2.异常处理实现

6.2.1.全局异常处理方式一
6.2.1.1.配置简单异常处理器

配置SimpleMappingExceptionResolver对象

配置SimpleMappingExceptionResolver对象
<!--配置全局异常统一处理的Bean(简单异常处理器)-->
<bean class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver">
<!--页面在转发时出现异常,设置默认的错误页面(error代表的是一个视图)-->
<property name="defaultErrorView"value="error"></property>
<!--异常发生时,设置异常的变量名-->
<property name="exceptionAttribute"value="ex"></property>
</bean>

可以在处理异常的页面获取异常信息

${ex}
6.2.1.2.使用自定义异常

参数异常

package com.zgz.ssm.exceptions;

/**
 * 自定义异常:参数异常
 */
public class ParamsException extends RuntimeException {
    private Integer code = 300;
    private String msg = "参数异常!";

    public ParamsException() {
        super("参数异常!");
    }

    public ParamsException(String msg) {
        super(msg);
        this.msg =msg;
    }

    public ParamsException(Integer code) {
        super("参数异常!");
        this.code =code;
    }

    public ParamsException(Integer code, String msg) {
        super(msg);
        this.code = code;
        this.msg = msg;
    }

    public Integer getCode() {
        return code;
    }

    public void setCode(Integer code) {
        this.code = code;
    }

    public String getMsg() {
        return msg;
    }

    public void setMsg(String msg) {
        this.msg = msg;
    }
}

业务异常

package com.zgz.ssm.exceptions;

/**
 * 业务异常
 */
public class BusinessException extends RuntimeException {

    private Integer code = 400;
    private String msg = "业务异常!";

    public BusinessException() {
        super("业务异常!");
    }

    public BusinessException(String msg) {
        super(msg);
        this.msg = msg;
    }

    public BusinessException(Integer code) {
        super("业务异常!");
        this.code = code;
    }

    public BusinessException(Integer code, String msg) {
        super(msg);
        this.code = code;
        this.msg = msg;
    }

    public Integer getCode() {
        return code;
    }

    public void setCode(Integer code) {
        this.code = code;
    }

    public String getMsg() {
        return msg;
    }

    public void setMsg(String msg) {
        this.msg = msg;
    }


}

6.2.1.3.设置自定义异常与页面的映射
    <!--配置全局异常统一处理的Bean(简单异常处理器)-->
    <bean class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver">
        <!--页面在转发时出现异常,设置默认的错误页面(error代表的是一个视图)-->
        <property name="defaultErrorView" value="error"></property>
        <!--异常发生时,设置异常的变量名-->
        <property name="exceptionAttribute" value="ex"></property>
        <!--设置自定义异常与页面的映射-->
        <property name="exceptionMappings">
            <props>
                <!--key:自定义异常对象的路径:标签中设置具体的处理页面的视图名-->
                <prop key="com.zgz.ssm.exceptions.BusinessException">buss_error</prop>
                <prop key="com.zgz.ssm.exceptions.ParamsException">params_error</prop>
            </props>
        </property>
    </bean>

​ 使用SimpleMappingExceptionResolver进行异常处理,具有集成简单、有良好的扩展性、对已有代码没有入侵性等优点,但该方法仅能获取到异常信息,若在出现异常时,对需要获取除异常以外的数据的情况不适用。

6.2.2.全局异常处理方式二(推荐)
6.2.2.1.实现HandlerExceptionResolver接口
@Component
public class GlobalExceptionResolver  implements HandlerExceptionResolver {
    @Override
    public ModelAndView resolveException(HttpServletRequest httpServletRequest,
                                         HttpServletResponse httpServletResponse,
                                         Object o, Exception e) {
        //得到ModelAndView对象,并设置默认视图
        ModelAndView modelAndView =new ModelAndView("error");
        //设置默认的异常信息
        modelAndView.addObject("ex","默认的异常信息,..");
        return modelAndView ;
    }
}        
6.2.2.2 自定义异常处理
/**
 * 全局异常统—处理
 *  方式二(推荐使用)
 *      实现Spring的异常处理接口HandlerExceptionResolver
 *      自定义自己的异常处理器具有集成简单、有良好的扩展性、对已有代码没有入侵性等优,点同时,
 *      在异常处理时能获取导致出现异常的对象,有利于提供更详细的异常处理信息。
 */
@Component
public class GlobalExceptionResolver  implements HandlerExceptionResolver {
    @Override
    public ModelAndView resolveException(HttpServletRequest httpServletRequest,
                                         HttpServletResponse httpServletResponse,
                                         Object o, Exception e) {
        //得到ModelAndView对象,并设置默认视图
        ModelAndView modelAndView =new ModelAndView("error");
        //设置默认的异常信息
        modelAndView.addObject("ex","默认的异常信息,..");

        if (e instanceof ParamsException){
            ParamsException p =(ParamsException)e;
            p.setCode(500);
            p.setMsg("这是个参数异常...");
            modelAndView.setViewName("params_error");
            modelAndView.addObject("e",p);
            return modelAndView ;
        }

        if (e instanceof BusinessException){
            BusinessException p =(BusinessException)e;
            p.setCode(500);
            p.setMsg("这是个业务异常...");
            modelAndView.setViewName("buss_error");
            modelAndView.addObject("e",p);
            return modelAndView ;
        }


        return modelAndView ;
    }
}

​ 使用实现HandlerExceptionResolver接口的异常处理器进行异常处理,具有集成简单、有良好的扩展性、对已有代码没有入侵性等优点,同时,在异常处理时能获取导致出现异常的对象,有利于提供更详细的异常处理信息。

6.2.3.全局异常处理方式三

页面处理器继承BaseController

/**
 *全局异常处理方式三
 */
public class BaseController {

    /**
     * 处理异常的方法
     **/
    @ExceptionHandler
    public String exc(HttpServletRequest request, HttpServletResponse response, Exception ex) {
        request.setAttribute("ex", ex);
        if (ex instanceof ParamsException) {
            return "error_param";
        }
        if (ex instanceof BusinessException) {
            return "error business";
        }

        return "error";
    }
}

​ 使用@ExceptionHandler注解实现异常处理,具有集成简单、有扩展性好(只需要将要异常处理的Controller类继承于BaseController即可)、不需要附加Spring配置等优点,但该方法对已有代码存在入侵性(需要修改已有代码,使相关类继承于BaseController),在异常处理时不能获取除异常以外的数据。

全局异常统一处理方式一:

  1. 使用Spring MVC提供的简单异常处理器SimpleMappingExceptionResolver使用简单、集成简单、扩展性好,对代码没有入侵性
    但仅能获取到异常信息,若在出现异常时,对需要获取除异常以外的数据的情况不适用

全局异常统一处理方式二(推荐使用)

  1. 实现Spring的异常处理接口HandlerExceptionResolver自定义自己的异常处理器具有集成简单、有良好的扩展性、对已有代码没有入侵性等优,点同时,在异常处理时能获取导致出现异常的对象,有利于提供更详细的异常处理信息。

全局异常统一处理方式三:

​ 1. 使用@ExceptionHandler注解实现异常处理,具有集成简单、有扩展性好(只需要将要异常处理的Controller类继承于BaseController即可)、不需要附加Spring配置等优点,但该方法对已有代码存在入侵性(需要修改已有代码,使相关类继承于BaseController),在异常处理时不能获取除异常以外的数据。

6.3.未捕获异常的处理

​ 对于Unchecked Exception而言,由于代码不强制捕获,往往被忽略,如果运行期产生了Unchecked Exception,而代码中又没有进行相应的捕获和处理,则我们可能不得不面对尴尬的404、500…等服务器内部错误提示页面。此时需要一个全面而有效的异常处理机制。目前大多数服务器也都支持在wb.xml中通过(Vebsphere/Weblogic)或者(Tomcat)节点配置特定异常情况的显示页面。修改web.xml文件,增加以下内容:

    <!--
        未捕获异常处理
        在web.xml中通过<error-page>(Websphere/Weblogic)或者<error-code>(Tomcat)节点配置特定异常情况的显示页面
        -->
    <error-page>
        <exception-type>java.lang.Throwable</exception-type>
        <location>/500.jsp</location>
    </error-page>
    <error-page>
        <error-code>404</error-code>
        <location>/404.jsp</location>
    </error-page>
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值