程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> JAVA編程 >> J2EE >> Java Bean屬性命名規范問題分析

Java Bean屬性命名規范問題分析

編輯:J2EE

問題由來:

最近在一個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接口的使用
【責任編輯:小林 TEL:(010)68476606】
  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved