SpringMVC模型属性在POST请求中返回null

3
我正在开发一个SpringMVC 3的小应用程序,但我卡在某个地方了。实际上,在GET操作中填充字段的模型属性在POST中返回为NULL(即使我没有对它们进行任何操作)。我已经在这个和其他论坛上检查过了,唯一的答案是为应该放入模型的类实现编辑器,一个初始化器可以用来注册自定义编辑器,并使其在应用程序中可用(在servletname-servlet.xml文件中)。我做的所有操作,但绝对没有运气。我想知道是否有人能帮我一下。 谢谢。
以下是控制器:
@Controller
@RequestMapping(value="/nourish")
public class NourishController {

private PetDAO pdao = new PetDAO();
private UserDAO udao = new UserDAO();
private FeedVirtualPet feedvp = new FeedVirtualPet();

@RequestMapping(method = RequestMethod.GET)
public String nourish(Model model, HttpServletRequest request){
    NourishPetDTO npdto = new NourishPetDTO();
    PetDTO      pdto=pdao.findPetByBusinessKey((PetDTO)request.getSession().getAttribute("pet"));
    npdto.setPet(pdto);
    npdto.setAmt(0);
    model.addAttribute("npdto", npdto);
    return "nourish";
}


@RequestMapping(method = RequestMethod.POST)
public String nourishPOST(@ModelAttribute("npdto") NourishPetDTO npdto,
        //BindingResult result,
        HttpServletRequest request){

    System.out.println("*****nourishPOST.npdto.amt: "+npdto.getAmt());
    System.out.println("*****nourishPOST.npdto.pet.petname: "+npdto.getPet().getPetName());
    System.out.println("*****nourishPOST.npdto.pet.hunger:     "+npdto.getPet().getHunger());
    PetDTO pdto = feedvp.feed(npdto.getPet(), npdto.getAmt());
    request.getSession().setAttribute("user", pdto.getOwner());
    return "redirect:detailPet";
}
}

具有GET和POST操作方法,并与以下jsp相关联 - 在此视图中,所有模型信息都通过EL正确显示:
<%@ page language="java" contentType="text/html; charset=ISO-8859-1" session="true"  pageEncoding="ISO-8859-1"%>
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
<%@ taglib prefix="form" uri="http://www.springframework.org/tags/form" %>
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=ISO-8859-1">
<title>Nourish your pet!!</title>
</head>
<body>
Your stats for <h3>${npdto.pet.petName}</h3><br>
    Please note that any value you put inside will:
        <ol>
            <li>Subtract value to your current hunger level</li>
            <li>Add (value) to your current health level</li>
        </ol>
    Please note that any value you'll put will in many manners "resized":
    <ol>
        <li>It must be even. If not, a default 0 value will be  applied</li>
        <li>It can't be greater than 4. If it's greater, the maxium value  of 4 will be anyway considered.</li>
        <li>If it ain't a number, a default zero value will be passed</li>
    </ol>
<table>
    <tr><td>Health</td><td>${npdto.pet.health}</td></tr>
    <tr><td>Hunger</td><td>${npdto.pet.hunger}</td></tr>
</table>
<form action="nourish" method="post"  >
    nourishment: <input type="text" name="amt"/>
    <input type="submit" value="Nourish!"/>
</form>
</body>
</html>

请注意,我没有使用返回 NULL 的 model 属性,以确保我没有对其进行任何操作。
POST 操作在指令上失败。
System.out.println("*****nourishPOST.npdto.pet.petname:"+npdto.getPet().getPetName());

由于Tomcat返回了NullPointerException。
如前所述,我一直在寻找解决这个问题的方法,但我能找到的所有方法都是添加编辑器类并将编辑器注册到绑定器中。结果仍然相同。
无论如何,以下是这些类:
NourishPetEditor.java
public class NourishPetEditor extends PropertyEditorSupport {

private PetEditor pedit;

public PetEditor getPedit() {
    return pedit;
}


public NourishPetEditor() {
    // TODO Auto-generated constructor stub
    pedit =  new PetEditor();
}

@Override 
public String getAsText(){
    NourishPetDTO npdto= (NourishPetDTO)getValue();
    return super.getAsText()+","+npdto.getAmt();
}

public NourishPetDTO makeNourishPetDTOInstance(String [] parts){

    NourishPetDTO npdto = new NourishPetDTO(); 
    npdto.setPet(pedit.makePetDTOInstance(parts));
    npdto.setAmt(Integer.parseInt(parts[9]));
    return npdto;

}


public void setAsText(String key){
    String []parts = key.split(",");
    NourishPetDTO npdto = makeNourishPetDTOInstance(parts);
    setValue(npdto);
}
}

PetEditor.java

package com.virtualpet.dtoeditors;
import java.beans.PropertyEditorSupport;

import com.virtualpet.virtualpet_daos.PetDAO;  
import com.virtualpet.virtualpet_dtos.PetDTO;
import com.virtualpet.virtualpet_dtos.UserDTO;
public class PetEditor extends PropertyEditorSupport{

private PetDAO pdao;

public PetEditor() {
    // TODO Auto-generated constructor stub
}

public PetEditor(PetDAO pdao) {
    // TODO Auto-generated constructor stub
    this.pdao = pdao;

}


public String getAsText(){
    PetDTO pdto = (PetDTO) this.getValue();

    return pdto.getClass().getName()+","+ //0
    pdto.getPetName()+","+ //1
    pdto.getHealth()+","+  //2
    pdto.getHunger()+","+  //3
    pdto.getMood()+","+","+ //4
    pdto.getOwner().getClass().getName()+","+ //5
    pdto.getOwner().getUsername()+","+ //6
    pdto.getOwner().getEmail()+","+pdto.getOwner().getPassword(); //7,8

}

public void setAsText(String key) throws IllegalArgumentException {
    String []parts = key.split(",");
    PetDTO pdto = makePetDTOInstance(parts);
    setValue(pdto);
}


public UserDTO makeUserDTOInstance(String[] parts)
        throws InstantiationException, IllegalAccessException,
        ClassNotFoundException {

            UserDTO udto = (UserDTO)Class.forName(parts[5]).newInstance();
            udto.setUsername(parts[6]);
            udto.setEmail(parts[7]);
            udto.setPassword(parts[8]);
            return udto;
}


public PetDTO makePetDTOInstance(String[]parts){
    try{
        PetDTO pdto = (PetDTO) Class.forName(parts[0]).newInstance();
        pdto.setPetName(parts[1]);
        pdto.setHealth(Integer.parseInt(parts[2]));
        pdto.setHunger(Integer.parseInt(parts[3]));
        pdto.setMood(Integer.parseInt(parts[4]));

        UserDTO udto = makeUserDTOInstance(parts);

        pdto.setOwner(udto);
        return pdto;
    }
    catch (Exception e){
        throw new IllegalArgumentException();
    }
}
}

我来翻译一下:

我会跳过 UserEditor,因为它与 PetEditor 非常相似。 最后,是初始化器,用于绑定自定义编辑器和模型类。

VirtualPetDTOInitializer.java

package com.virtualpet.dtoeditors;

import org.springframework.web.bind.WebDataBinder;
import org.springframework.web.bind.support.WebBindingInitializer;
import org.springframework.web.context.request.WebRequest;

import com.virtualpet.virtualpet_dtos.NourishPetDTO;
import com.virtualpet.virtualpet_dtos.PetDTO;
import com.virtualpet.virtualpet_dtos.UserDTO;

public class VirtualPetDTOInitializer implements WebBindingInitializer {

public void initBinder(WebDataBinder binder, WebRequest arg1) {
    // TODO Auto-generated method stub
        binder.registerCustomEditor(UserDTO.class, new UserEditor( ));
        binder.registerCustomEditor(PetDTO.class, new PetEditor( ));
        binder.registerCustomEditor(NourishPetDTO.class, new NourishPetEditor());
}
}

这个类在dispatcher-servlet.xml中定义了一个属性值:
<?xml version="1.0" encoding="UTF-8"?>
<beans ...>
<bean id="userDao"
    class="com.virtualpet.virtualpet_daos.UserDAO"/>
<bean id="petDao"
    class="com.virtualpet.virtualpet_daos.PetDAO" />
<bean    class="org.springframwork.web.servlet.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter">
    <property name="webBindingInitializer">
        <bean class="com.virtualpet.dtoeditors.VirtualPetDTOInitializer"/>
    </property>
</bean>
</beans>

作为完全的Spring MVC新手,我必须告诉你,这个错误是在我实现这些类之前就出现的。所以看起来它们不是问题的因素,但是它们的实现是我能找到的关于模型属性在POST后返回null的一切。 请问有人可以帮忙吗?提前感谢。

如果您从“POST”重定向到“/detailPet”,请确保您有一个映射。 - blurfus
已经有一个名为DetailPetController的控制器,我没有贴出来以保持可读性(同时,我对那个控制器也没有问题)。非常感谢您的时间。 - Luigi Iaderosa
1个回答

2
您需要执行以下操作之一:
  1. npdto的值存储在隐藏的表单字段中

  2. npdto存储在会话中

  3. 在您的post handler中重新从数据库中读取npdto

您可能需要选择第二个选项,在这种情况下,请在您的控制器顶部添加@SessionAttributes("npdto")
您还应该在您的post handler中添加一个SessionStatus参数,并调用sessionStatus.complete()来清除不再需要的会话项。
有关参考答案,请参见Spring MVC:验证,Post-Redirect-Get,部分更新,乐观并发性,字段安全

谢谢!那正是我所缺少的。但是,有些事情仍然困扰着我...如果我必须将变量或对象添加到SessionAttribute中,那么将其添加到Model中有何好处呢?我正在遵循Spring Recipe手册,如果你想进行“动手实践”,那么这很好,但是许多事情都被省略了,这就是其中之一...无论如何我至少能够继续前进。 - Luigi Iaderosa
1
@LuigiIaderosa Model只是您的视图为当前响应所需的数据。它不关心下一个请求。这就是session的作用。Model对于保持视图和数据分离非常有用。 - Neil McGuigan

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接