Sping MVC 入门(二)

控制器方法的返回值

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
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,098评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,213评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,960评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,519评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,512评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,533评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,914评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,574评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,804评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,563评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,644评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,350评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,933评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,908评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,146评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,847评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,361评论 2 342

推荐阅读更多精彩内容

  • 对于java中的思考的方向,1必须要看前端的页面,对于前端的页面基本的逻辑,如果能理解最好,不理解也要知道几点。 ...
    神尤鲁道夫阅读 797评论 0 0
  • 1、Spring MVC请求流程 (1)初始化:(对DispatcherServlet和ContextLoderL...
    拾壹北阅读 1,944评论 0 12
  • 第一部分 HTML&CSS整理答案 1. 什么是HTML5? 答:HTML5是最新的HTML标准。 注意:讲述HT...
    kismetajun阅读 27,400评论 1 45
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,082评论 1 32
  • SpringMVC介绍 Spring web mvc 和Struts2都属于表现层的框架,它是Spring框架的一...
    day_Sunny阅读 740评论 0 0