编码开发规范.doc
编码规范说明书 中海集运滞期费管理信息系统中海集运文档编号:生效日期:密级:版本:R1.0编制: 审核:Copyright Notice Copyright© 2003 - 2009 Gillion Technologies Ltd.No part of this publication may be copied without the express writtenPermission of Gillion Technologies Ltd.修订历史DateRevisionDescriptionAuthor2014-01-04R1.0Create林华导言51.1.编写目的51.2.内容边界51.3.术语及缩写51.4.文档结构51.5.参考文档51.声明5总体描述51.1.包51.2.类、接口61.3.方法61.4.字段61.5.代码示例72.命名7总体描述72.1.包与类命名82.1.1.包命名82.1.2.类命名82.1.3.Entity对象命名82.1.4.Service命名82.1.5.DAO命名92.1.6.Exception命名92.2.属性命名92.2.1.成员变量和参数命名92.2.2.常量命名102.2.3.项目中控件/组件命名102.3.方法命名112.3.1.Entity类中方法命名112.3.2.Service类中方法命名112.3.3.项目中的方法命名123.注释12总体描述123.1.类、接口注释123.2.方法注释133.3.其它注释133.4.常用JavaDoc标记134.代码组织与风格14总体描述144.1.缩进144.2.长度144.3.行宽144.4.间隔154.5.括号155.性能与安全15总体描述155.1.通用性能优化155.1.1.String与StringBugffer155.1.2.集合155.1.3.对象165.1.4.final165.1.5.abstract165.1.6.垃圾收集和资源释放165.1.7.慎用异常16 导言1.1. 编写目的本规范的目的是为了使海丰集运项目开发小组能以标准的、规范的方式进行编码。通过该规范的学习,以使开发人员养成良好的编码风格和习惯,并以此形成开发小组编码约定,提高程序的可靠性、可读性、可维护性和一致性等,增进团队间的交流,并保证软件产品的质量。1.2. 内容边界结合项目实际开发情况和Java基础语法知识,主要针对声明、命名、注释、代码组织与风格、性能和安全等方面进行规范和说明。1.3. 术语及缩写全称含义1.4. 文档结构1.5. 参考文档资料名称位置1. 声明总体描述声明的基本原则是遵守Java语言规范,并遵从项目习惯用法。 1.1. 包在导入包时当完全限制代码所有的类的名字,尽量少用通配符的方式,但导入一些通用包,或用到一个包下大部分类时,则可使用通配符方式,如:例如:import mons.beanutils.PropertyUtils;import java.util.*; 同意个包中的类的导入时当声明在一起,可由编辑器自动完成功能。Eclipse采用ctrl+shift+O 导入包1.2. 类、接口类、接口定义语法规范:可见性('abstract'|'final') Class|Interface class_name ('extends'|'implements')父类或接口名 例如: public class FuelSaveAwardByGradeView extends CommonDataView 其中: abstract类含抽象方法的类,不能实例化,但允许继承他的子类实例化,只能单继承。 final类最终类,设计十分完美的类,不能有子类,不能在继承的类。1.3. 方法良好的程序设计应该尽可能减小类与类之间耦合,所遵循的经验法则是:尽量限制成员函数的可见性。如果成员函数没必要公有 (public),就定义为保护 (protected);没必要保护 (protected),就定义为私有 (private)。 方法定义语法规范:可见性('abstract'|'final') 'synchronized'返回值类型 method_name(参数列表)('throws')异常列表 / 功能模块 例如: public List getAllUserList() throws DAOException 1.4. 字段字段定义语法规范:(public|private|protected) (final|volatile)statictransient data_type field_name = expression ;例如:private string strUserName=TOM;1.5. 代码示例package com.sitc.rpt.rcp.viewpart;import java.math.BigDecimal;import java.util.ArrayList;import java.util.List;import java.util.Map;/* * author suzx * create_date 2009-01-01 * view_name 耗油/船速查询(按月份) */public class FindConsumptionByMonthView extends BaseViewpublic static final String ID = "ful.FindConsumptionByMonthView" /-常量protected static String name ="love"/ -实例变量 public String userCode="sh12345" /-公有字段protected float price =(float) 1.0; /-受保护字段private int count; /-私有字段/公共方法public void getUserName() /受保护方法-protected void convert() /私有方法-private void getUserCode() 2. 命名总体描述规范的命名能使程序更易阅读,从而更易于理解。它们也可以提供一些标识功能方面的信息,有助于更好的理解代码和应用。 2.1. 包与类命名2.1.1. 包命名海丰集运项目开发的应用包命名要求带有前缀com.sitc,采用“com.sitc.模块名.rcp.子功能包名”的形式,如“com.sitc.mkt.rcp.viewpart”。包名和子功能包名要求具有实际的意义,从单词或缩写上能够看出包的意义。例如: 2.1.2. 类命名(1)类名首字母应该大写,之后连接的单词第一个字母大写,既所谓的驼峰法(2)类名要求不要过长,一般控制在30个字符以内(3)项目针对查询页面,编辑页面,弹出页面,报表等页面进一步做了一些规定:1.查询页面: Find+实体名+View 或者 Find+功能名+View例如: FindMarketAreaView 营销区查询 FindOnScheduleStatisticsView 准班率查询2.编辑或者查询带编辑页面: Edit+功能名+实体+View 例如: Edit+MarketAreaView3.弹出框编辑页面: 功能+实体+Dialog例如: SelectCustomerDialog2.1.3. Entity对象命名项目架构通过Hibernate映射数据库,所以涉及实体的命名,该部分通过ER图直接生成,所以相对规范,具体参考:<< 海丰集运数据库设计开发规范.doc>>例如:FuelSaveAward, FuelType, VoyagePartition2.1.4. Service命名Service层主要负责处理系统的业务逻辑,一个主要的实体对应一个Service,关于这个主要实体的操作都集中在Service里面,主要涉及对象的粒度来定义一个接口和一个到多个实现。该部分通过工具直接生成,所以相对规范。示例: UserService, UserServiceImpl1. Service接口命名规范为:对象名+Service2. Service实现命名规范为:对象名+Service+Impl2.1.5. DAO命名现在的框架不需要开发人员自己写DAO代码,DAO已经统一成CommonDAO和CommonDAOHibernateImpl,如果碰到特许情况需要自己开发DAO的时候,按照以下规则:例如:UserDAOHibernateImpl, UserDAOJDOImpl, UserDAOCMPImpl1. DAO接口命名规范为:对象名+DAO2. DAO实现命名规范为:对象名+DAO+具体的O/R实现框架+Impl2.1.6. Exception命名异常类名一般由表示异常类型的单词和Exception组成,并且第一个字母大写。如:ActionException。逻辑名称+Exception2.2. 属性命名Java中成员变量又称成员属性,它是描述对象状态的数据,是类中很重要的组成成分, 成员变量是定义在类里面,并和方法处于同一层次。2.2.1. 成员变量和参数命名类的成员变量和在方法中所声明的局部变量即参数都是用户标识符,它们的命名规则相同,语法结构都是:变量修饰符 类型说明符 变量名 例如:private int nLen;1. 变量修饰符:修饰符声明了成员变量的访问权限,分别是:私有(private)、保护(protected)和公共(public)。除此之外,还有一种默认的访问权限。类型说明符:类型可以是Java中的任意数据类型,包括基本类型、数组、类和接口。变量名:第一个字母小写,之后连接的单词第一个字母大写,如果是简写的单词全部大写。2. 静态成员变量:变量前加上static标识符 例如:protected static int stat=0;3. 静态变量的另一个用途是定义静态常量:public static double PI = 3.1415926;简单数据类型规范表:数据类型变量名前缀例子booleanbbCustomExist, baCustomExistcharccSymbol, caSymbolsbyteyyByte, yaBytesshortssHour, saHoursintnnLen, nLoopCounter, naLengthlongllMemory, laTimesfloatffLength, faLengthdoubleddMile, daMilesIntegerintintQuantityStringstrstrVesselNameBigDecimalbigbigPriceDatedtdtStart注:byte与int类型的前缀较特殊,使用时应注意2.2.2. 常量命名常量在运行过程中是不可改变的,一般通过final static来修饰, static用来修饰属性时称为静态属性,不属于某个具体的对象,属于类,但是所有的对象都能对其进行修改,单个final修饰的属性并不是常量,final表示的是不可改变性,但是属于某个具体的对象,两个加起来就是一个属于类并且不可改变的属性,即常量。包括:a.整型常量 123 , b.实型常量 3.123 , c.字符常量 a, d.逻辑常量 true,false,e.字符串常量“abcd”。常量命名:全部大写,相连的单词中间以”_”分开例如:public static final int DEFAULT_PAGE_SIZE=1; private static final String STR_WORLD = "Hello world!"2.2.3. 项目中控件/组件命名控件类型 gfa4j控件类前缀 例子Check ButtonChkButtonchkchkActiveRadion ButtonCommonButtonradradCityCommond Button CommonButtonbtnbtnExitCommon dialog CommonDialogdlgdlgSltOfficeLabelCommonLabellbllblCodeText(Text)CommonTexttxttxtNameText(Date)DateTexttxttxtETDText(Long)NumberTexttxttxtLengthText(BigDecimal)FloatNumberTexttxttxtRateComboAutoSelectCombocbocboCountrySelectCommonCodeSelectCommonCodesccsccEmpNoListListlstlstPlaceTreeTreetretreConditionTableCommonTabletbltblCustomerTableColumnTableColumncolcolPlaceCodeTableItemUdtTableItemtblItmToolBarCommonToolBartlbtlbSaveToolItemCommonToolItemtolItmMenuItemMenuItemmnuGridGridTablegrtgrdCustomerDatePickDatePickerCombodtpDatePick(Time)DateTimePickerCombodtpGridLayoutGridLayoutgdLyFillLayoutFillLayoutfilLyRowLayoutRowLayoutrowLyFormLayoutFormLayoutfrmLyBordLayoutBordLayoutbrdLyFlowLayoutFlowLayoutflwLyCompositeCompositecmpGroupGroupgrpTabFolderTabFoldertabTabItemTabItemtabItmScrollCompositeScrollCompositescmpGroupCommonGroupgrpFieldFieldfldEditCompositeEditCompositeecpQueryCompositeQueryCompositeqcmpMapAutoSelectComboMapAutoSelectCombomapCboSearchToolBarSearchToolBarstlbSelectTextSelectTextsltGridDataGridDatagrd针对查询区域中的控件,建议加上Srh,例如:lblSrhUserName,txtSrhUserName,目的是为了区分查询区的控件还是编辑区的控件。2.3. 方法命名 方法的命名采用完整的英文描述,大小写混合使用,第一个单词采用动词,并且小写开头,其他单词第一个字母大写。语法格式:访问性修饰符 非访问性修饰符 返回值 方法名() 2.3.1. Entity类中方法命名Entity是一个典型的值Bean,所以方法一般情况下只有以下两种:例如:get+属性名set+属性名2.3.2. Service类中方法命名因为XxxServiceImpl都是对XxxService的实现,所以XxxServiceImpl的方法命名和XxxService的方法命名一致。u 添加:add+对象名。u 修改:update+对象名u 删除:remove+对象名u 典型查询:query+对象名u 特殊查询:find+对象名+By+条件名u 验证方法:validate+自定义标识2.3.3. 项目中的方法命名u 创建组件/部件:create +事件名u 初始化功能(如组件监听器、service等):init +事件名u 执行操作(如执行删除按钮操作):do+动宾+事件名,如doSaveCustomer,doDeleteCustomeru 有返回值的:get+事件名,如getCustomerNameu 创建按钮权限createFunc+事件名,如createFunc2.4. ER实体命名和表名 如果存在多表关联的情况,建议子表(第二层)=主表+Item ,从表(第三层)=主表+Detl 例如:主表:RepairRate 子表:RepairRateItem 从表:RepairRateDetl如果表名太长采用:主表的缩写,缩写不超过4位,具体规则:多个单词组合的缩写取每个单词的首字母或头两个字母组成,同样不超过4个字符;例如:application_function = AF (1:1)application_function = APFU (2:2)application_function_role = APFR (2:1:1)application_function_role_banana = AFRB (1:1:1:1)3. 注释总体描述程序中的注释是程序设计者与程序阅读者之间通信的重要手段。应用注释规范对于软件本身和软件开发人员而言尤为重要。并且在流行的敏捷开发思想中已经提出了将注释转为代码的概念。好的注释规范可以改善软件的可读性,可以让开发人员尽快而彻底地理解新的代码。好的注释规范可以最大限度的提高团队开发的合作效率。3.1. 类、接口注释在类、接口定义之前当对其进行注释,包括类、接口的目的、作用、功能、创建日期和修改记录有必要时还可以对继承的父类、实现的接口、实现的算法等进行注释。例如:/* * <p>类或接口描述 <p>* * 燃油航次分摊计算 * author guxg * version 1.0 $date*/3.2. 方法注释依据标准JavaDoc规范对方法进行注释,以明确该方法功能、作用、各参数含义以及返回值等。参数注释时当注明其取值范围等返回值当注释出失败、错误、异常时的返回情况。异常当注释出什么情况、什么时候、什么条件下会引发什么样的异常例如:/* 计算天数间隔* * 计算两个日期的天数间隔,永远返回正数或者零。 * param 时间一 * param 时间二 * return 时间间隔的天数 */ 3.3. 其它注释应对重要的变量加以注释,以说明其含义等。应对不易理解的分支条件表达式加注释。不易理解的循环,应说明出口条件。过长的方法实现,应将其语句按实现的功能分段加以概括性说明。对于异常处理当注明正常情况及异常情况或者条件,并说明当异常发生时程序当如何处理。针对这类情况的注释直接采用: /时间交叉判断3.4. 常用JavaDoc标记标签说明JDK 1.1 doclet标准doclet标签类型author 作者作者标识 包、 类、接口version 版本号版本号 包、 类、接口param 参数名 描述方法的入参名及描述信息,如入参有特别要求,可在此注释。 构造函数、 方法return 描述对函数返回值的注释 方法deprecated 过期文本标识随着程序版本的提升,当前API已经过期,仅为了保证兼容性依然存在,以此告之开发者不应再用这个API。 包、类、接口、值域、构造函数、 方法throws异常类名 构造函数或方法所会抛出的异常。 构造函数、 方法exception 异常类名 同throws。 构造函数、 方法see 引用查看相关内容,如类、方法、变量等。 包、类、接口、值域、构造函数、 方法since 描述文本API在什么程序的什么版本后开发支持。 包、类、接口、值域、构造函数、 方法link包.类#成员 标签 链接到某个特定的成员对应的文档中。 包、类、接口、值域、构造函数、 方法value 当对常量进行注释时,如果想将其值包含在文档中,则通过该标签来引用常量的值。 (JDK1.4)静态值域4. 代码组织与风格总体描述代码的组织和风格的基本原则是:便于自己的开发,易于与他人的交流,Eclipse采用Ctrl+Shift+F格式化。4.1. 缩进子功能块当在其父功能块后缩进。当功能块过多而导致缩进过深时当将子功能块提取出来做为子函数。代码中以TAB(4个字符)缩进,在编辑器中请将TAB设置为以空格替代,否则在不同编辑器或设置下会导致TAB长度不等而影响整个程序代码的格式。注:eclipse中用空格(space)代替制表符(tab)的设置步骤窗口(windows)->首选项(preferences.), 在左边选Java->代码样式(code style)->格式化程序(Formatter), 右边点“显示(show)”按钮,选“缩进(Indentation)”选项卡, 在“常规设置(general settings)”里的“跳格策略(Tab policy)” 下拉列表选“仅空格(Spaces only)”。4.2. 长度为便于阅读和理解,单个函数的有效代码长度当尽量控制在300行以内(不包括注释行),当一个功能模块过大时往往造成阅读困难,因此当使用子函数等将相应功能抽取出来,这也有利于提高代码的重用度,尽量避免使用大类和长方法。4.3. 行宽页宽应该设置为100字符。一般不要超过这个宽度, 这会导致在某些机器中无法以一屏来完整显示, 但这一设置也可以灵活调整。在任何情况下, 超长的语句应该在一个逗号后或一个操作符前折行。一条语句折行后, 应该比原来的语句再缩进一个TAB或4个空格,以便于阅读。4.4. 间隔类、方法及功能块间等应以空行相隔,以增加可读性,但不得有无规则的大片空行;操作符两端应当各空一个字符以增加可读性;相应独立的功能模块之间可使用注释行间隔,并标明相应内容。4.5. 括号 中的语句应该单独作为一行,左括号""当紧跟其语句后,右括号""永远单独作为一行且与其匹配行对齐,并尽量在其后说明其匹配的功能模块。较长的方法以及类、接口等的右括号后应使用/end .等标识其结束。如:类的结束符:/EOC ClassName,方法结束符:/end methodName(),功能块结束:/end if.userName is null?循环快结束:/end for.every user in userList不要在程序中出现不必要的括号,但有时为了增加可读性和便于理解,当用括号限定相应项。5. 性能与安全总体描述 5.1. 通用性能优化5.1.1. String与StringBugffer不要使用如下String初始化方法:String str = new String(“abcdef”);这将产生两个对象,应当直接赋值:String str = “abcdef”;在处理可变 String 的时候要尽量使用 StringBuffer 类,StringBuffer 类是构成 String 类的基础。String 类将 StringBuffer 类封装了起来,(以花费更多时间为代价)为开发人员提供了一个安全的接口。当我们在构造字符串的时候,我们应该用 StringBuffer 来实现大部分的工作,当工作完成后将 StringBuffer 对象再转换为需要的 String 对象。比如:如果有一个字符串必须不断地在其后添加许多字符来完成构造,那么我们应该使用 StringBuffer 对象和她的 append() 方法。如果我们用 String 对象代替 StringBuffer 对象的话,将会花费许多不必要的创建和释放对象的 CPU 时间。5.1.2. 集合避免使用Vector和HashTable等旧的集合实现,这些实现的存在仅是为了与旧的系统兼容,而且由于这些实现是同步的,故而在大量操作时会带来不必要的性能损失。在新的系统设计中不当出现这些实现,使用ArrayList代替Vector,使用HashMap代替HashTable。若却是需要使用同步集合类,当使用如下方式获得同步集合实例:Map map = Collections.synchronizedMap(new HashMap();由于数组、ArrayList与Vector之间的性能差异巨大,故在能使用数组时不要使用ArrayList,尽量避免使用Vector。5.1.3. 对象避免在循环中频繁构建和释放对象。不再使用的对象应及时销毁。5.1.4. final将参数或方法声明成final可提高程序响应效率,故此:注意绝对不要仅因为性能而将类、方法等声明成final,声明成final的类、方法一定要确信不再被继承或重载!不需要重新赋值的变量(包括类变量、实例变量、局部变量)声明成final所有方法参数声明成final私有(private)方法不需要声明成final若方法确定不会被继承,则声明成final5.1.5. abstractabstract修饰符用来修饰类和成员方法 1:用abstract修饰的类表示抽象类,抽象类位于继承树的抽象层,抽象类不能被实例化。 2:用abstract修饰的方法表示抽象方法,抽象方法没有方法体。抽象方法用来描述系统具有什么功能,但 不提供具体的实现。 abstract 规则: 1:抽象类可以没有抽象方法,但是有抽象方法的类必须定义为抽象类,如果一个子类继承一个抽象类,子类没有实现父类的所有抽象方法,那么子类也要定义为抽象类,否则的话编译会出错的。 2:抽象类没有构造方法,也没有抽象静态方法。但是可以有非抽象的构造方法 3:抽象类不能被实例化,但是可以创建一个引用变量,类型是一个抽象类,并让它引用非抽象类的子类的一个实例 4:不能用final 修饰符修饰5.1.6. 垃圾收集和资源释放不要过分依赖JVM的垃圾收集机制,因为你无法预测和知道JVM在什么时候运行GC。尽可能早的释放资源,不再使用的资源请立即释放。5.1.7. 慎用异常异常对性能不利。抛出异常首先要创建一个新的对象。Throwable接口的构造函数调用名为fillInStackTrace()的本地(Native)方法,fillInStackTrace()方法检查堆栈,收集调用跟踪信息。只要有异常被抛出,VM就必须调整调用堆栈,因为在处理过程中创建了一个新的对象。 异常只能用于错误处理,不应该用来控制程序流程。5.1.8. QueryHandle.findByQuery()方法规范注意一下代码红色部分,由于传入id值为空,则QueryHandle会认为此条件无效而直接跳过,导致的情况则是加载了全表,若次势必会造成服务器内存泄露。故在代码编写过程中对于必输值需要判断是否为空,为空则赋值空格” ”以避免加载全表情况的发生。QueryInfo quer = new QueryInfo(BookingHead.class);quer.setListType(QueryInfo.LISTTYPE_LIST);Field field = new Field();field.setFieldName(BookingHeadDomain.bookingHeadId);field.setFieldValue(StringUtils.isEmpty(id)?" ":id);quer.addField(field);List<String> lstProps = new ArrayList<String>();lstProps.add(BookingHeadDomain.bkCargos);List<?> result = (List<?>) QueryHandle.findByQuery(quer,lstProps);