Sprig MVC AJAX提交数据整理

标签: SpringMVC @RequestBody Json AJAX


在使用SpringMVC+AJAX提交数据的时候一直提交不成功,发现自己对于SpringMVC AJAX提交数据的认知完全是错误的,重新学习整理。

什么是JSON对象/字符串

JSON对象:

    var jsonObj={
            "loginName":"huanyan",
            "name":"化烟雨",
            "gender":"1",
        }
   

这种格式的叫JSON对象而由"引号括起来的称为JSON字符串

输出数据表现形式的对比
JSON对象

——输出出来的是一个对象,而JSON字符串输出的是字符串形式

JSON字符串/对象之间的转换

使用的是JSONAPI

API 简介
JSON.stringify() JSON对象->JSON字符串
JSON.parse() JSON字符串->JSON对象

SpringMVC+AJAX提交形式

我们先来说说SpringMVC 数据提交的两个注解@RequestParam @RequestBody
对于Http协议,在不指定Content-Type的前提下默认为application/x-www-form-urlencoded
@RequestParam
JSON试例:

    //json对象格式1
    jsonobj={"user":{"departmentId": "1",//省略字段
        "description": "s"}};
     //json对象格式2
     jsonobj={"departmentId": "1",//省略字段
        "description": "s"};
        
    //json字符串格式1
    jsonStr='{"user":{"departmentId": "1",//省略字段
        "description": "s"}}';
     //json字符串格式2
     jsonStr='{"departmentId": "1",//省略字段
        "description": "s"}';
     //以上格式轮流替换在控制台测试
     
    $.ajax({
        type: "GET",//get与post 轮流测试
        url: "/user/test1",test1 or test2
        dataType: "json",
        data:jsonobj,
        success: function (jsonResult) {
            alert(jsonResult);
        }
    });

以下是controller后台代码

    @RequestMapping(value = "/test1")
    public String test1(@RequestParam("user") User user, ModelMap modelMap){
        return "public/message";
    }
    @RequestMapping(value = "/test2")
    public String test2(@RequestParam("departmentId") Long id,@RequestParam("longName") String name, ModelMap modelMap){
        return "public/message";
    }

小结
JSON对象:
对于JSON对象格式1对于test1 test2 都是无效语法错误:HTTP400,而只有JSON对象格式2对于test2可以接收数据。
JSON字符串:
对于JSON字符串格式1 JSON字符串格式2对于test1 test2 都是无效语法错误:HTTP400

@RequestBody

对于测试@RequestBody需要指定Content-Typeapplication/json并且AJAX需要提 交JSON字符串形式 的数据
这里就不多做测试了!懒~

方式一: 数据包装类的形式

$.ajax({
        type: "POST",
        url: "/user/add",
        contentType: "application/json",
        dataType: "json",
        data:JSON.stringify({
    "user":{
        "departmentId":"1",
        "loginName":"s"
    },
    "roles":[
        {
            "id":1
        },
        {
            "id":2
        }
    ]
}),
        success: function (jsonResult) {
            alert(jsonResult);
        }
    });

以下是controller后台代码

    @RequestMapping(value = "/add",method = RequestMethod.POST)
    public String add(@RequestBody UserDto userDto, ModelMap modelMap){
        return "public/message";
    }

以下是包装类后台代码

    public class UserDto {
        private User user;
        private List<Role> roles;
        //以下get/set
    }

方式二: 使用Map<String,Object>的方式
AJAX提交与方式一致
以下是controller后台代码

    @RequestMapping(value = "/add2",method = RequestMethod.POST)
public String add2(@RequestBody Map<String,Object> data, ModelMap modelMap) throws Exception {
    User user= JsonXMLUtils.map2obj((Map<String,Object>) data.get("user"),User.class); //使用工具类转为对象
    List<Role> roles=JsonXMLUtils.map2list(data.get("roles"),Role.class);使用工具类转为集合对象
    return "public/message";
}

以下是工具类 JsonXMLUtils后台代码

package com.oa.until;

import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.JSONArray;

import java.util.List;
import java.util.Map;

public class JsonXMLUtils {
    /**
     * 对象->JSON字符串
     * @param obj
     * @return
     * @throws Exception
     */
    public static String obj2json(Object obj) throws Exception {
        return JSON.toJSONString(obj);
    }

    /**
     * json字符串->对象
     * @param jsonStr
     * @param clazz
     * @param <T>
     * @return
     * @throws Exception
     */
    public static <T> T json2obj(String jsonStr, Class<T> clazz) throws Exception {
        return JSON.parseObject(jsonStr, clazz);
    }

    /**
     * JSON字符串->map
     * @param jsonStr
     * @param <T>
     * @return
     * @throws Exception
     */
    public static <T> Map<String, Object> json2map(String jsonStr)     throws Exception {
        return JSON.parseObject(jsonStr, Map.class);
    }

    /**
     * map数据参数->对象
     * @param map
     * @param clazz
     * @param <T>
     * @return
     * @throws Exception
     */
    public static <T> T map2obj(Map<?, ?> map, Class<T> clazz) throws Exception {
        return JSON.parseObject(JSON.toJSONString(map), clazz);
    }

    /**
     * map数组行参数->对象集合
     * @param object
     * @param clazz
     * @param <T>
     * @return
     * @throws Exception
     */
    public static <T> List<T> map2list(Object object, Class<T> clazz) throws Exception {
        return JSONArray.parseArray(JSON.toJSONString(object),clazz);
    }
}

方式三: bean对象接收形式
这种形式与包装类的JSON字符串形式是差别的。注意JSON字符串格式

$.ajax({
        type: "POST",
        url: "/user/add1",
        contentType: "application/json",
        dataType: "json",
        data:JSON.stringify({
    "departmentId":"1",
    "loginName":"s",
    "roles":[
        {
            "id":1
        },
        {
            "id":2
        }
    ]
}),
        success: function (jsonResult) {
            alert(jsonResult);
        }
    });

以下是controller后台代码

   @RequestMapping(value = "/add1",method = RequestMethod.POST)
    public String add1(@RequestBody User user, ModelMap modelMap){
        return "public/message";
    }

什么时候使用 @RequestParam @RequestBody
请求提交方式:
GET:会将参数以url?KEY=VALUE&KEY=VALUE的形式请求到服务器。
POST:

  • Content-Type为默认x-www-form-urlencoded
    在AJAX请求时会将参数转为?KEY=VALUE&....的参数形式封装到HTTP BODY请求到服务器,而数据不会显示到URL上。
  • Content-Type指定为application/json
    使用@RequestBody请求方式必须为POST,不然会报HTTP 400错误

GET/POST形式下:
@RequestParam:可以处理Content-Type的类型

application/x-www-form-urlencoded
multipart/form-data

@RequestBody:可以处理Content-Type的类型

application/json
application/xml

注:

在jsp中使用ajax提交时出现了下面的问题: 415 (Unsupported Media Type), 可能出现的原因如下:

  1. dataParam 格式错误,不符合json字符串规范
  2. dataParam 参数不能绑定到控制器方法的参数上面,不是一一对应
  3. 没有在ajax提交时设置contentType
    contentType指的是前台传递到后台的格式,dataType指的是由后台传到前台的格式
  4. jackson的jar包不对或者不全:
    不应该使用org.codehaus.jackson下面的jackson.jar, 应该使用com.fasterxml.jackson.core下面的>jackson->core和jackson-databind 的jar包
  5. 加上<mvc annotation-driven>
    6.springMVC配置文件没配好
    参考文档-3

参考文档
SpringMVC接受JSON参数详解及常见错误总结
@RequestBody, @ResponseBody 注解详解(转)
@requestBody @responseBody配置要点

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

推荐阅读更多精彩内容