阿里妹导读:DonRoberts提出的一条重构准则:第一次做某件事时只管去做;第二次做类似的事时会产生反感,但无论如何还是可以去做;第三次再做类似的事时,你就应该重构。
编码也是如此,当多次编写类似的代码时,我们需要考虑是否有一种***能够提高编码速度,让编码速度“起飞”?高德地图技术专家陈昌毅(常意)多年来致力于敏捷开发,总结了一套编码的***论,有助于程序员"快速、优质、高效"地进行编码。
***1:手工编写代码大多数刚学习Java的程序员,都会怀着一种崇敬的仪式感,一字一句地在开发工具上敲出以下代码:
publicclassTest{publicstaticvoidmain(String[]args){System.out.println("Helloworld!");}}没错,这就是经典的"Helloworld",这也是大多数人手工编写的第一个程序。
手工编写代码,更能体现一个程序员的基本素质。有很多公司,都把上机编程考试作为面试的重要手段之一。面试者需要根据题目的要求,挑选一款熟悉的编程工具(比如Eclipse),手工编写代码并调试运行通过。在整个过程中,不能通过网络搜索答案,不能查看联机帮助文档,要求面试者必须手工编写代码,主要是考察面试者手工编写代码的能力——语法、函数、逻辑、思维、算法以及动手能力。
手工编写代码,是一个优秀程序员必须具备的基础能力。手工编写代码正如提笔写文章,语法就是遣词造句的***、函数就是组成文章的词句、类库就是据经引典的掌故、架构就是行文表述的体裁、功能就是写作文章的主旨、算法就是组织语言的逻辑……所以,只要掌握一门程序语言的语法、学习一堆基础类库的函数、引用一些所需的第三方类库、选择一款成熟稳定的架构、明确一下产品需求的功能、挑选一种实现逻辑的算法……手工编写代码就会像写文章一样手到擒来。
***2:复制粘贴代码常言道:"熟读唐诗三百首,不会作诗也会吟。"编码也是同样的道理,编码的第一步就是模仿,简单地说就是"抄代码"——复制粘贴代码。复制粘贴代码是一门艺术,用好了编码会事半功倍。但是,没有检验过的东西,终究是不可全信的。当看到需要的代码时,在复制粘贴前,我们都需要仔细研读、认真思考、详细甄别……很多东西,都是仁者见仁、智者见智的东西,适合别的场景但不一定适合你的场景。作为一名合格的程序员,切不可一味地"拿来主义"。
1.为什么要复制粘贴代码
复制粘贴现有代码,可以节省开发时间;复制粘贴稳定代码,可以降低系统故障风险;复制粘贴网络代码,可以把别人的成果化为己用。2.复制粘贴代码带来问题
你对复制的代码理解程度是多少?实现逻辑是否合理?能不能稳定运行?存在多少潜在的Bug?这个代码在项目中已经复制粘贴了多少次?根据“三则重构”原则,你是否需要对这些相同代码进行重构?代码被复制粘贴次数越多,带来的代码维护问题越多。多个代码版本的更改和修正,要保持这些代码的同步,就必须需要在每一处进行同样的修改,增加了维护的成本和风险。总之,复制粘贴代码,跟其它编码***一样,没有优劣对错之分。它只是一种***,你可以善用,也可以滥用。如果我们用到了复制粘贴,我们就必须为结果负责。
***3:用文本替换生成代码1.生成代码样例
已经编写好的用户查询相关代码:
/**查询用户服务函数*/publicPageData<UserVO>queryUser(QueryUserParameterVOparameter){LongtotalCount=userDAO.countByParameter(parameter);List<UserVO>userList=null;if(Objects.nonNull(totalCount)&&totalCount.compareTo(0L)>0){userList=userDAO.queryByParameter(parameter);}returnnewPageData<>(totalCount,userList);}/**查询用户控制器函数*/@RequestMapping(path="/queryUser",method=RequestMethod.POST)publicResult<PageData<UserVO>>queryUser(@Valid@RequestBodyQueryUserParameterVOparameter){PageData<UserVO>pageData=userService.queryUser(parameter);returnResult.success(pageData);}如果我们要编写公司查询相关代码,其代码形式与用户查询类似,整理出替换关系如下:
把"用户"替换为"公司";把"User"替换为"Company";把"user"替换为"company"。利用Notepad、EditPlus等文本编辑器,选择区分大小写,进行普通文本替换,最终得到结果如下:
/**查询公司服务函数*/publicPageData<CompanyVO>queryCompany(QueryCompanyParameterVOparameter){LongtotalCount=companyDAO.countByParameter(parameter);List<CompanyVO>companyList=null;if(Objects.nonNull(totalCount)&&totalCount.compareTo(0L)>0){companyList=companyDAO.queryByParameter(parameter);}returnnewPageData<>(totalCount,companyList);}/**查询公司控制器函数*/@RequestMapping(path="/queryCompany",method=RequestMethod.POST)publicResult<PageData<CompanyVO>>queryCompany(@Valid@RequestBodyQueryCompanyParameterVOparameter){PageData<CompanyVO>pageData=companyService.queryCompany(parameter);returnResult.success(pageData);}利用文本替换生成代码,整段代码生成时间不会超过1分钟。
2.主要优缺点
主要优点:
生成代码速度较快。主要缺点:
必须编写样例代码;只适用于文本替换的情景。***4:用Excel公式生成代码Excel的公式非常强悍,可以用于编写一些公式化的代码。
1.利用Excel公式生成模型类
从WIKI上拷贝接口模型定义到Excel里,样例数据内容如下:
编写Excel公式如下:
="/**"&D6&IF(ISBLANK(F6),"","("&F6&")")&"*/"&IF(E6="否",IF(C6="String","@NotBlank","@NotNull"),"")&"private"&C6&""&B6&";"利用公式生成代码如下:
/**用户标识*/@NotNullprivateLongid;/**用户名称*/@NotBlankprivateStringname;/**用户性别(0:未知;1:男;2:女)*/@NotNullprivateIntegersex;/**用户描述*/privateStringdescription;创建模型类,整理代码如下:
/**用户DO类*/publicclassUserDO{/**用户标识*/@NotNullprivateLongid;/**用户名称*/@NotBlankprivateStringname;/**用户性别(0:未知;1:男;2:女)*/@NotNullprivateIntegersex;/**用户描述*/privateStringdescription;......}2.利用Excel公式生成枚举类
从WIKI上拷贝枚举定义到Excel里,样例数据内容如下:
编写Excel公式如下:
="/**"&D2&"("&B2&")*/"&C2&"("&B2&","""&D2&"""),"利用公式生成代码如下:
/**空(0)*/NONE(0,"空"),/**男(1)*/MAN(1,"男"),/**女(2)*/WOMAN(2,"女"),创建枚举类,整理代码如下:
/**用户性别枚举*/publicenumUserSex{/**枚举定义*//**空(0)*/NONE(0,"空"),/**男(1)*/MAN(1,"男"),/**女(2)*/WOMAN(2,"女");......}3.利用Excel公式生成数据库语句
用Excel整理的公司列表如下,需要整理成SQL语句直接插入数据库:
编写Excel公式如下:
="('"&B2&"','"&C2&"','"&D2&"','"&E2&"'),"利用公式生成SQL如下:
('高德','首开大厦','(010)11111111','gaode@xxx.com'),('阿里云','绿地中心','(010)22222222','aliyun@xxx.com'),('菜鸟','阿里中心','(010)33333333','cainiao@xxx.com'),添加into语句头,整理SQL如下:
insertintot_company(name,address,phone,email)values('高德','首开大厦','(010)11111111','gaode@xxx.com'),('阿里云','绿地中心','(010)22222222','aliyun@xxx.com'),('菜鸟','阿里中心','(010)33333333','cainiao@xxx.com');4.主要优缺点
主要优点:
适用于表格化数据的代码生成;写好公式后,拖拽生成代码,生成速度较快。主要缺点:
不适用于复杂功能的代码生成。***5:用工具生成代码用工具生成代码,顾名思义就是借用已有的工具生成代码。很多开发工具都提供一些工具生成代码,比如:生成构造函数,重载基类/接口函数,生成Getter/Setter函数,生成toString函数……能够避免很多手敲代码。还有一些生成代码插件,也可以生成满足某些应用场景的代码。
这里以mybatis-generator插件生成代码为例,介绍如何利用工具生成代码。
1.安装运行插件
具体***这里不再累述,自行上网搜索文档了解。
2.生成代码样例
|2.1.生成模型类代码
文件User.java内容:
......publicclassUser{privateLongid;privateStringuser;privateStringpassword;privateIntegerage;......}|2.2.生成映射接口代码
文件UserMapper.java内容:
......publicinterfaceUserMapper{UserselectByPrimaryKey(Longid);......}|2.3.生成映射XML代码
文件UserMapper.xml内容:
......<mappernamespace="com.test.dao.UserMapper"><resultMapid="BaseResultMap"type="com.test.pojo.User"><idcolumn="id"property="id"jdbcType="BIGINT"/><resultcolumn="user"property="user"jdbcType="VARCHAR"/><resultcolumn="password"property="password"jdbcType="VARCHAR"/><resultcolumn="age"property="age"jdbcType="INTEGER"/></resultMap><sqlid="Base_Column_List">id,user,password,age</sql><selectid="selectByPrimaryKey"resultMap="BaseResultMap"parameterType="java.lang.Long">select<includerefid="Base_Column_List"/>fromtest_userwhereid=#{id,jdbcType=BIGINT}</select>......</mapper>3.主要优缺点
主要优点:
利用生成代码插件,生成代码速度较快;利用插件配置文件,控制生成想要的功能代码。主要缺点:
需要时间研究和熟悉生成代码插件的使用;生成的代码不一定满足代码规范,每次生成后需进行代码合规;重新生成代码后,容易覆盖自定义代码(建议维护单独的生成代码库,通过DIFF工具比较代码差异,然后再赋值粘贴差异代码)。***6:用代码生成代码用代码生成代码,就是自己编写代码,按照自己的格式生成代码。下面,以生成基于MyBatis的数据库访问代码为例说明。
1.查询表格信息
首先,我们要从数据库中拿到我们生成代码所需要的表和列相关信息。
|1.1.查询表信息
查询表信息语句:
selectt.table_nameas'表名称',t.table_commentas'表备注'frominformation_schema.tablestwheret.table_schema=?andt.table_type='BASETABLE'andt.table_name=?;其中,第1个问号赋值数据库名称,第2个问号赋值表名称。
查询表信息结果:
|1.2.查询列信息
查询列信息语句:
selectc.column_nameas'列名称',c.column_commentas'列备注',c.data_typeas'数据类型',c.character_maximum_lengthas'字符长度',c.numeric_precisionas'数字精度',c.numeric_scaleas'数字范围',c.column_defaultas'',c.is_nullableas'是否可空',c.column_keyas'列键名'frominformation_schema.columnscwherec.table_schema=?andc.table_name=?orderbyc.ordinal_position;其中,第1个问号赋值数据库名称,第2个问号赋值表名称。
查询列信息结果:
2.编写生成代码
|2.1.编写生成模型类代码
/**生成模型类文件函数*/privatevoidgenerateModelClassFile(Filedir,Tabletable,List<Column>columnList)throwsException{try(PrintWriterwriter=newPrintWriter(newFile(dir,className+"DO.java"))){StringclassName=getClassName(table.getTableName());StringclassComments=getClassComment(table.getTableComment());writer.println("package"+groupName+"."+systemName+".database;");......writer.println("/**"+classComments+"DO类*/");writer.println("@Getter");writer.println("@Setter");writer.println("@ToString");writer.println("publicclass"+className+"DO{");for(Columncolumn:columnList){StringfieldType=getFieldType(column);StringfieldName=getFieldName(column.getColumnName());StringfieldComment=getFieldComment(column);writer.println(" /**"+fieldComment+"*/");writer.println(" private"+fieldType+""+fieldName+";");}writer.println("}");}}|2.2.编写生成DAO接口代码
/**生成DAO接口文件函数*/privatevoidgenerateDaoInterfaceFile(Filedir,Tabletable,List<Column>columnList,List<Column>pkColumnList)throwsException{try(PrintWriterwriter=newPrintWriter(newFile(dir,className+"DAO.java"))){StringclassName=getClassName(table.getTableName());StringclassComments=getClassComment(table.getTableComment());writer.println("package"+groupName+"."+systemName+".database;");......writer.println("/**"+classComments+"DAO接口*/");writer.println("publicinterface"+className+"DAO{");writer.println(" /**获取"+classComments+"函数*/");writer.print(" public"+className+"DOget(");booleanisFirst=true;for(ColumnpkColumn:pkColumnList){if(!isFirst){writer.print(",");}else{isFirst=false;}StringfieldType=getFieldType(pkColumn);StringfieldName=getFieldName(pkColumn.getColumnName());writer.print("@Param(""+fieldName+"")"+fieldType+""+fieldName);}writer.println(");");......writer.println("}");}}|2.3.编写生成DAO映射代码
/**生成DAO映射文件函数*/privatevoidgenerateDaoMapperFile(Filedir,Tabletable,List<Column>columnList,List<Column>pkColumnList)throwsException{try(PrintWriterwriter=newPrintWriter(newFile(dir,className+"DAO.xml"))){StringclassName=getClassName(table.getTableName());StringclassComments=getClassComment(table.getTableComment());writer.println("<?xmlversion="1.0"encoding="UTF-8"?>");......writer.println("<!--"+classComments+"映射-->");writer.println("<mappernamespace=""+groupName+"."+systemName+".database."+className+"DAO">");writer.println(" <!--所有字段语句-->");writer.println(" <sqlid="fields">");if(CollectionUtils.isNotEmpty(columnList)){booleanisFirst=true;StringcolumnName=getColumnName(pkColumn.getColumnName());for(Columncolumn:columnList){if(isFirst){isFirst=false;writer.println(" "+columnName);}else{writer.println(" ,"+columnName);}}}writer.println(" </sql>");writer.println(" <!--获取"+classComments+"函数语句-->");writer.println(" <selectid="get"resultType=""+groupName+"."+systemName+".database."+className+"DO">");writer.println(" select");writer.println(" <includerefid="fields"/>");writer.println(" from"+table.getTableName());booleanisFirst=true;for(ColumnpkColumn:pkColumnList){StringcolumnName=getColumnName(pkColumn.getColumnName());StringfieldName=getFieldName(pkColumn.getColumnName());writer.print(" ");if(isFirst){writer.print("where");isFirst=false;}else{writer.print("and");}writer.println(""+columnName+"=#{"+fieldName+"}");}writer.println(" </select>");writer.println("</mapper>");}}3.生成相关代码
|3.1.生成的模型类代码
/**组织公司DO类*/@Getter@Setter@ToStringpublicclassOrgCompanyDO{/**公司标识*/privateLongid;/**公司名称*/privateStringname;/**联系地址*/privateStringaddress;/**公司描述*/privateStringdescription;}|3.2.生成的DAO接口代码
/**组织公司DAO接口*/publicinterfaceOrgCompanyDAO{/**获取组织公司函数*/publicOrgCompanyDOget(@Param("id")Longid);}|3.3.生成的DAO映射代码
<!--组织公司映射--><mappernamespace="xxx.database.OrgCompanyDAO"><!--所有字段语句--><sqlid="fields">id,name,address,description</sql><!--获取组织公司函数语句--><selectid="get"resultType="xxx.database.OrgCompanyDO">select<includerefid="fields"/>fromorg_companywhereid=#{id}</select></mapper>3.主要优缺点
主要优点:
代码格式可以定制,保证生成代码合规;代码功能可以定制,只生成需要的代码;经过前期代码沉淀后,后期能够直接使用。主要缺点:
需要研究数据来源,保证能获取到生成代码所需的数据;需要建立数据模型、编写生成代码,耗费时间比较长。终极***:无招胜有招编码的终极***,是不是直接对着电脑说需求,然后电脑就自动生成代码了?未来科技发展到一定水平后,这种情况或许会变成现实。但是,目前这种情况是不现实的。现实中,想要做到"大口一张、代码就来",除非你是老板、产品经理或者技术管理者。
编码的终极***是“无招胜有招”,"无招"并不是不讲究"招式",而是不拘泥于某一"招式",信手拈来合适的"招式"为宜。本文中列举的各种编码***,没有高低优劣之分,只有合不合适之说。所以,灵活地运用各种编码***,就是编码的终极***。
代码规范化在上面的各种编码***中,很多***都需要手工编写样例代码。如果你的代码不遵循代码规范,就很难发现代码之间的共性,并抽象出能够作为标准的样例代码;如果作为标准的样例代码不满足代码规范,必然导致生成的代码也不满足代码规范,于是把这些不规范放大了十倍、百倍甚至千倍。所以,代码规范化是编码的重中之重。
作者:陈昌毅