Reputation: 3212
In Spring MVC how do I pass an object between two controller methods? I have an update form and an updateController. In the controller I have 2 methods, one for fetching the data and displaying it in a view. The second method of the controller is invoked when the user clicks update button with modified changes. What I am observing is that the object which I get in second method of the controller is not the same object which I passed to the view in the first controller method call. Its a new object altogether with all form fields mapped to it. How do I make sure that the same object is passed to the second controller method which was provided to the view by the first controller method?
@RequestMapping(value = "/showEmpDetail.html", method = RequestMethod.GET)
public String showEmpDetails(
@RequestParam(value = "page", required = false) Integer page,
HttpServletRequest request, @RequestParam("empId") Long empId,
ModelMap model) {
// Get employee using empId from DB
model.addAttribute("emp",emp);
return "showEmpDetail";
}
The above controller method gets the emp values from Db and displays it correctly in the view. Now the user changes some details and clicks submit button. The following controller method is called.
@RequestMapping(value = "/editEmpFormSubmission.html", method = RequestMethod.POST)
public String editEmpFormSubmission(
@RequestParam(value = "page", required = false) Integer page,
@ModelAttribute("emp") Employee emp, BindingResult result,
ModelMap model, HttpServletRequest request) {
// update changes in DB
}
In the above controller method when I check the emp object its not the same object which I passed in previous controller call. The fields which are not form backed but had values were changed to null. How can I make sure the same object is passed by view. I do not want to add the object as sessionAttribute since a user might modify many employees in a session.
Upvotes: 13
Views: 41610
Reputation: 19
If everything takes place within one controller then you can use:
protected Student studentTmp;
write the object to this field in the first method
public String submitStudentDetails(@Valid Student student) {
studentTmp = student;
}
}
and in the following methods refer to the field and pass it to the view
public String reviewStudentDetails(Model model) {
Student student = studentTmp;
model.addAttribute("student", student);
}
Upvotes: 0
Reputation: 125312
You have 3 options
@SessionAttributes
to store the object in the session in between requests.@ModelAttribute
annotated method to retrieve the object before each request@SessionAttributes
annotation to your controller classSessionStatus
as a parameter to your update method and the setComplete()
method when you are finished with the object@SessionAttributes("emp")
public class EmployeeController {
@RequestMapping(value = "/editEmpFormSubmission.html", method = RequestMethod.POST)
public String editEmpFormSubmission(
@RequestParam(value = "page", required = false) Integer page,
@ModelAttribute("emp") Employee emp, BindingResult result,
ModelMap model, HttpServletRequest request
SessionStatus status) {
// update changes in DB
status.setComplete();
}
}
@ModelAttribute
showEmpDetails
method as it should only return a view namepublic class EmployeeController {
@ModelAttribute("emp")
public Employee getEmployee(@RequestParam("empdId") Long id) {
// Get employee using empId from DB
return emp;
}
@RequestMapping(value = "/showEmpDetail.html", method = RequestMethod.GET)
public String showEmpDetails() {) {
return "showEmpDetail";
}
}
HttpSession
as an argumentshowDetails
method next to adding it to the model add it to the sessioneditEmpFormSubmission
use the one from the session and copy all non-null fields to the object from the session and store that in the database.I wouldn't go for option, I strongly would suggest option 1 especially including the setComplete()
on the SessionStatus
object for cleanup. You could also combine 1 and 2 (have a @ModelAttribute
annotated method and still use @SessionAttributes
.).
Upvotes: 28
Reputation: 4101
How can I make sure the same object is passed by view. I do not want to add the object as sessionAttribute since a user might modify many employees in a session
You could make a field in the object that is filled with a random number at the time of initial render and then store that object in the session. In the view you can map that field with a hidden input and now when user sends a request to edit action you can get that hidden field and fetch the original object from session by the number in the hidden field. That would resolve the multiple edits in different tabs conflict.
Upvotes: 0