【SpringMVC】RestFul风格&数据处理及跳转

1、RestFul风格

概念

Restful就是一个资源定位及资源操作的风格。不是标准也不是协议,只是一种风格。基于这个风格设计的软件可以更简洁,更有层次,更易于实现缓存等机制。

功能

资源:互联网所有的事物都可以被抽象为资源

资源操作:使用POST、DELETE、PUT、GET,使用不同方法对资源进行操作。

分别对应 添加、 删除、修改、查询。

传统方式操作资源 :通过不同的参数来实现不同的效果,方法单一,post 和 get

http://127.0.0.1/item/queryItem.action?id=1 查询,GET

http://127.0.0.1/item/saveItem.action 新增,POST

http://127.0.0.1/item/updateItem.action 更新,POST

http://127.0.0.1/item/deleteItem.action?id=1 删除,GET或POST
使用RESTful操作资源 :可以通过不同的请求方式来实现不同的效果,如下:请求地址一样,但是功能可以不同!

http://127.0.0.1/item/1 查询,GET

http://127.0.0.1/item 新增,POST

http://127.0.0.1/item 更新,PUT

http://127.0.0.1/item/1 删除,DELETE

实际运用

在Spring MVC中可以使用 @PathVariable 注解,让方法参数的值对应绑定到一个URI模板变量

下面这个例子也就是说参数a,b绑定到了URL上

package com.adie.controller;

import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.*;

@Controller
public class RestFulController {
    //原来的请求:http://localhost:8080/add?a=1&b=2
    //RestFul: http://localhost:8080/add/a/b
    @RequestMapping(value = "/add/{a}/{b}")
    public String test1(@PathVariable int a,@PathVariable int b, Model model){
        int res=a+b;
        model.addAttribute("msg","结果1为"+res);
        return "test";
    }
}

请添加图片描述

我们使用RestFul风格

  • 使路径变得更加简洁;
  • 获得参数更加方便,框架会自动进行类型转换。
  • 通过路径变量的类型可以约束访问参数,如果类型不一样,则访问不到对应的请求方法,如这里访问是的路径是/add/1/“10”,则路径与方法不匹配,而不会是参数转换失败。
    请添加图片描述

使用method属性指定请求类型

用于约束请求的类型,可以收窄请求范围。指定请求谓词的类型如GET, POST, HEAD, OPTIONS, PUT, PATCH, DELETE, TRACE等

修改我们刚刚的函数为

package com.adie.controller;

import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.*;

@Controller
public class RestFulController {
    //原来的请求:http://localhost:8080/add?a=1&b=2
    //RestFul: http://localhost:8080/add/a/b
    @RequestMapping(value = "/add/{a}/{b}",method = RequestMethod.POST)
    public String test1(@PathVariable int a,@PathVariable int b, Model model){
        int res=a+b;
        model.addAttribute("msg","结果1为"+res);
        return "test";
    }
}

结果

请添加图片描述

总结:

Spring MVC 的 @RequestMapping 注解能够处理 HTTP 请求的方法, 比如 GET, PUT, POST, DELETE 以及 PATCH。所有的地址栏请求默认都会是 HTTP GET 类型的。

方法级别的注解变体有如下几个:组合注解

@GetMapping
@PostMapping
@PutMapping
@DeleteMapping
@PatchMapping

@GetMapping就是我们前面所用的@RequestMapping(method =RequestMethod.GET) 的组合注解。

@RequestMapping(value = “/add/{a}/{b}”,method = RequestMethod.POST)换成组合注解就是@GetMapping("/add/{a}/{b}")

这样简洁方便了许多

2、数据处理及跳转

2.1、结果跳转方式

1.ModelAndView

1.先配置视图解析器 返回的页面是{视图解析器前缀} + viewName +{视图解析器后缀}

<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver" id="InternalResourceViewResolver">
    <!--前缀-->
    <property name="prefix" value="/WEB-INF/jsp/"/>
    <!--后缀-->
    <property name="suffix" value=".jsp"/>
</bean>

2.写对应的Controller类

package com.adie.controller;

import org.springframework.web.servlet.ModelAndView;
import org.springframework.web.servlet.mvc.Controller;

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

public class HelloController implements Controller {
    @Override
    public ModelAndView handleRequest(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse) throws Exception {
        //ModelAndView  模型和视图
        ModelAndView mv = new ModelAndView();


        String result="HelloSpringMVC";
        //封装对象,放在ModelAndView中
        mv.addObject("msg",result);

        //封装要跳转的视图,放在ModelAndView
        mv.setViewName("test");// /WEB-INF/jsp/test.jsp
        return mv;
    }
}

2.通过SpringMVC注解来实现转发和重定向

不需要视图解析器

请求转发

@Controller
public class ModelTest1 {
    @RequestMapping("/m1/t1")
    public String test1(Model model){
        model.addAttribute("msg","进来了");
        return "forward:/WEB-INF/jsp/test.jsp";
    }
    
    //请求转发
    @RequestMapping("/m1/t2")
    public String test2(Model model){
        return "/WEB-INF/jsp/test.jsp";
    }
}

重定向

@Controller
public class ModelTest1 {
    @RequestMapping("/m1/t1")
    public String test1(Model model){
        return "redirect:/index.jsp";
    }
}

加上视图解析器

重定向有无视图解析器的写法都是一样的

@Controller
public class ModelTest1 {
    //请求转发
    @RequestMapping("/m1/t2")
    public String test2(Model model){
        return "test";
    }

    //重定向
    @RequestMapping("/m1/t1")
    public String test1(Model model){
        return "redirect:/index.jsp";
    }
}

2.2、数据处理

1、提交的域名称和处理方法的参数名一致

//    提交数据 : http://localhost:8080/hello?name=adie
    @GetMapping("t1")
    public String test1(String name, Model model){
//        1.接收前端参数
        System.out.println("接收到前端的参数为"+name);
//        2.将返回的结果传递给前端
        model.addAttribute("msg",name);
//        3.跳转视图
        return "test";
    }

2、提交的域名称和处理方法的参数名不一致

这时候需要使用@RequestParam注解来统一名称

//    提交数据 : http://localhost:8080/hello?username=adie
    @GetMapping("t1")
//    只要是前端的参数都加上@RequestParam(),除了可以起别名还可以用于区分是否为前端传来的参数
//    不用@RequestParam()不管前端传递过来了什么这里都能接收到,用了之后前端传递名称不对的参数会报错
    public String test1(@RequestParam("username") String name, Model model){
//        1.接收前端参数
        System.out.println("接收到前端的参数为"+name);
//        2.将返回的结果传递给前端
        model.addAttribute("msg",name);
//        3.跳转视图
        return "test";
    }

只要是前端的参数都加上@RequestParam(),除了可以起别名还可以用于区分是否为前端传来的参数
不用@RequestParam()时,不管前端传递过来了什么这里都能会进入这个方法,但是用了之后前端传递名称不对的参数会报错

3、提交的是一个对象

1.我们先定义一个实体类

package com.adie.pojo;

import lombok.Data;

@Data
public class User {
    private int id;
    private String name;
    private int age;
}

2.Controller处理方法

http://localhost:8080/user/t2?id=1&name=a碟&age=19

//    前端传递的是一个对象:id,name,age
//    1.接收前端用户的传递的参数,判断参数的名字,假设名字直接在方法上,可以直接使用
//    2.假设传递的是一个对象user,匹配user对象中的字段名:如果名字一致则行,否则匹配不到
    @GetMapping("/t2")
    public String test2(User user){
        System.out.println(user);
        return "test";
    }

如果使用对象的话,前端传递的参数名和对象名必须一致,否则就是null。

2.3、数据显示到前端

1.ModelAndView

Controller使用接口定义时一般使用ModelAndView,用来封装对象和封装视图

package com.adie.controller;

import org.springframework.web.servlet.ModelAndView;
import org.springframework.web.servlet.mvc.Controller;

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

public class HelloController implements Controller {
    @Override
    public ModelAndView handleRequest(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse) throws Exception {
        //ModelAndView  模型和视图
        ModelAndView mv = new ModelAndView();


        String result="HelloSpringMVC";
        //封装对象,放在ModelAndView中
        mv.addObject("msg",result);

        //封装要跳转的视图,放在ModelAndView
        mv.setViewName("test");// /WEB-INF/jsp/test.jsp
        return mv;
    }
}

2.Model

注解实现的SpringMVC,返回的视图就是return的值经过视图解析器后解析出来的视图。所以只需要封装模型就是封装对象

    @GetMapping("t1")
    public String test1(@RequestParam("username") String name, Model model){
        System.out.println("接收到前端的参数为"+name);
        model.addAttribute("msg",name);
        return "test";
    }

3. 通过ModelMap

这种使用的比较少,简单来说,Model就是简单版本的ModelMap,ModelMap继承了 LinkedMap,除了实现了自身的一些方法,同样的继承 LinkedMap 的方法和特性

这三种的区别简单来说

  • Model 只有几个方法只适合用于储存数据,简化了新手对于Model对象的操作和理解;

  • ModelMap 继承了 LinkedMap ,除了实现了自身的一些方法,同样的继承 LinkedMap 的方法和特性;

  • ModelAndView 可以在储存数据的同时,可以进行设置返回的逻辑视图,进行控制展示层的跳转。

2.4、乱码问题处理

1.先写一个简单的表单

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
<form action="/e/t1" method="post">
    <input type="text" name="name">
    <input type="submit">
</form>
</body>
</html>

2.后台处理Controller

package com.adie.controller;

import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.PostMapping;
@Controller
public class EncodingController {
//    过滤器解决乱码
    @PostMapping("/e/t1")
    public String test1(String name, Model model){
        System.out.println(name);
        model.addAttribute("msg",name);
        return "test";
    }
}

输入中文测试一般都会出现乱码问题

乱码在我们的做项目过程中经常会出现,这里介绍几种处理的方法

1.自己写一个过滤器处理乱码

package com.adie.filter;

import javax.servlet.*;
import java.io.IOException;

public class EncodingFilter implements Filter {
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {

    }

    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain filterChain) throws IOException, ServletException {
        request.setCharacterEncoding("utf-8");
        response.setCharacterEncoding("utf-8");
        filterChain.doFilter(request,response);
    }

    @Override
    public void destroy() {

    }
}

接着需要在web.xml中注册过滤器

<filter>
    <filter-name>encoding</filter-name>
    <filter-class>com.adie.filter.EncodingFilter</filter-class>
</filter>
<filter-mapping>
    <filter-name>encoding</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>

2.使用SpringMVC提供的过滤器

<filter>
    <filter-name>encoding</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>encoding</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>

但是这个过滤器在有些极端情况下对get请求的乱码支持不好

3.修改Tomcat的配置文件

在server.xml中配置

<Connector URIEncoding="utf-8" port="8080" protocol="HTTP/1.1"
           connectionTimeout="20000"
           redirectPort="8443" />

4.参考网上大佬的万能自定义过滤器的解决方法

这个过滤器很好的处理了get和post请求

package com.kuang.filter;
 
import javax.servlet.*;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletRequestWrapper;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.util.Map;
 
/**
 * 解决get和post请求 全部乱码的过滤器
 */
public class GenericEncodingFilter implements Filter {
 
    @Override
    public void destroy() {
    }
 
    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
        //处理response的字符编码
        HttpServletResponse myResponse=(HttpServletResponse) response;
        myResponse.setContentType("text/html;charset=UTF-8");
 
        // 转型为与协议相关对象
        HttpServletRequest httpServletRequest = (HttpServletRequest) request;
        // 对request包装增强
        HttpServletRequest myrequest = new MyRequest(httpServletRequest);
        chain.doFilter(myrequest, response);
    }
 
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {
    }
 
}
 
//自定义request对象,HttpServletRequest的包装类
class MyRequest extends HttpServletRequestWrapper {
 
    private HttpServletRequest request;
    //是否编码的标记
    private boolean hasEncode;
    //定义一个可以传入HttpServletRequest对象的构造函数,以便对其进行装饰
    public MyRequest(HttpServletRequest request) {
        super(request);// super必须写
        this.request = request;
    }
 
    // 对需要增强方法 进行覆盖
    @Override
    public Map getParameterMap() {
        // 先获得请求方式
        String method = request.getMethod();
        if (method.equalsIgnoreCase("post")) {
            // post请求
            try {
                // 处理post乱码
                request.setCharacterEncoding("utf-8");
                return request.getParameterMap();
            } catch (UnsupportedEncodingException e) {
                e.printStackTrace();
            }
        } else if (method.equalsIgnoreCase("get")) {
            // get请求
            Map<String, String[]> parameterMap = request.getParameterMap();
            if (!hasEncode) { // 确保get手动编码逻辑只运行一次
                for (String parameterName : parameterMap.keySet()) {
                    String[] values = parameterMap.get(parameterName);
                    if (values != null) {
                        for (int i = 0; i < values.length; i++) {
                            try {
                                // 处理get乱码
                                values[i] = new String(values[i]
                                        .getBytes("ISO-8859-1"), "utf-8");
                            } catch (UnsupportedEncodingException e) {
                                e.printStackTrace();
                            }
                        }
                    }
                }
                hasEncode = true;
            }
            return parameterMap;
        }
        return super.getParameterMap();
    }
 
    //取一个值
    @Override
    public String getParameter(String name) {
        Map<String, String[]> parameterMap = getParameterMap();
        String[] values = parameterMap.get(name);
        if (values == null) {
            return null;
        }
        return values[0]; // 取回参数的第一个值
    }
 
    //取所有值
    @Override
    public String[] getParameterValues(String name) {
        Map<String, String[]> parameterMap = getParameterMap();
        String[] values = parameterMap.get(name);
        return values;
    }
}

然后同样要在web.xml中配置

乱码问题,需要平时多注意,在尽可能能设置编码的地方,都设置为统一编码 UTF-8

注意:前面在web.xml中配置过滤器的时候

<filter-mapping>
    <filter-name>encoding</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>

这一部分,/*,我们一定要配置过滤器处理的请求为**/ ***,而不是/,因为/是不会处理jsp页面的。

2.5、JSON数据的处理

1.jackson

Jackson应该是目前比较好的json解析工具了

使用Jackson,需要导入它的jar包;

<!-- https://mvnrepository.com/artifact/com.fasterxml.jackson.core/jackson-databind -->
<dependency>
    <groupId>com.fasterxml.jackson.core</groupId>
    <artifactId>jackson-databind</artifactId>
    <version>2.12.4</version>
</dependency>

编写一个User的实体类

package com.adie.pojo;

import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
 
//需要导入lombok
@Data
@AllArgsConstructor
@NoArgsConstructor
public class User {
 
    private String name;
    private int age;
    private String sex;
    
}

编写Controller

使用@ResponseBody //它就不会走视图解析器,会直接返回一个字符串

或者使用@RestController,就不用在每一个方法前加@ResponseBody了,在前后端分离中我们经常这样配置

@Controller
public class UserController {

    @RequestMapping(value = "/j1")
    @ResponseBody //它就不会走视图解析器,会直接返回一个字符串
    public String json1() throws JsonProcessingException {
        //jackson   ObjectMapper
        ObjectMapper mapper = new ObjectMapper();
        User user=new User("a碟",18,"男");
        String str = mapper.writeValueAsString(user);
        return str;
    }
}

测试

请添加图片描述

现了乱码问题,我们需要设置一下他的编码格式为utf-8,以及它返回的类型;

0通过配置@RequestMapping中的produces属性解决乱码问题

//produces:指定响应体返回类型和编码
@RequestMapping(value = "/json1",produces = "application/json;charset=utf-8")

还有一种乱码统一解决的方法

我们可以在springmvc的配置文件上添加一段消息StringHttpMessageConverter转换配置

<!--    JSON乱码问题配置-->
    <mvc:annotation-driven>
        <mvc:message-converters register-defaults="true">
            <bean class="org.springframework.http.converter.StringHttpMessageConverter">
                <constructor-arg value="UTF-8"/>
            </bean>
            <bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter">
                <property name="objectMapper">
                    <bean class="org.springframework.http.converter.json.Jackson2ObjectMapperFactoryBean">
                        <property name="failOnEmptyBeans" value="false"/>
                    </bean>
                </property>
            </bean>
        </mvc:message-converters>
    </mvc:annotation-driven>

处理多个对象

接着我们增加一个方法

@RequestMapping(value = "/j2")
@ResponseBody //它就不会走视图解析器,会直接返回一个字符串
public String json2() throws JsonProcessingException {
    //jackson   ObjectMapper
    ObjectMapper mapper = new ObjectMapper();
    List<User> userList=new ArrayList<User>();
    User user1=new User("a碟1",18,"男");
    User user2=new User("a碟2",18,"男");
    User user3=new User("a碟3",18,"男");
    User user4=new User("a碟4",18,"男");

    userList.add(user1);
    userList.add(user2);
    userList.add(user3);
    userList.add(user4);
    String str = mapper.writeValueAsString(userList);
    
    return str;
}

请添加图片描述

也是能够很好的处理的

输出时间对象

    @RequestMapping(value = "/j3")
    @ResponseBody //它就不会走视图解析器,会直接返回一个字符串
    public String json3() throws JsonProcessingException {
        //jackson   ObjectMapper
        ObjectMapper mapper = new ObjectMapper();
        Date date = new Date();
        //        ObjectMapper,时间解析后的默认格式为TimeStamp:时间戳
        return mapper.writeValueAsString(date);
    }

请添加图片描述

  • 默认日期格式会变成一个数字,是1970年1月1日到当前日期的毫秒数!
  • Jackson 默认是会把时间转成timestamps形式 时间戳

解决方案:取消timestamps形式 , 自定义时间格式

    @RequestMapping(value = "/j3")
    @ResponseBody //它就不会走视图解析器,会直接返回一个字符串
    public String json3() throws JsonProcessingException {
//        jackson   ObjectMapper
        ObjectMapper mapper = new ObjectMapper();

//        使用ObjectMapper 来格式化输出,默认使用时间戳,我们关闭时间戳
        mapper.configure(SerializationFeature.WRITE_DATE_KEYS_AS_TIMESTAMPS,false);

        Date date = new Date();
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");

        mapper.setDateFormat(sdf);
//        ObjectMapper,时间解析后的默认格式为TimeStamp:时间戳
        return mapper.writeValueAsString(sdf.format(date));
    }

进一步优化,如果经常使用,我们可以抽取为工具类

将代码封装为工具类

package com.adie.utils;


import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.SerializationFeature;

import java.text.SimpleDateFormat;
import java.util.Date;

public class JsonUtils {

    public static String getjson(Object object) throws JsonProcessingException {
//        默认设置日期格式为yyyy-MM-dd HH:mm:ss
        return getjson(object,"yyyy-MM-dd HH:mm:ss");
    }

    public static String getjson(Object object,String dateFormat) throws JsonProcessingException {
        ObjectMapper mapper = new ObjectMapper();
//        使用ObjectMapper 来格式化输出,默认使用时间戳,我们关闭时间戳
        mapper.configure(SerializationFeature.WRITE_DATE_KEYS_AS_TIMESTAMPS,false);
//        自定义日期的格式
        SimpleDateFormat sdf = new SimpleDateFormat(dateFormat);

        mapper.setDateFormat(sdf);

        return mapper.writeValueAsString(object);
    }
}

那么我们的代码就更加简洁了

    @RequestMapping(value = "/j3")
    @ResponseBody 
    public String json3() throws JsonProcessingException {
        Date date = new Date();

        return JsonUtils.getjson(date,"yyyy-MM-dd HH:mm:ss");
    }

2.FastJson

fastjson.jar是阿里开发的一款专门用于Java开发的包,可以方便的实现json对象与JavaBean对象的转换,实现JavaBean对象与json字符串的转换,实现json对象与json字符串的转换。实现json的转换方法很多,最后的实现结果都是一样的。

导入jar包

<dependency>
    <groupId>com.alibaba</groupId>
    <artifactId>fastjson</artifactId>
    <version>1.2.60</version>
</dependency>

fastjson 三个主要的类:

JSONObject 代表 json 对象

  • JSONObject实现了Map接口, 猜想 JSONObject底层操作是由Map实现的。

  • JSONObject对应json对象,通过各种形式的get()方法可以获取json对象中的数据,也可利用诸如size(),isEmpty()等方法获取"键:值"对的个数和判断是否为空。其本质是通过实现Map接口并调用接口中的方法完成的。

JSONArray 代表 json 对象数组

  • 内部是有List接口中的方法来完成操作的。

JSON代表 JSONObject和JSONArray的转化

  • JSON类源码分析与使用

  • 仔细观察这些方法,主要是实现json对象,json对象数组,javabean对象,json字符串之间的相互转化。

测试

@RequestMapping(value = "/j4")
@ResponseBody 
public String json4() throws JsonProcessingException {
    Date date = new Date();
    List<User> userList=new ArrayList<User>();
    User user1=new User("a碟1",18,"男");
    User user2=new User("a碟2",18,"男");
    User user3=new User("a碟3",18,"男");
    User user4=new User("a碟4",18,"男");

    userList.add(user1);
    userList.add(user2);
    userList.add(user3);
    userList.add(user4);
    String s = JSON.toJSONString(userList);
    return s;
}
  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

a碟

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值