在《Spring MVC的常用注解(学习笔记)》中提到了使用@ModelAttribute
注解,利用Model和ModelMap来存储数据模型。例如下面的代码表示将user对象添加到model的“user”属性中。
@ModelAttribute
public void userModel(String loginname,String password,Model model){
//创建User对象存储jsp页面传入的参数
...
//将USer对象添加到Model当中
model.addAttribute("user",user);
}
然后就可以在JSP页面通过requestScope访问得到。
form标签
而这里要介绍的form标签同样需要利用到Model,它可以自动绑定Model中的一个属性值到当前form对应的实体对象,默认为command属性。form标签虽然是JSP页面的标签,但是这里强大的是和Controller中的变量
交互的功能。
@Controller
public class UserController{
@RequestMapping(value="/registerForm",method=RequestMethod.GET)
public String registerForm(Model model) {
User user = new User("jack","男",28);
// model中添加属性command,值是user对象
model.addAttribute("command",user);
return "registerForm";
}
}
在JSP页面中就能够通过form标签获取到具体的值。
<%@ taglib prefix="form" uri="http://www.springframework.org/tags/form" %>
<form:form method="post" action="...">
...
<form:input path="username">
...
</form:form>
假设User类中定义的属性为username,那么通过<form:input path="username">
就能够获取到对应的值,也就是“jacky”。如果不想使用默认的Model的command属性,而是想使用其他的属性时,可以使用form标签的modelAttribute
属性或者commandName
属性进行指定,这里以modelAttribute
属性为例。
<form:form modelAttribute="user" mothod="post" action="...">
这么多的标签,实现的步骤都有些一致。input中文本框的默认显示值、checkbox、radiobutton、下拉菜单的默认选择值现在都可以不在Html中直接指定,而是通过Model的属性来指定。一般而言,在Model中设置两个属性,一个用于存放需要勾选的选项,另外一个用于存放需要显示出来的所有可选值。而input标签不需要显示多余的选项。
在input标签中,使用modelAttribute属性指向Model的“user”属性,从而获取到一个user对象,在input页面直接可以访问:这样就会显示默认的username的值。
<form:input path="username">
checkbox标签等的实现
而其它需要展示多余选项的如何操作呢。下面只举checkbox为例。
思路一:
以checkbox标签为例,第一种思路是直接在jsp页面中将所有的可选项列出来。
<form:form modelAttribute="user" mothod="post" action="...">
...
<form:checkbox path="courses" value="JAVAEE" label="JAVAEE"/>
<form:checkbox path="courses" value="Mybatis" label="Mybatis"/>
<form:checkbox path="courses" value="Spring" label="Spring"/>
首先创建一个User类用于存放要展示出来的数据。User类有reader
和courses
两个属性。reader是boolean类型变量,设置为true的话,在页面中的checkbox复选框就会被选中。而courses
是一个List<String>
类型变量。所以在Controller中创建一个user对象,然后将reader属性设置为true,并且将“JavaEE”添加到一个list中,然后将该list
传递给courses
属性,这样的话JSP页面中JAVAEE对应的选项就会被选中。
思路二:
第二种思路是不在JSP页面中将可选项列出来,无论是可选项还是被选中项全部都在Controller中进行实现。这样的话,JSP页面中上面的代码就变成:
<form:checkbox items="${courseList}" path="coureses" >
也就是说原来<form:checkbox>
标签中的value属性和label属性都消失了。但是多了一个items属性。这时候的User类只有一个courses属性,是一个List<String>类型的变量。也就是说User类可以保存一个List
对象到其courses
属性之中(这跟思路一的操作差别不大)。
在Controller中新建一个user对象,然后新建一个List(并将“JAVAEE”添加到该list中),到这为止跟第一种思路都是基本一样的操作。但是在这里,还需要另外创建一个List,用于保存所有的可选项。
List<String> courseList = new ArrayList<String>();
courseList.add("JAVAEE");
courseList.add("Mybatis");
courseList.add("Spring");
model.addAttribute("user",user);
model.addAttribute("courseList",courseList);
这里model需要设置两个属性。一个用于设置勾选项
,一个用于设置所有可选项
。
<form:checkbox items="${courseList}" path="coureses" >
然后就可以通过这样在JSP页面进行显示了。如果希望呈现出来的Lable和Value值不同,那么就可以创建一个Map来替代上面的List,这时候Map的value作为Label进行显示,而key则是真正的复选框的value。例如:
Map<String,String> courseMap = new HashMap<String ,String>();
courseMap.put("1","JAVAEE");
courseMap.put("2","Mybatis")
courseMap.put("3","Spring");
并且这时候user对象添加的List中,添加的不是Map的值,而是key的值(例如原来要添加“JAVAEE”到list中,然后把list保存到user对象中,现在则是添加“1”到list中,然后把list保存到user对象中)。
这还没完,还有一种情况是需要添加对象的。也是使用的思路二中List的原理,不需要使用Map,但是由于需要创建对象,所以显得繁琐一些。同样,这时候Model设置两个属性,一个用于指向对象(存储了勾选项
),一个用于指向列表(存储了所有可选项
),这跟之前的还是一样,不同的地方在于指向的列表是一个对象列表
。
一个Employee可能具有多个不同的部门。所以Employee有一个depts属性,它是一个List<Dept>
类型的变量,用于记录属于哪些部门。
model.addAttribute("employee",employee);
model.addAttribute("deptList",deptList);
其实,我们所选择的可选项都是同属于某一个对象的属性,可能是一个属性中的多个列表元素,也可能是多个属性。例如(username,sex,age),再比如(JAVAEE,Mybatis,Spring),前者是user这个对象的多个属性
,后者都是User的某一属性
(courses)的列表元素
。而这里要显示出来的各个部门都存在于Employee这个类的depts属性中。depts是一个List<dept>类型的属性。所以也是某一属性的列表元素。
也正是因为这里depts是一个List<dept>
而非List<String>
类型的变量,所以在该例子中的操作会显得稍微繁琐一些。需要创建对象,并且将对象保存到List中传递给Employee的depts属性。而如果是List<String>类型,那么就可以略去创建对象的步骤,直接将字符串
添加到列表继而传递给depts属性。