Java Bean属性命名规范问题分析

开发 后端
Eclipse根据这种方式生成getter访问器和setter设值器, 由于属性名isShowCode的is没有去掉, 以致java bean类违背了java bean 的命名规范。

问题由来:

最近在一个java bean类中定义了一个boolean类型的变量:

  1. //boolean属性:是否显示  
  2. private boolean isShowCode ;   
  3.       
  4. //使用Eclipse自动生成getter/setter方法如下:  
  5. public boolean isShowCode() {  
  6.    return isShowCode;  
  7. }  
  8. public void setShowCode(boolean isShowCode) {  
  9.     this.isShowCode = isShowCode;  

spring在给java bean 设置值的时候, 抛出异常:

  1. Caused by: org.springframework.beans.NotWritablePropertyException:   
  2. Invalid property 'isShowCode' of bean class [com.codemouse.beans.Country]:   
  3. Bean property 'isShowCode' is not writable or has an invalid setter method.   
  4. Did you mean 'showCode'?  
  5. at org.springframework.beans.BeanWrapperImpl.setPropertyValue(BeanWrapperImpl.java:1064)  

代码运行环境: jdk 1.6 + eclipse 3.2 + spring 3.1, 本文下面的代码都在该环境下测试。

原因跟踪分析:跟踪Spring源码, 可以看到下面这段代码:

  1. private CachedIntrospectionResults(Class beanClass,   
  2.     boolean cacheFullMetadata) throws BeansException {  
  3.     ... ...  
  4.     this.beanInfo = new ExtendedBeanInfo(Introspector.getBeanInfo(beanClass));  
  5.     ... ...  

方法Introspector.getBeanInfo(beanClass)返回的时候,获取到的java bean 信息中的isShowCode属性的名称已经被改成了"showCode"。

 

到这里可以确定问题不是出在spring代码中, 网上有不少帖子说是spring的处理规则导致了这个问题,这里是不是可以否定这种看法?

问题跟踪到这,也即跟踪到了java.beans包。尝试继续跟踪JDK源码,可能由于我的JDK的jar包和源码不匹配的原因, eclipse总是监控不到中间变量。也就没有再跟踪进去了。可以知道的是,javabean中 的isShowCode 属性 和 对应的getter/setter方法应该是没有遵循javabean规范。Eclipse自动生成的getter/setter方法看来也是存在一些问题的。

Eclipse自动生成boolean类型属性的方法是不是有点奇怪呢? 属性 isShowCode 的getter访问器是isShowCode()而不是getIsShowCode(), setter设值器是setShowCode()而不是setIsShowCode()。原来在java bean 规范关于中提到, boolean属性<propertyName>的getter访问器可以使用下面这种模式

public boolean is<PropertyName>(){...};

来代替

public boolean get<PropertyName>(){...};

Javabean 规范(下载链接:http://download.oracle.com/otndocs/jcp/7224-javabeans-1.01-fr-spec-oth-JSpec/ ) 在8.3 章节"Design Patterns for Properties" 中的描述:

Eclipse根据这种方式生成getter访问器和setter设值器, 由于属性名isShowCode的is没有去掉, 以致java bean类违背了java bean 的命名规范。

JavaBean 的属性名和getter/setter存取方法规则小结:

1. 对于常规属性 <propertyName> , 属性名称的***个单词小写且字母个数大于1,第二个单词首字母大写 。对应的getter/setter方法名为:get /set + <PropertyName>(), 即属性名称的***个单词的首字母改成大写, 前面再加上"get"或"set"前缀。

2. 对于布尔类型 <propertyName> , 可以按常规属性的规则编写getter/setter方法外, getter方法可以使用 is + <PropertyName>()的形式来代替。

3. 对于非常规属性<pName>, 属性名称的***个单词小写且字母个数等于1,第二个单词首字母大写 。

3.1 ) 对应的getter/setter方法名可以为:get/set + <PName>(), 即***个单词的首字母为改为大写,前面再加上"get"或"set"前缀。Eclipse3.2 按这种方式自动生成getter/setter方法。代码片段:

  1. <bean id="country" class="com.codemouse.beans.Country" lazy-init="true">  
  2.     <property name="pName">  
  3.         <value>中国</value>  
  4.     </property>  
  5.     <property name="code">  
  6.         <value>CN</value>  
  7.     </property>  
  8.     <property name="showCode">  
  9.         <value>true</value>  
  10.     </property>  
  11. </bean> 
  1.    private String pName;  
  2.  
  3. public String getPName() {  
  4.     return pName;  
  5. }  
  6. public void setPName(String name) {  
  7.     pName = name;  

3.2 )对应的getter/setter方法名也可以为:get/set+ <pName>(), 即属性名称不变,***个单词的首字母任然为小写,前面再加上"get"或"set"前缀。这种方式也可以正常运行。网上有帖子说Eclipse3.5按这种方式自动生成getter/setter方法。

代码片段:

  1.    <bean id="country" class="com.codemouse.beans.Country" lazy-init="true">  
  2.     <property name="pName">  
  3.         <value>中国</value>  
  4.     </property>  
  5.     <property name="code">  
  6.         <value>CN</value>  
  7.     </property>  
  8.     <property name="showCode">  
  9.         <value>true</value>  
  10.     </property>  
  11. </bean> 
  1.    private String pName;  
  2.  
  3. public String getpName() {  
  4.     return pName;  
  5. }  
  6. public void setpName(String name) {  
  7.     pName = name;  
  8. }  

4. 对于非常规属性<PName>, 属性名称的前两个字母都是大写 。即连续两个大写字母开头的属性名。

对应的getter/setter方法名为: get/set + <PName>(), 即属性名称不变,前面再加上"get"或"set"前缀。

spring3.1 配置文件代码片段:

  1. <bean id="country" class="com.codemouse.beans.Country" lazy-init="true">  
  2.     <property name="PName">  
  3.         <value>中国</value>  
  4.     </property>  
  5.     <property name="code">  
  6.         <value>CN</value>  
  7.     </property>  
  8.     <property name="showCode">  
  9.         <value>true</value>  
  10.     </property>  
  11. </bean> 
  1. private String PName;  
  2.  
  3. public String getPName() {  
  4.     return PName;  
  5. }  
  6. public void setPName(String name) {  
  7.     PName = name;  

5. 对于非常规属性<Property>或<PropertyName>, 属性名称***个字母大写 。网上有帖子说这是不符合JSR规范的,会报 "属性找不到" 的错误。

(如帖子1: http://lzh166.iteye.com/blog/631838 ;

帖子2: http://hi.baidu.com/w8y56f/blog/item/4fd037e845bbbe372cf5342a.html)。我在我的 环境下测试了下, 是不会报错的,可以正常运行,虽然这种命名方式是令人难以忍受的:

  1. <bean id="country" class="com.codemouse.beans.Country" lazy-init="true"> 
  2.     <property name="PropertyName"> 
  3.         <value>中国</value> 
  4.     </property> 
  5.     <property name="code"> 
  6.         <value>CN</value> 
  7.     </property> 
  8.     <property name="showCode"> 
  9.         <value>true</value> 
  10.     </property> 
  11.     <property name="Xcoordinate"> 
  12.         <value>12.345</value> 
  13.     </property> 
  14. </bean> 
  1. private String PropertyName;  
  2. public String getPropertyName() {  
  3.     return PropertyName;  
  4. }  
  5. public void setPropertyName(String propertyName) {  
  6.     PropertyName = propertyName;  
  7. }  
  8.  
  9. private Double Xcoordinate;  
  10. public Double getXcoordinate() {  
  11.     return Xcoordinate;  
  12. }  
  13. public void setXcoordinate(Double xcoordinate) {  
  14.     Xcoordinate = xcoordinate;  

测试方法: ***个@test方法用普通javabean调用方式测试; 第二个@test方法使用spring创建bean

  1.    @Test 
  2. public void testJavaBeanNamingRule0(){  
  3.     Country country = new Country();  
  4.     country.setPropertyName("中国");  
  5.     country.setXcoordinate(Double.valueOf(123.456f));  
  6.     System.out.println(country.getPropertyName());  
  7.     System.out.println(country.getXcoordinate());  
  8. }  
  9.  
  10. @Test 
  11. public void testJavaBeanNamingRule(){  
  12.     ApplicationContext ctx = new ClassPathXmlApplicationContext("myBeans.xml");  
  13.     Country country = (Country)ctx.getBean("country");  
  14.     System.out.println(country.getPropertyName());  
  15.     System.out.println(country.getXcoordinate());  

运行结果:都可以正常运行。

  1. 中国  
  2. 123.45600128173828 
  3. log4j:WARN No appenders could be found for logger (org.springframework.core.env.StandardEnvironment).  
  4. log4j:WARN Please initialize the log4j system properly.  
  5. 中国  
  6. 12.345 

总结:

1. javabean属性命名尽量使用常规的驼峰式命名规则

2. 属性名***个单词尽量避免使用一个字母:如eBook, eMail。

3. boolean属性名避免使用 “is” 开头的名称

4. 随着jdk, eclipse, spring 等软件版本的不断提高, 底版本的出现的问题可能在高版本中解决了, 低版本原来正常的代码可能在高版本环境下不再支持。

原文链接:http://blog.csdn.net/yunye114105/article/details/7364264

【编辑推荐】

  1. Java Socket编程:初识TCP Socket
  2. Java Socket编程:如何识别网络主机
  3. Java Socket编程:基本的术语和概念
  4. Java集合类ArrayList循环中删除特定元素
  5. Java集合框架总结:Set接口的使用
责任编辑:林师授 来源: 成家荣的博客
相关推荐

2023-12-06 07:24:42

属性命名云原生

2010-01-08 16:28:31

VB.NET方法和属性

2011-07-22 17:14:53

java

2020-04-16 21:02:35

前端命名规范html规范

2012-03-22 17:16:24

Java

2016-05-17 14:03:07

Android命名解决方案

2023-11-22 08:00:56

Go命名规范

2009-07-31 18:18:33

Camel命名法C#命名规范

2010-09-07 15:53:02

CSS规范化

2009-08-03 16:57:42

ASP.NET编程规范

2009-08-21 08:52:40

C#语言命名

2009-08-13 13:38:30

C#命名规范

2009-08-27 16:30:08

C#编程命名规范

2009-08-19 15:24:30

.NET命名规范

2022-08-02 07:48:06

容器镜像版本

2009-05-22 11:01:53

C++JavaC#

2013-05-27 11:52:16

CiscoIOS思科交换机

2009-10-27 14:49:11

VB.NET命名规范

2009-09-01 15:08:07

C#命名规范

2009-08-28 09:30:48

C#命名属性
点赞
收藏

51CTO技术栈公众号