Spring 3 AJAX POST请求与@RequestBody以及@ModelAttribute和@SessionAttribute一起使用?

时间:2013-05-10 01:19:24

标签: java ajax json spring-mvc controller

拥有一个Java spring MVC Web应用程序,并且正在发出一个jquery ajax post请求。我的控制器设置为接收和发送json数据。一切正常,JSON字符串格式良好,Controller可以创建并填充Command对象,并使用JSON请求数据的内容填充它。但是,我正在更新Contact对象的数据,而我的JSP表单元素仅包含数据库更新所需的所有数据的子集。在我对具有表单的JSP页面的初始GET请求中,我从数据库中检索所有必需的数据,填充Contact Command对象,然后将该命令对象绑定到Model。

如果我正在进行正常的POST提交表单提交,我相信只需将我的命令对象声明为@SessionAttribute,并在我的onSubmit()POST方法中使用@ModelAttribute引用该Command对象就足够了。 Spring将从我的会话中检索已填充的命令对象,然后绑定(覆盖)由于POST请求而更改的那些值。然后,可以将此更新的命令对象用作数据库更新的参数。

但是,我正在使用Spring 3并利用@RequestBody参数类型。我无法让Spring给我会话对象并自动绑定请求中的新值。它或者只提供旧的会话命令对象(不应用更改)或仅提供POST请求中的值的新命令对象。

这是一个小代码 - 不起作用:

@SessionAttributes("contactCommand")
@Controller
public class ContactController {


  @RequestMapping(value = "/editContact", method=RequestMethod.GET)
public String init(ModelMap model, Locale locale, HttpServletRequest request, HttpServletResponse response) throws GeneralException {
    final ContactCommand cmd = new ContactCommand();
    // populate with data from DB etc
    model.addAttribute("contactCommand", cmd);
    // etc
}

@RequestMapping(value="/editContact",method=RequestMethod.POST, consumes = "application/json", produces = "application/json")
public @ResponseBody Map<String, ? extends Object> editContactInfo(@RequestBody @ModelAttribute("contactCommand") ContactCommand cmd, HttpServletRequest request, HttpServletResponse response) throws GeneralException {

// do business logic with command object here

}

任何人都可以告诉我使用@RequestBody与JSON请求数据的“标准”或“最简单”方法是什么,并将其绑定到现有的/ @ModelAttribute填充的Command对象,以便Command对象完全由旧的构成和新数据(使用完整的POST http提交很容易实现)。

相关的问题是上面的代码有什么问题? @SessionAttribute和带有JSON内容的@RequestBody可以一起使用吗?如果是这样,请解释如何!非常感谢您的任何意见。

我的工作是让Spring创建新的Command对象并自动填充表单数据。然后从会话中手动单独调用/检索旧命令对象,最后手动将表单提交中不存在的所有属性复制到新命令对象中。现在,我将所有必要的数据放在一个命令对象中,以应用我的SQL更新。必须有一个更简单的方法......;)

更新:

今天发现这篇SOF帖子,同时进一步研究这个问题:

Spring Partial Update Object Data Binding

似乎没有开箱即用的SPRING解决方案,但需要了解处理它的最佳方法。在我的情况下,是的,我使用的是嵌套域对象,因此帖子中提供的解决方法并不好。有没有人有任何其他想法?为了清楚起见,我希望将JSON格式数据发布到Controller(而不仅仅是http表单发布数据)。

好的,我已经为这个开了一个Spring Source JIRA请求,也许这是一个非常需要的改进:

https://jira.springsource.org/browse/SPR-10552

或者,这是一个利用杰克逊转换功能的巧妙方式,听起来像很多管道。

2 个答案:

答案 0 :(得分:1)

这不是一个完整的答案,但我希望它会指出你正确的方向。

以下是我们用来使用Jackson从JSON到现有对象进行深度绑定的类。这是根据Jackson的错误报告改编的:https://jira.springsource.org/browse/SPR-10552

public class JsonBinder
{
    private ObjectMapper objectMapper;

    public JsonBinder( ObjectMapper objectMapper )
    {
        super();
        this.objectMapper = checkNotNull( objectMapper );
    }

    public void bind( Object objToBindInto, InputStream jsonStream ) throws JsonProcessingException, IOException
    {
        JsonNode root = objectMapper.readTree( checkNotNull( jsonStream ) );
        applyRecursively( checkNotNull( objToBindInto ), root );
    }

    private void applyRecursively( Object objToBindInto, JsonNode node ) throws JsonProcessingException, IOException
    {
        PropertyAccessor propAccessor = null;

        for( Iterator<Entry<String, JsonNode>> i = node.fields(); i.hasNext(); )
        {
            Entry<String, JsonNode> fieldEntry = i.next();
            JsonNode child = fieldEntry.getValue();
            if( child.isArray() )
            {
                // We ignore arrays so they get instantiated fresh every time
                // root.remove(fieldEntry.getKey());
            }
            else
            {
                if( child.isObject() )
                {
                    if( propAccessor == null )
                    {
                        propAccessor = PropertyAccessorFactory.forDirectFieldAccess( objToBindInto );
                    }
                    Object o2 = propAccessor.getPropertyValue( fieldEntry.getKey() );
                    if( o2 != null )
                    {

                        // Only remove the JsonNode if the object already exists
                        // Otherwise it will be instantiated when the parent gets
                        // deserialized
                        i.remove();
                        applyRecursively( o2, child );
                    }
                }
            }
        }
        ObjectReader jsonReader = objectMapper.readerForUpdating( objToBindInto );
        jsonReader.readValue( node );
    }
}

我们将它与Spring的HandlerMethodArgumentResolver实现一起使用。

我们没有使用很多Spring的MVC框架。我们只是使用Spring的许多不同部分构建JSON API后端。这是一个非常好的管道,让它全部工作,但现在我们的控制器非常简单。

不幸的是,我无法显示所有代码,反正它很长。我希望这至少解决了部分问题。

答案 1 :(得分:0)

为什么要使用@RequestBody注释ModelAttribute,只要拥有@SessionAttribute并使用@ModelAttribute引用该Command对象就足够了。

使用@RequestBody

背后的动机是什么?

请参阅http://static.springsource.org/spring/docs/3.0.x/javadoc-api/org/springframework/web/bind/annotation/ModelAttribute.html

http://static.springsource.org/spring/docs/3.0.x/javadoc-api/org/springframework/web/bind/annotation/RequestBody.html