美文网首页
Sping MVC 入门(二)

Sping MVC 入门(二)

作者: 月哥说了算 | 来源:发表于2019-07-08 00:04 被阅读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

    相关文章

      网友评论

          本文标题:Sping MVC 入门(二)

          本文链接:https://www.haomeiwen.com/subject/dilihctx.html