day08 Spring MVC

spring MVC相当于Servlet

mvc解释:模型,视图,控制器

**使用该思想的作用:**减少耦合性,提高可维护性

Spring MVC前端控制器

方式1

1.在web.xml中配置前端控制器

方式2

要是用前端控制器,必须在web.xml中配置DidpatcherServlet类

    <!--前端控制器-->
<servlet>
    <servlet-name>dispatcherServlet</servlet-name>
    <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
   <!--指定加载的文件,不加的话默认在webapp里面找-->
    <init-param>
        <param-name>contextConfigLocation</param-name>
        <!--里面的文件就是spring的xml配置文件-->
        <param-value>classpath:applicationContext.xml</param-value>
    </init-param>
    <!--启动时tomcat初始化-->
    <load-on-startup>1</load-on-startup>
</servlet>
    <servlet-mapping>
        <servlet-name>dispatcherServlet</servlet-name>
        <url-pattern>/</url-pattern>
    </servlet-mapping>

spring中的xml文件的基本配置

<!--ioc注解解析器-->
    <context:component-scan base-package="cn.wolfcode"/>
    <!--mvc注解解析器-->
    <mvc:annotation-driven/>

Java文件中的处理器书写,使用Controller和RequestMapping注解

@Controller
 @RequestMapping("/HelloController")
public class HelloController {
    // 提供方法处理请求,在浏览器地址栏输入如下 localhost/hello,就会执行下面的方法
    @RequestMapping("/hello")
    public ModelAndView save() {
        ModelAndView mv = new ModelAndView();
        // 往作用域或者模型中存入数据
        mv.addObject("msg", "Hello Spring MVC");
        // 找视图
        mv.setViewName("WEB-INF/views/hello.jsp");
        return mv;

注意事项:

@RequestMapping(“/hello”)可以加在方法上,也可以加在类上面

加在类上面的作用:与方法里面进行联用来区分同名的不同的方法

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-y3tbPgBP-1684998497797)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\image-20230512155009317.png)]

mvc配置联用
<!--ioc注解解析器-->
<context:component-scan base-package="cn.wolfcode"/>
<!--mvc注解解析器-->
<mvc:annotation-driven/>
<!--使用tomcat内置的拦截器访问静态资源-->
<mvc:default-servlet-handler/>

**注意:**上述配置会在 Spring MVC 上下文中创建存入一个 DefaultServletHttpRequestHandler 的 bean,它会对进入 DispatcherServlet 的请求进行筛查,若不是映射的请求,就将该请求交由容器默认的 Servlet 处理。

web.xml拦截规则

*. :要求处理器的地址以.do,静态资源不会进入到前端控制器中,所以可以访问静态资源

/ :对处理地址没有要求,静态资源会被前端控制器进行拦截,里面找不到对应的处理器.解决办法:在spring配置文件中配置<mvc:default-servlet-handler/>,使用tomcat内置的处理器,jsp不会被拦截

/* :对处理地址没有要求,会拦截所有资源(html/jsp),要访问静态资源,配置**mvc:default-servlet-hander**之后可以访问,jsp可以访问但是不会渲染

处理响应的两种方式

1.返回MoudelAndView

@Controller
@RequestMapping("/HelloConctroller")
public class HelloConctroller {
    /**
     * 使用ModelAndView处理响应
     * @return
     */
    @RequestMapping("/hello")
    public ModelAndView save() {
        ModelAndView mv = new ModelAndView();
        //向作用域中或者模型中存入数据,并进行共享
        mv.addObject("msg", "hello");
        //寻找视图
        mv.setViewName("WEB-INF/hello.jsp");
        return mv;
    }

2.返回String

注意:传了个model接口作为参数进行使用

@Controller
public class StringController {
    /*
     * 使用model处理响应
     * */
    @RequestMapping("/getrq1")
    public String rq1(Model model) {
        model.addAttribute("msg", "123456");
        /*   return "WEB-INF/hello.jsp";*/

        //使用配置文件消除视图的前缀和后缀用于代替上方的放回的String类型的地址
        //默认是转发
        return "hello";
    }
消除视图前缀和后缀

作用:消除视图路径重复.在spring.xml中进行配置,只需要在java文件中写文件名

<!--配置视图解析器,消除前缀以及后缀-->
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
    <!--视图前缀-->
    <property name="prefix" value="/WEB-INF/"/>
    <!--视图后缀-->
    <property name="suffix" value=".jsp"/>
</bean>
@Controller
public class StringController {
    /*
     * 使用model处理响应
     * */
    @RequestMapping("/getrq1")
    public String rq1(Model model) {
        model.addAttribute("msg", "123456");
        
        //使用配置文件消除视图的前缀和后缀用于代替上方的放回的String类型的地址
        //默认是转发
        return "hello";
    }
请求转发和重定向

1.请求转发

在ModelAndView/Model的返回值里面添加forward前缀,转发后地址不变,可以进行数据共享

@Controller
public class ResponseConctroller {
    @RequestMapping("/responseController")
    public String responseController(Model model){
        model.addAttribute("msg","请求转发前缀forward");
        /*使用请求转发的forward前缀,进行页面的跳转和数据的共享*/
       return "forward:WEB-INF/hello.jsp";
    }

2.重定向

在ModelAndView/Model的返回值里面添加redirect前缀,转发后地址改变,不能进行数据共享

@Controller
public class ResponseConctroller {
    @RequestMapping("/responseController")
    public String responseController(Model model){
        model.addAttribute("msg","请求转发前缀forward");

        /*使用重定向的redirect前缀,进行页面的跳转,但是不会对数据进行共享,只能访问WEB-INF外的数据*/
        return "redirect:index.jsp";
    }

注意:返回值添加了forward和redirect前缀之后不能和消除视图进行联用

3.路径问题

在请求转发或者重定向的时候,路径:

​ 加/: 使用绝对路径进行寻找,从项目的根目录找

​ 不加/: 使用相对路径寻找,在该路径的同级目录寻找

该模块的总结

1.spring默认使用转发的方式相当于在跳转的视图前面加上 forward

2.spring重定向是在条状的页面之前添加redirect

3.强调:重定向不能直接访问到WEB-INF下的资源

4.使用model()其实就是向request作用域中设定数据, /modeAndView.addObjiect() 也是一样

5.如果使用重定向后,一定要从作用域中取值,可以使用req.getSession().setAttr(),和之前所学的servlet中作用域的知识联用

6.在转发或者冲定向的时候跳转的页面加上 / 和不加 / 的区别

​ 加 / :从根目录开始寻找----绝对路径

​ 不加 /: 从同级目录开始寻找--------相对路径

处理简单类型请求参数

注意:

​ ModelAndView返回值为null的时候,返回的是空白页面

​ String返回值为null的时候,就会报空指针异常

​ 底层会帮自动进行类型的转化

​ 接收的参数类型一定要保持一定

1.请求名和形参名相同

​ 可以接收到传进来的值

@Controller
public class ParaTypeController {
    /**
     * 接收一个账号和一个密码
     */
    //http://localhost:8080/paraTypeController?name=%E7%81%B0%E7%81%B0&password=123456
    
    @RequestMapping("/paraTypeController")
    public ModelAndView paraTypeController(String name,Integer password){
        System.out.println(name);
        System.out.println(password);
        return null;
    }
2.请求名和形参名不同

​ 会造成接收的参数值为null

​ 使用**@RequestParam(“请求名”)注**解保证接收参数名和形参名相同

/**
 * 接收一个账号和一个密码,请求参数和形参不同
 */
//http://localhost:8080/paraTypeController2?username=灰灰&password=123456

@RequestMapping("/paraTypeController2")
//使用@RequestParam解决参数名不一致问题
public ModelAndView paraTypeController2(@RequestParam("username")String name, Integer password){
    System.out.println(name);
    System.out.println(password);
    return null;
}
3.出现了中文乱码

浏览器发送中文的参数值,服务端接收到后出现乱码

3.1:在tomcat7以下的版本使用GET方式传递中文参数值会出现该问题,在pom.xml中的tomcat插件添加编码设置

​		[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-uiFw3XDj-1684998497800)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\image-20230513093714911.png)]

​ **3.2:**post方式出现乱码,和tomcat版本没有关系,解决办法,web.xml进行配置

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-K2fGO46a-1684998497800)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\image-20230513100311238.png)]

​ 使用表单的方式进行post请求的提交

<%--解决中文乱码--%>
<form method="post" action="/paraTypeController2">
  <input type="text" name="username"/>
  <input type="password" name="password"/>
  <input type="submit">
</form>

复合类型请求参数

1.接收数组类型参数

​ 接收数组类型的参数,用于每个参数的名字相同的情况(复选框),若里面的参数不能进行匹配,值就是null

	/**
     * 接收多个同名的参数使用Long类型的数组进行接收
     * 复合类型参数---数组类型
     */
    //http://localhost:8080/idsController?ids=1&ids=2&ids=3

    @RequestMapping("/idsController")
    public ModelAndView idsController(Long[] ids){
        System.out.println(ids);
        return null;
    }
2.接收自定义类型参数

​ 接收的参数名和自定义类型的属性一 一对应,使用包装类型,若传进来的类型和属性的类型不匹配就找不到,报400

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-63Xj3Jjp-1684998497801)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\image-20230513103022084.png)]

    /**
     * 自定义类型参数的接收
     * 使用自定义方式接收和请求的方式没有关系
     */
    //http://localhost:8080/userController?username=huihui&password=123456
    @RequestMapping("/userController")
    public ModelAndView userController(User user){
        System.out.println(user);
        return null;
    }

打印结果如下:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-u14q5jnr-1684998497802)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\image-20230513103145138.png)]

3.处理日期格式参数

​ 接收的参数需要指定格式,底层不会自动将日期类型的参数进行转换,使用**@DateTimeFormat(pattern = “yyyy-MM-dd”)注解**,这个注解可以贴在自定义类型的属性上面

// 增加下面这个字段,并贴注解
@DateTimeFormat(pattern="yyyy-MM-dd")
private Date date;

注意:传进来的日期不能超出范围,否则报400错误

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-uvXDLHlC-1684998497802)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\image-20230513104209227.png)]

/**
 * 处理日期类型的参数,将传进来的日期进行转换,需要指定格式
 */
//http://localhost:8080/dataController?date=2012-25-3
@RequestMapping("/dataController")
public ModelAndView dataController(@DateTimeFormat(pattern = "yyyy-MM-dd")Date date){
    System.out.println(date);
    return null;
}

使用ModeAttribute注解

​ **作用:**用于回显数据

​ 使用ModeAttribute注解是把数据存到request作用域,代替了手动设置参数值作用域中的

​ 使用String的返回值类型更加简洁

/**
 * 用于回显数据,使用的返回类型使用String
 * 使用User自定义参数进行接收---接受的参数展示在页面上
 * 使用ModelAttribute注解将数据存放到request作用域,同时可以修改在共享作用域的名字
 * 默认使用的请求转发作用域,无法使用重定向共享数据
 *
 */
//http://localhost:8080/AttributeController?username=huihui&password=123456
@RequestMapping("/AttributeController")
public String AttributeController(@ModelAttribute("u") User user){
    System.out.println(user);
    return "hello";
}

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-hWRlUavM-1684998497803)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\image-20230513112512054.png)]

文件上传

注意及步骤:

​ 1.使用的post请求:有文件大小限制

​ 2.指定文件上传的大小,在web.xml中的中进行指定

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-IkiTJvan-1684998497803)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\image-20230513113903039.png)]

​ 3.必须在applicationContext.xml文件中配置StandardServletMultipartResolver解析器,同时解析器必须加上id=“multipartResolver”,否则报400错误

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-BSNhyC08-1684998497804)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\image-20230513114731871.png)]

​ 4.使用的part类型进行接收文件,该Part是javax中的类

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-BhZGrgpK-1684998497804)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\image-20230513201124067.png)]

​ 5.若文件没有进行上传,会报400错误

拦截器

步骤:

1.编写拦截器,实现的是HandlerInterceptor接口,并重写preHandle方法

​ 若返回的的true,会进行拦截,之后会放行这个方法

​ 若返回的false,会拦截这个方法,并不会放行(不会执行这个方法)

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-IrbaucUs-1684998497805)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\image-20230513201811304.png)]

2.在applicationContext.xml中配置拦截器

/* 和/**的区别

​ /* 只会拦截/a /b 这种路径的方法,不会拦截/a/b这种方法

​ /** 会拦截根目录中所有的方法

<mvc:interceptors>
	<!-- 配置拦截器 -->
	<mvc:interceptor>
		<!-- 拦截的路径 -->
		<mvc:mapping path="/**"/>
		<!-- 被排除的路径
		<mvc:exclude-mapping path="排除的路径"/> -->
		<!-- 拦截器类 -->
		<bean class="cn.wolfcode.web.interceptor.MyInterceptor"/>
	</mvc:interceptor>
</mvc:interceptors>

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-bunMskFT-1684998497805)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\image-20230513202224609.png)]

拦截器和过滤器的区别

1.过滤器是在web.xml中进行配置, 拦截器是在applicationContext.xml中配置mvc

2.过滤器解决的如字符编码,日志记录,权限验证,拦截器解决业务流程等

执行分析流程

1.用户发请求到前端控制器DispatcherServlet(用于分发请求)

2.DispatcherServlet收到请求调用HandlerMapping(处理映射器)

3.HandlerMapping找到具体的处理器,生成处理器对象,和处理器拦截器,在返会给DispatcherServlet

4.DispatcherServlet 调用HandlerAdapter

5.HandlerAdapter通过适配调用具体的处理器的某个方法controller/hander

6.controller执行后返回ModelAndViwe对象给HandlerAdapter

7.HandlerAdapter在返回给DispatcherServlet

8.DispatcherServlet将ModleAndView传给视图解析器(viewReslover)

9.viewReslover解析后返回具体VIew(视图)

10.DispatcherServlet根据View进行渲染视图

11.DispatcherServlet将视图响应给用户

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-xh7KAEpZ-1684998497806)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\image-20230513210809013.png)]

spatcherServlet收到请求调用HandlerMapping(处理映射器)

3.HandlerMapping找到具体的处理器,生成处理器对象,和处理器拦截器,在返会给DispatcherServlet

4.DispatcherServlet 调用HandlerAdapter

5.HandlerAdapter通过适配调用具体的处理器的某个方法controller/hander

6.controller执行后返回ModelAndViwe对象给HandlerAdapter

7.HandlerAdapter在返回给DispatcherServlet

8.DispatcherServlet将ModleAndView传给视图解析器(viewReslover)

9.viewReslover解析后返回具体VIew(视图)

10.DispatcherServlet根据View进行渲染视图

11.DispatcherServlet将视图响应给用户

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值