Sping MVC 入门(二)

2019-07-08  本文已影响0人  月哥说了算

控制器方法的返回值

package com.gzy.web.controller;

import com.fasterxml.jackson.databind.ObjectMapper;
import com.itheima.domain.User;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.servlet.ModelAndView;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.util.Date;

@Controller
@RequestMapping("/user")
public class UserController {
    @RequestMapping("/test1")
    public ModelAndView test1(ModelAndView modelAndView){

        String s = new Date().toString();
        modelAndView.addObject("content",s);
        modelAndView.setViewName("user/test");
        return modelAndView;
    }

    /**
     * 可以采用原始的方案来做 傻子才做呢
     * @param request
     * @param response
     * @throws ServletException
     * @throws IOException
     */
    @RequestMapping("/test2")
    public void test2(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {

        //完全按照以前的写法

        String s = new Date().toString();
        request.setAttribute("content",s);
        request.getRequestDispatcher("/WEB-INF/pages/user/test.jsp").forward(request,response);
    }

    /**
     * 玩重定向原始方案
     * @param request
     * @param response
     * @throws ServletException
     * @throws IOException
     */
    @RequestMapping("/test3")
    public void test3(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {

        //完全按照以前的写法
        response.sendRedirect("http://www.baidu.com");
    }
    /**
     * 玩重定向 springmvc方式
     * @throws ServletException
     * @throws IOException
     */
    @RequestMapping("/test4")
    public ModelAndView test4(ModelAndView modelAndView) throws ServletException, IOException {
        modelAndView.setViewName("redirect:http://www.baidu.com");
        //return "redirect:http://www.baidu.com";
        return modelAndView;
    }
    /**
     * 玩重定向 springmvc方式 这个好
     *redirect相当于“response.sendRedirect(url)”。需要注意的 
      是, 如果是重定向到 jsp 页面,则 jsp 页面不
     能写在 WEB-INF 目录中,否则无法找到。
     * @throws ServletException
     * @throws IOException
     */
    @RequestMapping("/test5")
    public String test5() throws ServletException, IOException {
        return "redirect:http://www.baidu.com";
    }

    /**
     * springmvc方  没有必须非要返回 modelandview  完全可以返回字符串
     *
     * 如果字符串 以redirect起头 重定向了 如果不是 返回的是一个视图
     * springmvc 自动包装成一个 modelAndView
     *
     * //先开始springmvc代码
     *
     * //找到你的方法
     *
     * //帮你创建 model对象
     * var model =new model();
     *
     * var obj=userController.test(model);
     *
     * if(obj instanceof modelandview){
     *     //之前的正常逻辑
     * }else if(obj instanceof modelandview){
     *     //检查是否已redirect开头
     *     //是否 直接 resonse.sendRedirect(...)
     *     //不是以它开头
     *     new modelandview().setViewName(obj);
     *     modelandview.setmodel(model);
     *
     * }

     * @throws ServletException
     * @throws IOException
     */
    @RequestMapping("/test6")
    public String test6(Model model) {
        String s = new Date().toString();
        model.addAttribute("content",s);
        return "user/test";
    }

交互 json 数据

ResponseBody

作用:
该注解用于将 Controller 的方法返回的对象,通过 HttpMessageConverter 接口转换为指定格式的
数据如:json,xml 等,通过 Response 响应给客户端。

用法

1.配置springmvc.xml文件(配json转换器)

<mvc:annotation-driven>
        <mvc:message-converters>
            <bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter"/>
        </mvc:message-converters>
    </mvc:annotation-driven>

2.使用注解

 @RequestMapping("test5")
//告诉程序以json字符串返回给客户端
    @ResponseBody
    public User test5() {
        User user = new User(1, "张撒", "44");
        return user;
    }

RequestBody 使用说明

作用:

用于获取请求体内容。直接使用得到是 key=value&key=value...结构的数据。
get 请求方式不适用。

属性:

required:是否必须有请求体。默认值是:true。当取值为 true 时,get 请求方式会报错。如果取值为 false,get 请求得到是 null。

使用

举个例子
1.一个html页面ajax发送数据
注:后台得到前端返回的数据后要有请求头:
contentType:"application/json;charset=utf-8"
否则后台无法转成json数据。

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="/static/js/jquery.min.js"></script>
    <script>
       function f() {
          $.ajax({
              url:"/user/test6",
              type:"post",
              contentType:"application/json;charset=utf-8",
              data:"{\"name\":\"xiaoming\",\"age\":\"18\",\"id\":2}",
              success:function(data){
                  alert(data)
              }
          })
       }
    </script>
</head>
<body>
<h3>演示ajax发送请求 并且发送的数据 不再是传统键值对了 而是json格式的数据</h3>
<input type="button" value="点我发请求" onclick="f()">
</body>
</html>

前端效果:


Snipaste_2019-07-07_23-16-33.png

2.后台代码

 @RequestMapping("test6")
    @ResponseBody
    public String test6(@RequestBody User user) { 
        System.out.println(user);
        return "success";
    }

后台效果:


Snipaste_2019-07-07_23-18-42.png

Restful 风格的 URL

restful风格的url设计

传统风格设计:
商品的增删改查

    1.查询某个商品
        /product/findById
        参数 id=xxxx
    2.添加一个商品
        /product/save
        参数 id=xxx&name=xx&price=xxx
    3.更新一个商品
        /product/update
        参数 id=xxx&name=xx&price=xxx
    4.删除一个商品
        /product/delById
        参数 id=xxx
    5.查询所有商品
        /product/findAll

restful风格:
    商品的增删改查

    1.查询某个商品
         路径                 请求方式
        /product/{id}         GET
    2.添加一个商品
        路径                 请求方式
        /product              PUT
        参数 id=xxx&name=xx&price=xxx
    3.更新一个商品
        路径                 请求方式
        /product              POST
        参数 id=xxx&name=xx&price=xxx
    4.删除一个商品
        路径                 请求方式
        /product/{id}         DELETE
    5.查询所有商品
        路径                 请求方式
        /products            GET
总结:
    就将放在服务器上任何当做一个资源
    操作 增删改查 不在路径写动作名词  而是采用请求方式 表明动作


优点:
    路径看起来简洁

缺点:
    表现力还不够
       注意在本身设计想法中  只要出现名词 代表的是另外一个资源

       商品{
            名字:
            价格:
            id:
            产地:{
                xx省
                xx市
            }

       }

    /product/1/address       GET

    分页查询
    /product/_page
    根据商品名字
    /product/_query

使用

举个例子
前端

注意

规定用<input type="hidden" name="_method" value="xxx">
来模拟特殊请求。(name必须叫"_method" ),框架底层会自动识别到value中的值及请求方式。

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
    <h3>模拟restful风格请求方式</h3>
    <fieldset>
        <lengend>模拟post请求</lengend>
        <form action="/product" method="post">
            <input type="hidden" name="id" value="2">
            名字:<input type="text" name="name"><br>
            价格:<input type="text" name="price"><br>
            <input type="submit" value="点我提交更新">
        </form>

    </fieldset>
    <fieldset>
        <lengend>模拟put请求 添加商品</lengend>
        <form action="/product" method="post">
            <input type="hidden" name="_method" value="PUT">
            名字:<input type="text" name="name"><br>
            价格:<input type="text" name="price"><br>
            <input type="submit" value="点我提">
        </form>

    </fieldset>
    <fieldset>
        <lengend>模拟delete 添加商品</lengend>
        <form action="/product" method="post">
            <input type="hidden" name="_method" value="delete">
            <input type="text" name="id">
            <input type="submit" value="点我提交">
        </form>

    </fieldset>
</body>
</html>

web.xml配置
配置org.springframework.web.filter.HiddenHttpMethodFilte过滤器,对模拟的请求方式睁一只眼闭一只眼。

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd"
         version="3.1">

  <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:dispatcher-servlet.xml</param-value>
    </init-param>
  </servlet>
  <servlet-mapping>
    <servlet-name>springmvc</servlet-name>
    <url-pattern>/</url-pattern>
    <!--
      这个地方你可以  *.do *.action / 千万不要写/*

      /* 属于目录匹配的
      /属于最后那个特殊
      实际含义 / 和/*  表示匹配所有
      因为 *.jsp的存在
      精确匹配 目录匹配 后缀名匹配  /
      第二个方案  写 *.do 你在访问的时候 必须写xxx.do
    -->
  </servlet-mapping>

<!--
  继续将静态资源访问 交给默认servlet来处理
-->
  <servlet-mapping>
    <servlet-name>default</servlet-name>
    <url-pattern>*.css</url-pattern>
    <url-pattern>*.png</url-pattern>
    <url-pattern>*.html</url-pattern>
    <url-pattern>*.htm</url-pattern>
    <url-pattern>*.txt</url-pattern>
    <url-pattern>*.jpg</url-pattern>
    <url-pattern>*.js</url-pattern>
  </servlet-mapping>
  <filter>
    <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>
    <!--<init-param>
      <param-name>forceResponseEncoding</param-name>
      <param-value>true</param-value>
    </init-param>-->
  </filter>
  <filter-mapping>
    <filter-name>encodingFilter</filter-name>
    <url-pattern>/*</url-pattern>
  </filter-mapping>


  <filter>
    <filter-name>hiddenMethodFilter</filter-name>
    <filter-class>org.springframework.web.filter.HiddenHttpMethodFilter</filter-class>
  </filter>
  <filter-mapping>
    <filter-name>hiddenMethodFilter</filter-name>
    <url-pattern>/*</url-pattern>
  </filter-mapping>
</web-app>

后台

@Controller

public class ProductController {
    @RequestMapping(value = "/product/{id}",method = RequestMethod.GET)
    @ResponseBody
    public Product findById(@PathVariable("id") int id){
        //mock呗
        Product product = new Product();
        product.setId(id);
        product.setName("虾米手机");
        product.setPrice(2999);
        return product;
    }

    @RequestMapping(value = "/product",method = RequestMethod.POST)
    @ResponseBody
    public String update(Product product){
        //mock呗
        System.out.println(product);

        return "success";
    }
    @RequestMapping(value = "/product",method = RequestMethod.PUT)
    @ResponseBody
    public String save(Product product){
        //mock呗
        System.out.println(product);

        return "success";
    }
    @RequestMapping(value = "/product",method = RequestMethod.DELETE)
    @ResponseBody
    public String delete(int id){
        //mock呗
        System.out.println(id);

        return "success";
    }
    @RequestMapping(value = "/product",method = RequestMethod.GET)
    @ResponseBody
    public List findAll(){
        //mock呗
        ArrayList<Object> list = new ArrayList<>();
        Product product1 = new Product();
        product1.setId(1);
        product1.setName("虾米手机");
        product1.setPrice(2999);
        list.add(product1);
        Product product2 = new Product();
        product2.setId(2);
        product2.setName("华为手机");
        product2.setPrice(4999);
        list.add(product2);
        return list;
    }
}

springMVC方式文件上传

1.添加Maven依赖

<!-- https://mvnrepository.com/artifact/commons-fileupload/commons-fileupload -->
<!--apache提供了 一个工具类 来帮助解析这种格式-->
    <dependency>
      <groupId>commons-fileupload</groupId>
      <artifactId>commons-fileupload</artifactId>
      <version>1.4</version>
    </dependency>
    <!-- https://mvnrepository.com/artifact/commons-io/commons-io -->
<!---操作流的工具包-->
    <dependency>
      <groupId>commons-io</groupId>
      <artifactId>commons-io</artifactId>
      <version>2.6</version>
    </dependency>

2.注入文件解析对象

<!--这里id必须叫multipartResolver,与底层有关-->
<bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
        <property name="maxUploadSize" value="100000000"/>
    </bean>

3.前端开发

注意:

springmvc对于文件上传的支持
完成文件上传本质就是流的传输
1.必须有一个表单
2.表单提交方式必须post
3.表单必须存在file框
4.表单属性 必须改 enctype="multipart/form-data",默认提交的数据是键值对形式的,设置此属性表示,有什么数据就提交什么数据。

代码:
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>上传</title>
</head>
<body>
<form enctype="multipart/form-data" method="post" action="/upload">
    上传头像:<input type="file" name="avatar">
    <br><input type="submit" value="上传">
</form>
</body>
</html>
后端:

小问题:
1.文件名重复问题
拿到文件 就名字改了 保证唯一了 廉价 uuid
2.某个目录文件过多问题
分散目录

 package com.gzy.controller;

import org.apache.commons.io.IOUtils;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.multipart.MultipartFile;


import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.Random;
import java.util.UUID;

@Controller
public class UploadController {
    @RequestMapping("upload")
    @ResponseBody
   //注意:这里avatar必须与前端的<input type="file"name="avatar">
   //的name 一致。
    public String upload(MultipartFile avatar) throws IOException {
        //获取上传文件名
        String filename = avatar.getOriginalFilename();
        System.out.println(filename);
        //防止文件名重复利用uuid 获取随机名称
        String uuidName = getUUIDName(filename);
        System.out.println(uuidName);
       //得到上传文件流对象
        InputStream inputStream = avatar.getInputStream();
      //获取随机路径
        String secondDir = randDir();
        String path="E:\\"+secondDir;
        mustExist(path);
        FileOutputStream fileOutputStream = new FileOutputStream(path+uuidName);
        IOUtils.copy(inputStream,fileOutputStream);
        inputStream.close();
        fileOutputStream.close();
        return "success";
    }
    /**
     * 获取随机名称
     * @param realName 真实名称
     * @return uuid 随机名称
     */
    public String getUUIDName(String realName){
        //realname  可能是  1.jpg   也可能是  1
        //获取后缀名
        int index = realName.lastIndexOf(".");
        if(index==-1){
            return UUID.randomUUID().toString().replace("-", "").toUpperCase();
        }else{
            return UUID.randomUUID().toString().replace("-", "").toUpperCase()+realName.substring(index);
        }
    }
    /**
     * 获取文件目录,可以获取256个随机目录
     * @return 随机目录 /a/4/  /b/c/
     */
    public String randDir(){
        String s="0123456789ABCDEF";
        Random r = new Random();
        return "/"+s.charAt(r.nextInt(16))+"/"+s.charAt(r.nextInt(16))+"/";
    }
//判断文件夹是否存在,否则创建
    public static void mustExist(String path){
        File file = new File(path);
        if (!file.exists()){
            file.mkdirs();
        }

    }
}


SpringMVC全局异常处理

自定义异常处理器
package com.gzy.excition;

import org.springframework.web.servlet.HandlerExceptionResolver;
import org.springframework.web.servlet.ModelAndView;

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

public class MyExceptinHander implements HandlerExceptionResolver {
    @Override
    public ModelAndView resolveException(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, Exception e) {
        ModelAndView modelAndView = new ModelAndView();
        if(e instanceof ArithmeticException){
            modelAndView.addObject("code","1234");
        }
        modelAndView.setViewName("error/beautiful");
        return modelAndView;
    }
}

springmvc.xml配置异常处理器
<bean id="error" class="com.gzy.excition.MyExceptinHander"/>
错误演示

代码:

 @RequestMapping("test7")
    @ResponseBody
    public String test7() {
       int i=8/0;
        return "success";
    }

异常跳转页面

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
    <h3>sorry!!你访问页面去浪了.........</h3>
    <hr>
    不好意思,发生错误:${code}

    <br>
    如有疑问,请拨打:1231992213
</body>
</html>

效果


Snipaste_2019-07-08_19-50-15.png
上一篇下一篇

猜你喜欢

热点阅读