`

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

    博客分类:
  • java
 
阅读更多
转:http://blog.chinabyte.com/a/2883297.html
问题由来:
  
最近在一个java bean类中定义了一个boolean类型的变量:
  
//boolean属性:是否显示 
private boolean isShowCode ;  
     
//使用Eclipse自动生成getter/setter方法如下: 
public boolean isShowCode() { 
   return isShowCode; 

public void setShowCode(boolean isShowCode) { 
    this.isShowCode = isShowCode; 
}




spring在给java bean 设置值的时候, 抛出异常:
  
Caused by: org.springframework.beans.NotWritablePropertyException:  
Invalid property 'isShowCode' of bean class [com.codemouse.beans.Country]:  
Bean property 'isShowCode' is not writable or has an invalid setter method.  
Did you mean 'showCode'? 
at org.springframework.beans.BeanWrapperImpl.setPropertyValue(BeanWrapperImpl.java:1064) 




代码运行环境: jdk 1.6 + eclipse 3.2 + spring 3.1, 本文下面的代码都在该环境下测试。
  
原因跟踪分析:跟踪Spring源码, 可以看到下面这段代码:
  
private CachedIntrospectionResults(Class beanClass,  
    boolean cacheFullMetadata) throws BeansException { 
    ... ... 
    this.beanInfo = new ExtendedBeanInfo(Introspector.getBeanInfo(beanClass)); 
    ... ... 
}




方法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方法。代码片段:
  
<bean id="country" class="com.codemouse.beans.Country" lazy-init="true"> 
    <property name="pName"> 
        <value>中国</value> 
    </property> 
    <property name="code"> 
        <value>CN</value> 
    </property> 
    <property name="showCode"> 
        <value>true</value> 
    </property> 
</bean>




   private String pName; 

public String getPName() { 
    return pName; 

public void setPName(String name) { 
    pName = name; 
}




3.2 )对应的getter/setter方法名也可以为:get/set+ <pName>(), 即属性名称不变,第一个单词的首字母任然为小写,前面再加上"get"或"set"前缀。这种方式也可以正常运行。网上有帖子说Eclipse3.5按这种方式自动生成getter/setter方法。
  
代码片段:
  
   <bean id="country" class="com.codemouse.beans.Country" lazy-init="true"> 
    <property name="pName"> 
        <value>中国</value> 
    </property> 
    <property name="code"> 
        <value>CN</value> 
    </property> 
    <property name="showCode"> 
        <value>true</value> 
    </property> 
</bean>




   private String pName; 

public String getpName() { 
    return pName; 

public void setpName(String name) { 
    pName = name; 





4. 对于非常规属性<PName>, 属性名称的前两个字母都是大写 。即连续两个大写字母开头的属性名。
  
对应的getter/setter方法名为: get/set + <PName>(), 即属性名称不变,前面再加上"get"或"set"前缀。
  
spring3.1 配置文件代码片段:
  
<bean id="country" class="com.codemouse.beans.Country" lazy-init="true"> 
    <property name="PName"> 
        <value>中国</value> 
    </property> 
    <property name="code"> 
        <value>CN</value> 
    </property> 
    <property name="showCode"> 
        <value>true</value> 
    </property> 
</bean>




private String PName; 

public String getPName() { 
    return PName; 

public void setPName(String name) { 
    PName = name; 
}




5. 对于非常规属性<Property>或<PropertyName>, 属性名称第一个字母大写 。网上有帖子说这是不符合JSR规范的,会报 "属性找不到" 的错误。
  
(如帖子1: http://lzh166.iteye.com/blog/631838 ;
  
帖子2: http://hi.baidu.com/w8y56f/blog/item/4fd037e845bbbe372cf5342a.html)。我在我的 环境下测试了下, 是不会报错的,可以正常运行,虽然这种命名方式是令人难以忍受的:
  
<bean id="country" class="com.codemouse.beans.Country" lazy-init="true">
    <property name="PropertyName">
        <value>中国</value>
    </property>
    <property name="code">
        <value>CN</value>
    </property>
    <property name="showCode">
        <value>true</value>
    </property>
    <property name="Xcoordinate">
        <value>12.345</value>
    </property>
</bean>




private String PropertyName; 
public String getPropertyName() { 
    return PropertyName; 

public void setPropertyName(String propertyName) { 
    PropertyName = propertyName; 


private Double Xcoordinate; 
public Double getXcoordinate() { 
    return Xcoordinate; 

public void setXcoordinate(Double xcoordinate) { 
    Xcoordinate = xcoordinate; 
}




测试方法: 第一个@test方法用普通javabean调用方式测试; 第二个@test方法使用spring创建bean
  
   @Test
public void testJavaBeanNamingRule0(){ 
    Country country = new Country(); 
    country.setPropertyName("中国"); 
    country.setXcoordinate(Double.valueOf(123.456f)); 
    System.out.println(country.getPropertyName()); 
    System.out.println(country.getXcoordinate()); 


@Test
public void testJavaBeanNamingRule(){ 
    ApplicationContext ctx = new ClassPathXmlApplicationContext("myBeans.xml"); 
    Country country = (Country)ctx.getBean("country"); 
    System.out.println(country.getPropertyName()); 
    System.out.println(country.getXcoordinate()); 
}




运行结果:都可以正常运行。
  
中国 
123.45600128173828
log4j:WARN No appenders could be found for logger (org.springframework.core.env.StandardEnvironment). 
log4j:WARN Please initialize the log4j system properly. 
中国 
12.345




总结:
  
1. javabean属性命名尽量使用常规的驼峰式命名规则
  
2. 属性名第一个单词尽量避免使用一个字母:如eBook, eMail。
  
3. boolean属性名避免使用 “is” 开头的名称
  
4. 随着jdk, eclipse, spring 等软件版本的不断提高, 底版本的出现的问题可能在高版本中解决了, 低版本原来正常的代码可能在高版本环境下不再支持。
分享到:
评论

相关推荐

    JavaBean属性命名规范问题分析

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

    使用Betwixt将XML转为JAVA Bean(内有JAR包)

    通过配置和API的使用,开发者可以根据需要定制XML和Java Bean之间的映射规则,提高代码的可读性和可维护性。在实际项目中,结合Java Bean的封装性和XML的灵活性,可以实现更高效的代码组织和数据交换。

    动态生成Java Bean测试页面

    1. Java Bean规范:介绍Java Bean的基本特点,包括公共构造函数、getter/setter方法、序列化支持以及属性、事件和方法的标准化命名。 2. 动态生成原理:解释了如何分析Java Bean的元数据(如反射API),以构建测试...

    JAVA bean 技术.JSP

    类名应与包含它的源文件名相同,遵循Java的命名规则。 3. **属性访问**:Java Bean的属性通常通过getter和setter方法来访问。getter方法用于获取属性值,setter方法用于设置属性值。 4. **方法封装**:Java Bean的...

    Java_Bean教程

    Java Bean的命名规范遵循一定的驼峰命名法,如包名全小写,类名首字母大写,属性和方法名则遵循首字母小写,后续单词首字母大写的规则。常量名则全部大写,以示区分。 在实际应用中,Java Bean通常会被封装在特定的...

    java bean和map之间的互转

    Java Bean是一种符合JavaBeans规范的Java类,它遵循以下规则: 1. 类是公共的(public)。 2. 至少有一个无参数的构造器。 3. 属性通过getter和setter方法暴露,这些方法遵循驼峰命名法,例如getAge()和setAge(int ...

    java开发命名规范

    ### Java开发命名规范详解 #### 一、概述 在Java软件开发过程中,遵循一套统一的命名规范至关重要。它不仅能够提高代码的可读性和可维护性,还能够增强团队协作效率。本文将根据提供的文件信息,详细介绍Java开发...

    java命名规范 开发规范

    1. 模块命名、数据库表命名、域模型命名、各分层的类/方法命名、页面的命名; 模块命名: a. 包命名:com.project_name.module_name.action/service/dao/ws; service的实现都置于...

    java Map转换成bean,利用java反射以及java和bean包

    一个符合JavaBeans规范的类通常有无参构造器,公有的getter和setter方法,以及符合命名规则的属性(通常是驼峰命名)。在Map到Bean的转换中,我们通常会寻找Map中的键与Bean的属性名相对应,然后通过反射调用setter...

    jsp java bean

    **标题:“jsp java bean”** **内容概要:** 本项目是基于JSP和JavaBean技术构建的一个电子商务系统,主要用于实现购书流程,包括用户注册、浏览书籍、选择书籍以及下单购买等功能。JSP(JavaServer Pages)是Java...

    Java Beans 讲解

    17.2.3 Java Bean 的命名规范 Java Bean的命名规范包括: - 包名:全小写字母。 - 类名:每个单词首字母大写。 - 属性名:首个单词全小写,后续单词首字母大写。 - 方法名:与属性名相同规则。 - 常量名:全大写字母...

    使用mybatis自动建bean

    1. 命名规则:为了提高可读性和可维护性,Java Bean的属性命名应遵循驼峰命名法,而数据库列名则采用下划线命名法。MyBatis可以通过`&lt;resultMap&gt;`中的`column`属性进行字段映射。 2. 类型匹配:确保Java Bean的属性...

    逆向生成工具,有数据库表生成java bean实体类

    对于每一个表,工具会自动生成一个对应的Java Bean类,这个类包含了表中的字段作为类的属性,并为每个属性提供了getter和setter方法。此外,如果在XML配置文件中启用了相关选项,工具还会生成toString、equals和...

    Java Bean自动生成系统的设计与实现.zip

    1. **需求分析**:首先,我们需要明确系统的目标用户,了解他们对于Java Bean的需求,如属性类型、访问级别、构造函数、getter/setter方法等。此外,还需要考虑是否支持注解、序列化以及与其他框架(如Spring)的...

    使用jaxb将XML转化为JAVA BEAN

    当一个Java对象被转换为XML时,JAXB会根据对象的属性和注解生成相应的XML元素和属性。例如,一个名为`Person`的类,包含`name`和`age`属性,可以通过以下方式标注: ```java @XmlRootElement(name = "person") ...

    用户注册系统: 采用Java Bean技术的EJB及JSP,可以发送邮件,连接MySQL数据库

    Java Bean遵循特定的命名和设计规则,使其易于在不同应用程序之间集成。开发者可以通过属性(如setter和getter方法)来访问和修改Bean中的数据。 **Enterprise JavaBeans (EJB)** EJB是Java EE平台的一部分,用于...

    自动读取mysql数据库字段并自动生成java属性和set和get方法

    字段名通常需要遵循Java命名规范,将下划线转为驼峰式命名。 5. **生成getter和setter**:对于每个属性,生成相应的getter和setter方法。这些方法遵循Java Bean规范,用于获取和设置属性值。 6. **生成源代码文件*...

    JAVA BEAN技术

    - **属性的命名规则**:属性名称应遵循一定的命名规则,如使用驼峰命名法。 - **属性的访问方式**:通过`getXXX()`和`setXXX()`方法来访问和修改属性值。 - **事件的触发机制**:JavaBean可以通过监听器机制触发特定...

    Hibernate Bean Validation jar包

    如果验证的是属性(getter 方法),那么必须遵从 Java Bean 的命名习惯(JavaBeans 规范); 静态的字段和方法不能进行约束验证; 约束适用于接口和基类; 约束注解定义的目标元素可以是字段、属性或者类型等; 可以...

Global site tag (gtag.js) - Google Analytics