使用springmvc的controller层获取到请求的数据方式

目录
  • content-type
  • 1.application/x-www-form-urlencoded
    • 1.1 String 参数值=reqeust.getParameter(“参数名”);
    • 1.2 Map<String,String[]> paramMap=request.getParameterMap();
  • 2.multipart/form-data
  • 3.text/xml
  • 4.application/json
    • 4.1 页面请求的数据获取
      • 4.1.1 页面js请求如下
      • 4.1.2 如果页面js请求如下
    • 4.1总结
    • 4.2 当别的应用调用我们的服务

javaweb应用开发,最常用到的就是应用之间的信息交互,调用别的应用模块得到自己想获取的数据信息,然后解析数据并进行处理返回到上一层。那controller层的方法如何获取http请求/webservice请求中的参数,或者说请求数据呢?

按照我个人的接触和理解,把请求可以分为两大类:

1.页面请求到后台controller层的请求。

2.别的应用http/webservice调用本应用中封装接口的请求。

无论哪种方式的请求,都是http请求(webservice底层也是http请求,待考证),controller层对请求数据的解析是按照json还是xml的依据就是请求头content-type,接下来我们就先来详细介绍一下。

content-type

content-type请求头是干吗的呢,http请求头有四种类型,分别是通用头部,请求头部,响应头部以及内容头部. 首先,我们要弄清楚,content-type是属于内容头部,既然是内容头部,那这个请求头是用来向接收端解释传递的该内容主体的,content-type的取值是告诉服务端,你传递过去的内容是啥,你应该准备好如何接收和解析。常用的content-type可以分为application/x-www-form-urlencoded , multipart/form-data ,application/json , text/xml四种。

下面一一介绍

1.application/x-www-form-urlencoded

application/x-www-form-urlencoded是常用的表单发包方式,普通的表单提交(即不带文件上传的提交),或者js (例如:post方式)发包,默认都是通过这种方式,提交代码demo如下:

function searchInfo() {
    $("#cxjg").datagrid({
        url : basepath + 'jijin/getList',
        queryParams : getParam($("#query").serializeArray()),
        method : "post"
    });
function searchInfo(){
    $("#cxjg").datagrid({
        url:basepath+'jijin/getList',
        queryParams : {'age':23,'name':'zhangsan'},
        method:"post"
});

请求头信息如下所示,可以看到content-type和FormData(表单的数据格式)

如何获取这种数据结构的请求?有两种方式来解决

1.1 String 参数值=reqeust.getParameter(“参数名”);

(以springmvc 的controller层的方法为例)

@RequestMapping("/getParam")
public String getParamFromRequest(HttpServletRequest request){
    String jiJinCode=(String)request.getParameter("jinJinCode");
    ...
}

1.2 Map<String,String[]> paramMap=request.getParameterMap();

@RequestMapping("/getParam")
public String getParamFromRequest(HttpServletRequest request){
    Map<String,String[]> paramMap=request.getParameterMap();
    StringBuffer inputDate=new StringBuffer();
    inputDate.append("<jiJinBean>");
    for(Map.Entry<String,String[]> entry :paramMap.entrySet()){
        String key=entry.getKey();
        String[] values=entry.getValue();
        String value=null;
        if(values.length>0){
            value=values[0];
        }
        inputDate.append("<"+key+">"+value+"</"+key+">");
    }
    return inputDate.toString();
}

两种方法比较来说,第一种适合当参数不多时,可以一个一个取。但第二种则是遍历出所有的请求参数,并生成标签格式,利于自动生成报文格式。

2.multipart/form-data

multipart/form-data用在页面(提交)发送文件的POST请求。之所以会和第一种content-type同是post提交,为啥在这里就变成multipart/form-data了呢?

当然程序不会自动识别是不是文件,而是我们在<form>这个标签里加上了这句话:

enctype="multipart/form-data",这才是让请求头发生变化的原因!具体代码demo如下

$("#batchAddInfo").form('submit',{
                url:basepath+"jijin/fileUpload",
                onSubmit:function(){},
                success:function(data){
                    data=eval('('+data+')');
                    var flag=data.flag;
                    if(flag){
                        $("#batchDivDialog").window("close");
                        $.messager.alert('提示',"批量新增成功!");
                        searchInfo();
                    }else{
                        $("#batchDivDialog").window("close");
                        $.messager.alert('提示',"批量新增失败!");
                        searchInfo();
                    }
                }
            });

对应的谷歌页面request调试信息如下:

这里Content-Type告诉我们,发包是以multipart/form-data格式来传输,另外,还有boundary用于分割数据。

这种方式的请求数据如何获取?出文件外的表单参数值当然还是和上面一样——request.getParameter(“parameterName”)来获取,不一样的是文件怎么来获取?要用到MultipartHttpServletRequest,这个MultipartHttpServletRequest 继承于HttpServletRequest,MultipartRequest。

我们要从这个子类里面来获取,具体代码如下:

@RequestMapping("/fileUpload")
public  void fileUpload(HttpServletRequest request,HttpServletResponse response){
    InputStream is=null;
    //创建解析器
    CommonsMultipartResolver resolver=new CommonsMultipartResolver(request.getSession().getServletContext());
    //判断request是否有文件上传,即多部请求
    if(resolver.isMultipart(request)){
        MultipartHttpServletRequest multiRequest=(MultipartHttpServletRequest)request;
        //获取多部request中的文件名
        Iterator<String> iter=multiRequest.getFileNames();
        while(iter.hasNext()){
            //取得上传的文件
            MultipartFile file=multiRequest.getFile(iter.next());
            if(file!=null){
                is=new ByteArrayInputStream(file.getBytes());
            }
        }
    }
Workbook wb=new XSSFWorkbook(is);
...
}

3.text/xml

这种请求头一般出现实在webservice请求时才会有。由于别的应用请求我们,所以我们这边就是服务端,他们是客户端。这里无法用浏览器模拟,只能给出客户端和服务端代码,然后看如何解析客户端的请求数据。

客户端代码

public class AxisClientTest {
    public static void main(String[] args) throws Exception {
  String nameSpaceURI = "com.serviceTargetName";
  String publishUrl = "http://localhost:8080/springCXFWebserviceDemo01/service/HelloWorldService?wsdl";
 Service service = new Service();
// 通过service创建call对象
 Call call = (Call) service.createCall();
 // 设置webservice接口地址
 call.setTargetEndpointAddress(new URL(publishUrl));
 call.setOperationName(new QName(nameSpaceURI, "sayHello"));
 call.addParameter("parameterName", XMLType.XSD_STRING, ParameterMode.IN);
call.setReturnType(XMLType.XSD_STRING);
  // 给方法传递参数,并且调用方法
  String name = "zhanglifeng";
  String temp = getXml(name);
  // 这里的obj{}是放入几个入参,完全由service提供的接口方法的入参决定,且顺序和你存放的顺序一致!一般入参为String类型的xml报文,回参也是xml报文。
  Object[] obj = new Object[] { temp };
  String result = (String) call.invoke(obj);
}
    private static String getXml(String name) {
        StringBuffer sb = new StringBuffer(
                "<?xml version=\"1.0\" encoding=\"UTF-8\"?>");
        sb.append("<userBean>");
        sb.append("<userName>" + name + "</userName>");
        sb.append("</userBean>");
        return sb.toString();
    }
}

上面的方法getXml(String name)就是客户端发给服务端的请求:

<?xml version="1.0" encoding="UTF-8"?>
    <userBean>
        <userName>" + 入参:人名 + </userName>
    </userBean>

下面我们再看服务端,也就是我们这边接受请求的代码是啥样的:

/**
 * 由于实现类和接口不在同一个包中。所以要加上targetNamespace属性。
 * 另外,这里的endpointInterface是实现类对应接口的全路径
 * @author Administrator
 */
@WebService(targetNamespace="com.serviceTargetName",endpointInterface="com.zlf.HelloWorld")
@Component("HelloWord")//spring注入用
public class HelloWorldImpl implements HelloWorld {
    @Override
     public String sayHello(String str) {
        String username="aaa";
        Document document = XMLUtils.parse(str);
        //首先接口开发肯定是双发都知道此方法要接受的报文格式的。我们获取报文中人名对应的节点即可。
          Node node = document.getElementsByTagName("userName").item(0);
          if(node !=null){
              username=node.getTextContent();
          }
        return "你好,"+username+"  你已成功访问了webservice服务端!" ;
    }
}

这里我们是直接拿着入参str进行Documen解析,因为客户端的请求数据已经绑定到这个入参上了。由于我们知道他的请求报文格式是xml的,所以可以进行Document转换,然后我们从节点中取值既可以实现请求数据的解析了。

解析的代码请查看我以前的文章: 基于cxf和axis两种框架下的webservice客户端开发文的最后就是这个解析的工具类。

4.application/json

application/json 这个 Content-Type 作为响应头大家肯定不陌生。实际上,现在越来越多的人把它作为请求头,用来告诉服务端消息主体是序列化后的 JSON 字符串。由于 JSON 规范的流行,除了低版本 IE 之外的各大浏览器都原生支持 JSON.stringify,服务端语言也都有处理 JSON 的函数,使用 JSON 不会遇上什么麻烦。

JSON 格式支持比键值对复杂得多的结构化数据,这一点也很有用。

这个请求头还有一个特殊点就是页面通过jquery过来的数据结构和其他应用请求的数据结构不一样,因为jquery对json的数据做了转换:

 "data":{
                "mobilenum":"15867426421",'age':23,'name':'zhangsan'
              }
         ——>mobilenum=15867426421&age=23&name=zhangsan

然而应用没有发生转化,还是json结构的数据。所以获取上会有所不同。这里我们分成页面和应用两种请求来分别说

4.1 页面请求的数据获取

4.1.1 页面js请求如下

数据结构为json,且data为var data={}的json类型时

$.ajax({
type: "post",
data:{'mobilenum':'15867426421','age':'23','name':'zhangsan'},
contentType: "application/json; charset=utf-8",
url :basepath+'jijin/getList',
success: function(data){
                   ... }
});

但是当用浏览器来看是这样的:

我们的数据结构明显发生了变化,由json变成了用&符号连接的普通传值方式。所以这时候获取参数值通过:String name=request.getParameter(“name”)这种来获取。

4.1.2 如果页面js请求如下

数据结构为json,且data为var data=”“的String类型时,controller获取数据则可以和下面的4.2一样通过注解@RequestBody来绑定到入参上。

  $.ajax(
          {
              dataType: "json",
              type: "POST",
              data:"{'mobilenum':'15867426421','age':'23','name':'zhangsan'}",
              contentType: "application/json; charset=utf-8",
              url :basepath+'jijin/getList',
              success: function(data){
              }
           });
    }

这时对应的浏览器的请求头信息如下:

可以看到串到后台的数据结构为json格式的数据结构,和4.1.1完全不同了。就是因为data的数据类型不同导致的,4.1.1的data是json类型,结果转成普通的get方式拼接的参数结构,4.1.2的data是String类型,反而保持住了json类型的数据结构。从而可以实现在controller层通过参数绑定来获取请求参数。

4.1总结

通过上面两个比较可以知道,页面的jquery还是能够识别json数据的,真是由于识别才导致数据转换而。String类型的字符串没有转换的意义,而是直接传给后台,从而保证了数据结构的不变。故:我们真想让后台接受json数据结构的参数话,还是传String类型的吧!

4.2 当别的应用调用我们的服务

发送json数据的请求时如何解析(springmvc controller层)?总的来说通过@RequestBody 注解方法的入参,可以让请求的参数绑定到该参数上,然后用Gson转换成json即可获得jsonObject/实体类(这里的UserBo),具体代码如下

@RequestMapping("/url")
public void getUrlByPhone(@RequestBody String json,HttpServletRequest request){
    Gson gson=new Gson();
    UserBo ub=gson.fromJson(json,UserBo.class);
    //这里还可以通过jackson的ObjectMapper()把String json转成JsonNode,然后用jsonNode.get(fieldName)来获取各个json字段。例如:
    JsonNode node=new ObjectMapper().readTree(json);
    String name=node.get("name");
    ......
}

以上为个人经验,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

  • 详解利用SpringMVC拦截器控制Controller返回值

    背景:需求是在Controller中方法没有实现时,返回模拟结果.主要用于项目初期前台跟后台的交互,Web项目就是在前台发出请求然后后台响应并返回结果.本示例利用拦截器和注解实现跳过执行方法直接返回定义结构的功能. 通过定义一个StringResult注解,在访问方法的时候返回StringResult中的内容.通过Debug注解来定义方法是否要返回StringResult中的内容. Debug默认为TRUE package com.tiamaes.dep.annotation; import j

  • SpringMVC中Controller类数据响应的方法

    目录 1. 方法返回值类型 2. 页面跳转 2.1 直接返回字符串 2.2 返回 ModelAndView 对象 2.3 视图前缀和后缀 2.3 重定向和转发 3. 回写数据 3.1 直接返回字符串 3.2 返回对象或集合 上篇博客我们了解了请求参数的获取,那么获取到请求参数之后,需要对参数进行出来,然后进行数据响应.那么这篇博客我们就来了解 Controller 类如何进行数据响应. 1. 方法返回值类型 在 web 阶段我们也了解过数据响应,我们可以简单的将数据响应分为:页面跳转和回写数据

  • 学习SpringMVC——如何获取请求参数详解

    @RequestParam,你一定见过:@PathVariable,你肯定也知道:@QueryParam,你怎么会不晓得?!还有你熟悉的他(@CookieValue)!她(@ModelAndView)!它(@ModelAttribute)!没错,仅注解这块,spring mvc就为你打开了五彩斑斓的世界.来来来,不要兴(mi)奋(hu),坐下来,我们好好聊聊这么些个注解兄弟们~~~(wait, 都没有听过? 好,来,你坐前排,就你!)  一.spring mvc如何匹配请求路径--"请求路径哪家

  • 如何在springMVC的controller中获取request

    这篇文章主要介绍了如何在springMVC的controller中获取request,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes(); HttpServletRequest request = attributes.getReque

  • 使用springmvc的controller层获取到请求的数据方式

    目录 content-type 1.application/x-www-form-urlencoded 1.1 String 参数值=reqeust.getParameter("参数名"); 1.2 Map<String,String[]> paramMap=request.getParameterMap(); 2.multipart/form-data 3.text/xml 4.application/json 4.1 页面请求的数据获取 4.1.1 页面js请求如下 4

  • 关于SpringMVC在Controller层方法的参数解析详解

    目录 自定义参数解析器 实现效果 实现和配置 Spring提供解析器 一些使用Tricky 总结 使用版本: spring-boot: 2.1.6.RELEASE sping: 5.1.8.RELEASE java: openjdk 11.0.13 自定义参数解析器 为了能够实现自定义参数解析器,只需要实现 HandlerMethodArgumentResolver 并将添加到 WebMvcConfigurer#addArgumentResolvers 即可. 实现效果 下面以添加一个获取请求头

  • Springmvc调用存储过程,并返回存储过程返还的数据方式

    目录 Springmvc调用存储过程,并返回存储过程返还的数据 实现如下 这里要重点说明一下 Springmvc调用存储过程,entity文件写法 Springmvc调用存储过程,并返回存储过程返还的数据 java后端很多时候都需要和数据库进行交互,并返回业务数据.一般情况下都会采用执行SQL的方式来进行交互,但有些特别的场景时,也可以直接利用存储过程返回数据. 存储过程返回数据的好处是只需要一个调用,即可根据不同的参数返回不同的业务数据,这些业务数据有可能列名完全不一样. 实现如下 首先要先定

  • SpringMVC实现controller中获取session的实例代码

    平时使用springMVC,在方法中访问session中经常很自然地调用Servlet API.用起来非常直观方便,一直没有多考虑什么. 比如这样: @RequestMapping(value = "/logout") public String logout(HttpSession session) { session.removeAttribute("user"); return "/login"; } 但毕竟这样对Servlet API产生

  • SpringMVC框架中使用Filter实现请求日志打印方式

    目录 查找资料后确定两种技术方案 1. 使用AOP对所有Controller的方法进行环绕通知处理: 2. 使用Filter拦截所有的Request和Response,并获取body. 最后选择了第二种方式,具体实现记录如下. 具体实现 日志记录过滤器 public class RequestFilter implements Filter{private static final String LOG_FORMATTER_IN = "请求路径:{%s},请求方法:{%s},参数:{%s},来源

  • ssm框架controller层返回json格式数据到页面的实现

    通常前端页面发送ajax请求,我们只需要返回json格式数据就行 1.在pom.xml加入依赖 <properties> <jackson.version>2.5.4</jackson.version> </properties> <dependencies> <!-- controller返回json --> <dependency> <groupId>com.fasterxml.jackson.core&l

  • Spring实现在非controller中获取request对象

    目录 在非controller中获取request对象 首先在web.xml中添加一个监听器 然后在程序中就可以使用 非controller层获取response和request对象 如下所示 在非controller中获取request对象 在Controller中,我们可以直接在参数中加上HttpServletRequest request,springMVC就会自动将request绑定进参数,而如果我们想在其他地方获取request就要如下这么做了. 首先在web.xml中添加一个监听器

  • springboot如何为web层添加统一请求前缀

    目录 如何为web层添加统一请求前缀 配置文件方式 实现WebMvcConfigurer接口 spring web访问页面出现多余前缀和后缀情况 页面中出现hello.jsp 解决方法 如何为web层添加统一请求前缀 配置文件方式 application.properties全局配置文件配置: server.servlet.context-path=/api 实现WebMvcConfigurer接口 重写configurePathMatch()方法,代码: @Configuration publ

  • Springmvc获取前台请求数据过程解析

    1)基本数据类型或String,在方法参数中定义参数,参数名与请求传递数据名一致即可自动封装: // RequestMapping:指定方法对应的请求地址 //return:页面地址,表示方法执行完成之后跳转到对应的页面(转发) //springmvc:接收请求参数,直接在方法的参数中定义名称与传递参数名一致的形参即可 //name:会自动接收请求传递的name值 @RequestMapping("/hello") public String hello(String name,Int

随机推荐