浅谈SpringMVC数据绑定

运维 系统运维
查看spring源码可以看出spring支持转换的数据类型: org.springframework.beans.PropertyEditorRegistrySupport:

查看spring源码可以看出spring支持转换的数据类型:
org.springframework.beans.PropertyEditorRegistrySupport:

 

1

 

2

 

3

 

4

 

5

 

6

 

7

 

8

 

9

 

10

 

11

 

12

 

13

 

14

 

15

 

16

 

17

 

18

 

19

 

20

 

21

 

22

 

23

 

24

 

25

 

26

 

27

 

28

 

29

 

30

 

31

 

32

 

33

 

34

 

35

 

36

 

37

 

38

 

39

 

40

 

41

 

42

 

43

 

44

 

45

 

46

 

47

 

48

 

49

 

50

 

51

 

52

 

53

 

54

 

55

 

56

 

57

 

58

 

59

 

60

 

61

 

62

 

63

 

64

 

65

 

66

 

67

 

68

 

69

 

70

 

 

/**

 

 * Actually register the default editors for this registry instance.

 

 */

 

private void createDefaultEditors() {

 

    this.defaultEditors = new HashMap<Class, PropertyEditor>(64);

 

 

 

    // Simple editors, without parameterization capabilities.

 

    // The JDK does not contain a default editor for any of these target types.

 

    this.defaultEditors.put(Charset.class, new CharsetEditor());

 

    this.defaultEditors.put(Class.class, new ClassEditor());

 

    this.defaultEditors.put(Class[].class, new ClassArrayEditor());

 

    this.defaultEditors.put(Currency.class, new CurrencyEditor());

 

    this.defaultEditors.put(File.class, new FileEditor());

 

    this.defaultEditors.put(InputStream.class, new InputStreamEditor());

 

    this.defaultEditors.put(InputSource.class, new InputSourceEditor());

 

    this.defaultEditors.put(Locale.class, new LocaleEditor());

 

    this.defaultEditors.put(Pattern.class, new PatternEditor());

 

    this.defaultEditors.put(Properties.class, new PropertiesEditor());

 

    this.defaultEditors.put(Resource[].class, new ResourceArrayPropertyEditor());

 

    this.defaultEditors.put(TimeZone.class, new TimeZoneEditor());

 

    this.defaultEditors.put(URI.class, new URIEditor());

 

    this.defaultEditors.put(URL.class, new URLEditor());

 

    this.defaultEditors.put(UUID.class, new UUIDEditor());

 

 

 

    // Default instances of collection editors.

 

    // Can be overridden by registering custom instances of those as custom editors.

 

    this.defaultEditors.put(Collection.class, new CustomCollectionEditor(Collection.class));

 

    this.defaultEditors.put(Set.class, new CustomCollectionEditor(Set.class));

 

    this.defaultEditors.put(SortedSet.class, new CustomCollectionEditor(SortedSet.class));

 

    this.defaultEditors.put(List.class, new CustomCollectionEditor(List.class));

 

    this.defaultEditors.put(SortedMap.class, new CustomMapEditor(SortedMap.class));

 

 

 

    // Default editors for primitive arrays.

 

    this.defaultEditors.put(byte[].class, new ByteArrayPropertyEditor());

 

    this.defaultEditors.put(char[].class, new CharArrayPropertyEditor());

 

 

 

    // The JDK does not contain a default editor for char!

 

    this.defaultEditors.put(char.class, new CharacterEditor(false));

 

    this.defaultEditors.put(Character.class, new CharacterEditor(true));

 

 

 

    // Spring's CustomBooleanEditor accepts more flag values than the JDK's default editor.

 

    this.defaultEditors.put(boolean.class, new CustomBooleanEditor(false));

 

    this.defaultEditors.put(Boolean.class, new CustomBooleanEditor(true));

 

 

 

    // The JDK does not contain default editors for number wrapper types!

 

    // Override JDK primitive number editors with our own CustomNumberEditor.

 

    this.defaultEditors.put(byte.class, new CustomNumberEditor(Byte.class, false));

 

    this.defaultEditors.put(Byte.class, new CustomNumberEditor(Byte.class, true));

 

    this.defaultEditors.put(short.class, new CustomNumberEditor(Short.class, false));

 

    this.defaultEditors.put(Short.class, new CustomNumberEditor(Short.class, true));

 

    this.defaultEditors.put(int.class, new CustomNumberEditor(Integer.class, false));

 

    this.defaultEditors.put(Integer.class, new CustomNumberEditor(Integer.class, true));

 

    this.defaultEditors.put(long.class, new CustomNumberEditor(Long.class, false));

 

    this.defaultEditors.put(Long.class, new CustomNumberEditor(Long.class, true));

 

    this.defaultEditors.put(float.class, new CustomNumberEditor(Float.class, false));

 

    this.defaultEditors.put(Float.class, new CustomNumberEditor(Float.class, true));

 

    this.defaultEditors.put(double.class, new CustomNumberEditor(Double.class, false));

 

    this.defaultEditors.put(Double.class, new CustomNumberEditor(Double.class, true));

 

    this.defaultEditors.put(BigDecimal.class, new CustomNumberEditor(BigDecimal.class, true));

 

    this.defaultEditors.put(BigInteger.class, new CustomNumberEditor(BigInteger.class, true));

 

 

 

    // Only register config value editors if explicitly requested.

 

    if (this.configValueEditorsActive) {

 

        StringArrayPropertyEditor sae = new StringArrayPropertyEditor();

 

        this.defaultEditors.put(String[].class, sae);

 

        this.defaultEditors.put(short[].class, sae);

 

        this.defaultEditors.put(int[].class, sae);

 

        this.defaultEditors.put(long[].class, sae);

 

    }

 

}

 

 

 

 



下面挑选一些常用的数据类型,举例说明它们的绑定方式

1. 基本数据类型(以int为例,其他类似):
    Controller代码:

 

1

 

2

 

3

 

4

 

 

@RequestMapping("test.do")

 

public void test(int num) {

 

     

 

}

 

 

 

 


    JSP表单代码:

 

1

 

2

 

3

 

4

 

 

<form action="test.do" method="post">

 

   <input name="num" value="10" type="text"/>

 

   ......

 

</form>

 

 

 

 


表单中input的name值和Controller的参数变量名保持一致,就能完成基本数据类型的数据绑定,如果不一致可以使用@RequestParam标注实现。值得一提的是,如果Controller方法参数中定义的是基本数据类型,但是从jsp提交过来的数据为null或者""的话,会出现数据转换的异常。也就是说,必须保证表单传递过来的数据不能为null或"",所以,在开发过程中,对可能为空的数据,***将参数数据类型定义成包装类型,具体参见下面的第二条。

2. 包装类型(以Integer为例,其他类似):
    Controller代码:

 

1

 

2

 

3

 

4

 

 

@RequestMapping("test.do")

 

public void test(Integer num) {

 

     

 

}

 

 

 

 


    JSP表单代码:

 

1

 

2

 

3

 

4

 

 

<form action="test.do" method="post">

 

   <input name="num" value="10" type="text"/>

 

   ......

 

</form>

 

 

 

 


和基本数据类型基本一样,不同之处在于,JSP表单传递过来的数据可以为null或"",以上面代码为例,如果jsp中num为""或者表单中无num这个input,那么,Controller方法参数中的num值则为null。

3. 自定义对象类型:
    Model代码:

 

1

 

2

 

3

 

4

 

5

 

6

 

7

 

8

 

9

 

10

 

11

 

12

 

13

 

14

 

15

 

16

 

17

 

18

 

19

 

20

 

21

 

22

 

23

 

 

public class User {

 

 

 

    private String firstName;

 

 

 

    private String lastName;

 

 

 

    public String getFirstName() {

 

        return firstName;

 

    }

 

 

 

    public void setFirstName(String firstName) {

 

        this.firstName = firstName;

 

    }

 

 

 

    public String getLastName() {

 

        return lastName;

 

    }

 

 

 

    public void setLastName(String lastName) {

 

        this.lastName = lastName;

 

    }

 

 

 

}

 

 

 

 


    Controller代码:

 

1

 

2

 

3

 

4

 

 

@RequestMapping("test.do")

 

public void test(User user) {

 

     

 

}

 

 

 

 


    JSP表单代码:

 

1

 

2

 

3

 

4

 

5

 

 

<form action="test.do" method="post">

 

   <input name="firstName" value="张" type="text"/>

 

   <input name="lastName" value="三" type="text"/>

 

   ......

 

</form>

 

 

 

 


非常简单,只需将对象的属性名和input的name值一一对应即可。

4. 自定义复合对象类型:
    Model代码:

 

1

 

2

 

3

 

4

 

5

 

6

 

7

 

8

 

9

 

10

 

11

 

12

 

13

 

14

 

15

 

16

 

17

 

18

 

19

 

20

 

21

 

22

 

23

 

24

 

25

 

26

 

27

 

28

 

29

 

30

 

31

 

32

 

33

 

34

 

35

 

36

 

37

 

38

 

39

 

40

 

41

 

42

 

43

 

44

 

45

 

46

 

47

 

48

 

49

 

50

 

51

 

52

 

53

 

54

 

55

 

56

 

57

 

 

public class ContactInfo {

 

 

 

    private String tel;

 

 

 

    private String address;

 

 

 

    public String getTel() {

 

        return tel;

 

    }

 

 

 

    public void setTel(String tel) {

 

        this.tel = tel;

 

    }

 

 

 

    public String getAddress() {

 

        return address;

 

    }

 

 

 

    public void setAddress(String address) {

 

        this.address = address;

 

    }

 

 

 

}

 

 

 

public class User {

 

 

 

    private String firstName;

 

 

 

    private String lastName;

 

 

 

    private ContactInfo contactInfo;

 

 

 

    public String getFirstName() {

 

        return firstName;

 

    }

 

 

 

    public void setFirstName(String firstName) {

 

        this.firstName = firstName;

 

    }

 

 

 

    public String getLastName() {

 

        return lastName;

 

    }

 

 

 

    public void setLastName(String lastName) {

 

        this.lastName = lastName;

 

    }

 

 

 

    public ContactInfo getContactInfo() {

 

        return contactInfo;

 

    }

 

 

 

    public void setContactInfo(ContactInfo contactInfo) {

 

        this.contactInfo = contactInfo;

 

    }

 

 

 

}

 

 

 

 


    Controller代码:

 

1

 

2

 

3

 

4

 

5

 

6

 

7

 

 

@RequestMapping("test.do")

 

public void test(User user) {

 

    System.out.println(user.getFirstName());

 

    System.out.println(user.getLastName());

 

    System.out.println(user.getContactInfo().getTel());

 

    System.out.println(user.getContactInfo().getAddress());

 

}

 

 

 

 


    JSP表单代码:

 

1

 

2

 

3

 

4

 

5

 

6

 

7

 

 

<form action="test.do" method="post">

 

   <input name="firstName" value="张" /><br>

 

   <input name="lastName" value="三" /><br>

 

   <input name="contactInfo.tel" value="13809908909" /><br>

 

   <input name="contactInfo.address" value="北京海淀" /><br>

 

   <input type="submit" value="Save" />

 

</form>

 

 

 

 


User对象中有ContactInfo属性,Controller中的代码和第3点说的一致,但是,在jsp代码中,需要使用“属性名(对象类型的属性).属性名”来命名input的name。

5. List绑定:
    List需要绑定在对象上,而不能直接写在Controller方法的参数中。
    Model代码:

 

1

 

2

 

3

 

4

 

5

 

6

 

7

 

8

 

9

 

10

 

11

 

12

 

13

 

14

 

15

 

16

 

17

 

18

 

19

 

20

 

21

 

22

 

23

 

24

 

25

 

26

 

27

 

28

 

29

 

30

 

31

 

32

 

33

 

34

 

35

 

36

 

37

 

 

public class User {

 

 

 

    private String firstName;

 

 

 

    private String lastName;

 

 

 

    public String getFirstName() {

 

        return firstName;

 

    }

 

 

 

    public void setFirstName(String firstName) {

 

        this.firstName = firstName;

 

    }

 

 

 

    public String getLastName() {

 

        return lastName;

 

    }

 

 

 

    public void setLastName(String lastName) {

 

        this.lastName = lastName;

 

    }

 

 

 

}

 

 

 

    public class UserListForm {

 

 

 

    private List<User> users;

 

 

 

    public List<User> getUsers() {

 

        return users;

 

    }

 

 

 

    public void setUsers(List<User> users) {

 

        this.users = users;

 

    }

 

 

 

}

 

 

 

 


    Controller代码:

 

1

 

2

 

3

 

4

 

5

 

6

 

 

@RequestMapping("test.do")

 

public void test(UserListForm userForm) {

 

    for (User user : userForm.getUsers()) {

 

        System.out.println(user.getFirstName() + " - " + user.getLastName());

 

    }

 

}

 

 

 

 


    JSP表单代码:

 

1

 

2

 

3

 

4

 

5

 

6

 

7

 

8

 

9

 

10

 

11

 

12

 

13

 

14

 

15

 

16

 

17

 

18

 

19

 

20

 

21

 

22

 

23

 

24

 

25

 

26

 

27

 

28

 

29

 

 

<form action="test.do" method="post">

 

   <table>

 

      <thead>

 

         <tr>

 

            <th>First Name</th>

 

            <th>Last Name</th>

 

         </tr>

 

      </thead>

 

      <tfoot>

 

         <tr>

 

            <td colspan="2"><input type="submit" value="Save" /></td>

 

         </tr>

 

      </tfoot>

 

      <tbody>

 

         <tr>

 

            <td><input name="users[0].firstName" value="aaa" /></td>

 

            <td><input name="users[0].lastName" value="bbb" /></td>

 

         </tr>

 

         <tr>

 

            <td><input name="users[1].firstName" value="ccc" /></td>

 

            <td><input name="users[1].lastName" value="ddd" /></td>

 

         </tr>

 

         <tr>

 

            <td><input name="users[2].firstName" value="eee" /></td>

 

            <td><input name="users[2].lastName" value="fff" /></td>

 

         </tr>

 

      </tbody>

 

   </table>

 

</form>

 

 

 

 


其实,这和第4点User对象中的contantInfo数据的绑定有点类似,但是这里的UserListForm对象里面的属性被定义成List,而不是普通自定义对象。所以,在JSP中需要指定List的下标。值得一提的是,Spring会创建一个以***下标值为size的List对象,所以,如果JSP表单中有动态添加行、删除行的情况,就需要特别注意,譬如一个表格,用户在使用过程中经过多次删除行、增加行的操作之后,下标值就会与实际大小不一致,这时候,List中的对象,只有在jsp表单中对应有下标的那些才会有值,否则会为null,看个例子:
    JSP表单代码:

 

1

 

2

 

3

 

4

 

5

 

6

 

7

 

8

 

9

 

10

 

11

 

12

 

13

 

14

 

15

 

16

 

17

 

18

 

19

 

20

 

21

 

22

 

23

 

24

 

25

 

26

 

27

 

28

 

29

 

 

<form action="test.do" method="post">

 

   <table>

 

      <thead>

 

         <tr>

 

            <th>First Name</th>

 

            <th>Last Name</th>

 

         </tr>

 

      </thead>

 

      <tfoot>

 

         <tr>

 

            <td colspan="2"><input type="submit" value="Save" /></td>

 

         </tr>

 

      </tfoot>

 

      <tbody>

 

         <tr>

 

            <td><input name="users[0].firstName" value="aaa" /></td>

 

            <td><input name="users[0].lastName" value="bbb" /></td>

 

         </tr>

 

         <tr>

 

            <td><input name="users[1].firstName" value="ccc" /></td>

 

            <td><input name="users[1].lastName" value="ddd" /></td>

 

         </tr>

 

         <tr>

 

            <td><input name="users[20].firstName" value="eee" /></td>

 

            <td><input name="users[20].lastName" value="fff" /></td>

 

         </tr>

 

      </tbody>

 

   </table>

 

</form>

 

 

 

 


这个时候,Controller中的userForm.getUsers()获取到List的size为21,而且这21个User对象都不会为null,但是,第2到第19的User对象中的firstName和lastName都为null。打印结果:

 

1

 

2

 

3

 

4

 

5

 

6

 

7

 

8

 

9

 

10

 

11

 

12

 

13

 

14

 

15

 

16

 

17

 

18

 

19

 

20

 

21

 

 

aaa - bbb

 

ccc - ddd

 

null - null

 

null - null

 

null - null

 

null - null

 

null - null

 

null - null

 

null - null

 

null - null

 

null - null

 

null - null

 

null - null

 

null - null

 

null - null

 

null - null

 

null - null

 

null - null

 

null - null

 

null - null

 

eee - fff

 

 

 

 



6. Set绑定:
    Set和List类似,也需要绑定在对象上,而不能直接写在Controller方法的参数中。但是,绑定Set数据时,必须先在Set对象中add相应的数量的模型对象。
    Model代码:

 

1

 

2

 

3

 

4

 

5

 

6

 

7

 

8

 

9

 

10

 

11

 

12

 

13

 

14

 

15

 

16

 

17

 

18

 

19

 

20

 

21

 

22

 

23

 

24

 

25

 

26

 

27

 

28

 

29

 

30

 

31

 

32

 

33

 

34

 

35

 

36

 

37

 

38

 

39

 

40

 

41

 

42

 

43

 

 

public class User {

 

 

 

    private String firstName;

 

 

 

    private String lastName;

 

 

 

    public String getFirstName() {

 

        return firstName;

 

    }

 

 

 

    public void setFirstName(String firstName) {

 

        this.firstName = firstName;

 

    }

 

 

 

    public String getLastName() {

 

        return lastName;

 

    }

 

 

 

    public void setLastName(String lastName) {

 

        this.lastName = lastName;

 

    }

 

 

 

}

 

 

 

public class UserSetForm {

 

 

 

    private Set<User> users = new HashSet<User>();

 

     

 

    public UserSetForm(){

 

        users.add(new User());

 

        users.add(new User());

 

        users.add(new User());

 

    }

 

 

 

    public Set<User> getUsers() {

 

        return users;

 

    }

 

 

 

    public void setUsers(Set<User> users) {

 

        this.users = users;

 

    }

 

 

 

}

 

 

 

 


    Controller代码:

 

1

 

2

 

3

 

4

 

5

 

6

 

 

@RequestMapping("test.do")

 

public void test(UserSetForm userForm) {

 

    for (User user : userForm.getUsers()) {

 

        System.out.println(user.getFirstName() + " - " + user.getLastName());

 

    }

 

}

 

 

 

 


    JSP表单代码:

 

1

 

2

 

3

 

4

 

5

 

6

 

7

 

8

 

9

 

10

 

11

 

12

 

13

 

14

 

15

 

16

 

17

 

18

 

19

 

20

 

21

 

22

 

23

 

24

 

25

 

26

 

27

 

28

 

29

 

 

<form action="test.do" method="post">

 

   <table>

 

      <thead>

 

         <tr>

 

            <th>First Name</th>

 

            <th>Last Name</th>

 

         </tr>

 

      </thead>

 

      <tfoot>

 

         <tr>

 

            <td colspan="2"><input type="submit" value="Save" /></td>

 

         </tr>

 

      </tfoot>

 

      <tbody>

 

         <tr>

 

            <td><input name="users[0].firstName" value="aaa" /></td>

 

            <td><input name="users[0].lastName" value="bbb" /></td>

 

         </tr>

 

         <tr>

 

            <td><input name="users[1].firstName" value="ccc" /></td>

 

            <td><input name="users[1].lastName" value="ddd" /></td>

 

         </tr>

 

         <tr>

 

            <td><input name="users[2].firstName" value="eee" /></td>

 

            <td><input name="users[2].lastName" value="fff" /></td>

 

         </tr>

 

      </tbody>

 

   </table>

 

</form>

 

 

 

 


基本和List绑定类似。
需要特别提醒的是,如果***下标值大于Set的size,则会抛出org.springframework.beans.InvalidPropertyException异常。所以,在使用时有些不便。暂时没找到解决方法,如果有网友知道,请回帖共享你的做法。

5. Map绑定:
    Map最为灵活,它也需要绑定在对象上,而不能直接写在Controller方法的参数中。
    Model代码:

 

1

 

2

 

3

 

4

 

5

 

6

 

7

 

8

 

9

 

10

 

11

 

12

 

13

 

14

 

15

 

16

 

17

 

18

 

19

 

20

 

21

 

22

 

23

 

24

 

25

 

26

 

27

 

28

 

29

 

30

 

31

 

32

 

33

 

34

 

35

 

36

 

37

 

 

public class User {

 

 

 

    private String firstName;

 

 

 

    private String lastName;

 

 

 

    public String getFirstName() {

 

        return firstName;

 

    }

 

 

 

    public void setFirstName(String firstName) {

 

        this.firstName = firstName;

 

    }

 

 

 

    public String getLastName() {

 

        return lastName;

 

    }

 

 

 

    public void setLastName(String lastName) {

 

        this.lastName = lastName;

 

    }

 

 

 

}

 

 

 

public class UserMapForm {

 

 

 

    private Map<String, User> users;

 

 

 

    public Map<String, User> getUsers() {

 

        return users;

 

    }

 

 

 

    public void setUsers(Map<String, User> users) {

 

        this.users = users;

 

    }

 

 

 

}

 

 

 

 


    Controller代码:

 

1

 

2

 

3

 

4

 

5

 

6

 

7

 

 

@RequestMapping("test.do")

 

public void test(UserMapForm userForm) {

 

    for (Map.Entry<String, User> entry : userForm.getUsers().entrySet()) {

 

        System.out.println(entry.getKey() + ": " + entry.getValue().getFirstName() + " - " +

 

                              entry.getValue().getLastName());

 

    }

 

}

 

 

 

 


    JSP表单代码:

 

1

 

2

 

3

 

4

 

5

 

6

 

7

 

8

 

9

 

10

 

11

 

12

 

13

 

14

 

15

 

16

 

17

 

18

 

19

 

20

 

21

 

22

 

23

 

24

 

25

 

26

 

27

 

28

 

29

 

 

<form action="test.do" method="post">

 

   <table>

 

      <thead>

 

         <tr>

 

            <th>First Name</th>

 

            <th>Last Name</th>

 

         </tr>

 

      </thead>

 

      <tfoot>

 

         <tr>

 

            <td colspan="2"><input type="submit" value="Save" /></td>

 

         </tr>

 

      </tfoot>

 

      <tbody>

 

         <tr>

 

            <td><input name="users['x'].firstName" value="aaa" /></td>

 

            <td><input name="users['x'].lastName" value="bbb" /></td>

 

         </tr>

 

         <tr>

 

            <td><input name="users['y'].firstName" value="ccc" /></td>

 

            <td><input name="users['y'].lastName" value="ddd" /></td>

 

         </tr>

 

         <tr>

 

            <td><input name="users['z'].firstName" value="eee" /></td>

 

            <td><input name="users['z'].lastName" value="fff" /></td>

 

         </tr>

 

      </tbody>

 

   </table>

 

</form>

 

 

 

 


打印结果:

 

1

 

2

 

3

 

 

x: aaa - bbb

 

y: ccc - ddd

 

z: eee - fff

 

 

 

 

【编辑推荐】

  1. 技术解析 容错服务器技术还是双机冗余?
  2. Linux下使用mke2fsk格式化分区的方法
  3. Ubuntu 11.10 利用终端环境备份还原
责任编辑:赵宁宁
相关推荐

2009-07-22 17:21:27

ASP.NET 2.0

2010-03-05 16:56:42

Python绑定C++

2021-09-01 14:36:14

鸿蒙HarmonyOS应用

2021-09-01 10:37:25

鸿蒙HarmonyOS应用

2010-07-28 13:11:13

Flex数据绑定

2010-07-28 13:31:10

Flex数据绑定

2009-07-06 14:42:24

Visual Basi

2010-09-17 20:40:09

2009-09-16 11:15:52

Linq联接数据

2009-09-07 17:32:14

LINQ检索数据

2017-07-11 06:07:59

金融大数据互联网

2010-07-30 10:45:08

Flex数据绑定

2010-08-05 15:06:19

Flex数据绑定

2010-12-20 15:13:11

Windows For

2021-02-11 08:27:28

数据

2010-07-30 09:08:21

Flex数据绑定

2010-08-12 11:34:15

Flex数据绑定

2009-12-24 11:15:59

WPF数据绑定

2009-08-10 16:47:45

Visual C#数据

2023-10-07 11:04:58

WPF数据UI
点赞
收藏

51CTO技术栈公众号