Don Roberts提出的一条重构准则:java
第一次作某件事时只管去作;第二次作相似的事时会产生反感,但不管如何仍是能够去作;第三次再作相似的事时,你就应该重构。程序员
编码也是如此,当屡次编写相似的代码时,咱们须要考虑是否有一种方法可以提升编码速度。做者多年来致力于敏捷开发,总结了一套编码的方法论,有助于程序员"快速、优质、高效"地进行编码。面试
大多数刚学习Java的程序员,都会怀着一种崇敬的仪式感,一字一句地在开发工具上敲出如下代码:算法
public class Test { public static void main(String[] args) { System.out.println("Hello world!"); } }
没错,这就是经典的"Hello world",这也是大多数人手工编写的第一个程序。sql
手工编写代码,更能体现一个程序员的基本素质。有不少公司,都把上机编程考试做为面试的重要手段之一。面试者须要根据题目的要求,挑选一款熟悉的编程工具(好比Eclipse),手工编写代码并调试运行经过。在整个过程当中,不能经过网络搜索答案,不能查看联机帮助文档,要求面试者必须手工编写代码,主要是考察面试者手工编写代码的能力——语法、函数、逻辑、思惟、算法以及动手能力。数据库
手工编写代码,是一个优秀程序员必须具有的基础能力。手工编写代码正如提笔写文章,语法就是遣词造句的方法、函数就是组成文章的词句、类库就是据经引典的掌故、架构就是行文表述的体裁、功能就是写做文章的主旨、算法就是组织语言的逻辑……因此,只要掌握一门程序语言的语法、学习一堆基础类库的函数、引用一些所需的第三方类库、选择一款成熟稳定的架构、明确一下产品需求的功能、挑选一种实现逻辑的算法……手工编写代码就会像写文章同样手到擒来。编程
常言道:"熟读唐诗三百首,不会做诗也会吟。"编码也是一样的道理,编码的第一步就是模仿,简单地说就是"抄代码"——复制粘贴代码。复制粘贴代码是一门艺术,用好了编码会事半功倍。可是,没有检验过的东西,终究是不可全信的。当看到须要的代码时,在复制粘贴前,咱们都须要仔细研读、认真思考、详细甄别……不少东西,都是仁者见仁、智者见智的东西,适合别的场景但不必定适合你的场景。做为一名合格的程序员,切不可一味地"拿来主义"。网络
总之,复制粘贴代码,跟其它编码方法同样,没有优劣对错之分。它只是一种方法,你能够善用,也能够滥用。若是咱们用到了复制粘贴,咱们就必须为结果负责。mybatis
已经编写好的用户查询相关代码:架构
/** 查询用户服务函数 */ public PageData<UserVO> queryUser(QueryUserParameterVO parameter) { Long totalCount = userDAO.countByParameter(parameter); List<UserVO> userList = null; if (Objects.nonNull(totalCount) && totalCount.compareTo(0L) > 0) { userList = userDAO.queryByParameter(parameter); } return new PageData<>(totalCount, userList); } /** 查询用户控制器函数 */ @RequestMapping(path = "/queryUser", method = RequestMethod.POST) public Result<PageData<UserVO>> queryUser(@Valid @RequestBody QueryUserParameterVO parameter) { PageData<UserVO> pageData = userService.queryUser(parameter); return Result.success(pageData); }
若是咱们要编写公司查询相关代码,其代码形式与用户查询相似,整理出替换关系以下:
利用Notepad、EditPlus等文本编辑器,选择区分大小写,进行普通文本替换,最终获得结果以下:
/** 查询公司服务函数 */ public PageData<CompanyVO> queryCompany(QueryCompanyParameterVO parameter) { Long totalCount = companyDAO.countByParameter(parameter); List<CompanyVO> companyList = null; if (Objects.nonNull(totalCount) && totalCount.compareTo(0L) > 0) { companyList = companyDAO.queryByParameter(parameter); } return new PageData<>(totalCount, companyList); } /** 查询公司控制器函数 */ @RequestMapping(path = "/queryCompany", method = RequestMethod.POST) public Result<PageData<CompanyVO>> queryCompany(@Valid @RequestBody QueryCompanyParameterVO parameter) { PageData<CompanyVO> pageData = companyService.queryCompany(parameter); return Result.success(pageData); }
利用文本替换生成代码,整段代码生成时间不会超过1分钟。
主要优势:
主要缺点:
Excel的公式很是强悍,能够用于编写一些公式化的代码。
从WIKI上拷贝接口模型定义到Excel里,样例数据内容以下:
A | B | C | D | E | F | |
---|---|---|---|---|---|---|
1 | 序号 | 字段名称 | 字段类型 | 字段描述 | 是否可空 | 附加信息 |
2 | 1 | id | Long | 用户标识 | 否 | |
3 | 2 | name | String | 用户名称 | 否 | |
4 | 3 | sex | Integer | 用户性别 | 否 | 0:未知;1:男;2:女 |
5 | 4 | description | String | 用户描述 | 是 |
编写Excel公式以下:
= "/** "&D6&IF(ISBLANK(F6), "", "("&F6&")")&" */ "&IF(E6 = "否", IF(C6 = "String", "@NotBlank", "@NotNull"), "")&" private "&C6&" "&B6&";"
利用公式生成代码以下:
/** 用户标识 */ @NotNull private Long id; /** 用户名称 */ @NotBlank private String name; /** 用户性别(0:未知;1:男;2:女) */ @NotNull private Integer sex; /** 用户描述 */ private String description;
建立模型类,整理代码以下:
/** 用户DO类 */ public class UserDO { /** 用户标识 */ @NotNull private Long id; /** 用户名称 */ @NotBlank private String name; /** 用户性别(0:未知;1:男;2:女) */ @NotNull private Integer sex; /** 用户描述 */ private String description; ...... }
从WIKI上拷贝枚举定义到Excel里,样例数据内容以下:
A | B | C | D | |
---|---|---|---|---|
1 | 序号 | 字段取值 | 字段名称 | 字段描述 |
2 | 1 | 0 | NONE | 空 |
3 | 2 | 1 | MAN | 男 |
4 | 3 | 2 | WOMAN | 女 |
编写Excel公式以下:
="/** "&D2&"("&B2&") */"&C2&"("&B2&", """&D2&"""),"
利用公式生成代码以下:
/** 空(0) */NONE(0, "空"), /** 男(1) */MAN(1, "男"), /** 女(2) */WOMAN(2, "女"),
建立枚举类,整理代码以下:
/** 用户性别枚举 */ public enum UserSex { /** 枚举定义 */ /** 空(0) */ NONE(0, "空"), /** 男(1) */ MAN(1, "男"), /** 女(2) */ WOMAN(2, "女"); ...... }
用Excel整理的公司列表以下,须要整理成SQL语句直接插入数据库:
A | B | C | D | E | |
---|---|---|---|---|---|
1 | 序号 | 公司名称 | 公司地址 | 公司电话 | 公司邮箱 |
2 | 1 | 高德 | 首开大厦 | (010)11111111 | gaode@xxx.com |
3 | 2 | 阿里云 | 绿地中心 | (010)22222222 | aliyun@xxx.com |
4 | 3 | 菜鸟 | 阿里中心 | (010)33333333 | cainiao@xxx.com |
编写Excel公式以下:
= "('"&B2&"', '"&C2&"', '"&D2&"', '"&E2&"'),"
利用公式生成SQL以下:
('高德', '首开大厦', '(010)11111111', 'gaode@xxx.com'), ('阿里云', '绿地中心', '(010)22222222', 'aliyun@xxx.com'), ('菜鸟', '阿里中心', '(010)33333333', 'cainiao@xxx.com'),
添加into语句头,整理SQL以下:
insert into t_company(name, address, phone, email) values ('高德', '首开大厦', '(010)11111111', 'gaode@xxx.com'), ('阿里云', '绿地中心', '(010)22222222', 'aliyun@xxx.com'), ('菜鸟', '阿里中心', '(010)33333333', 'cainiao@xxx.com');
主要优势:
主要缺点:
用工具生成代码,顾名思义就是借用已有的工具生成代码。不少开发工具都提供一些工具生成代码,好比:生成构造函数,重载基类/接口函数,生成Getter/Setter函数,生成toString函数……可以避免不少手敲代码。还有一些生成代码插件,也能够生成知足某些应用场景的代码。
这里以mybatis-generator插件生成代码为例,介绍如何利用工具生成代码。
具体方法这里再也不累述,自行上网搜索文档了解。
文件User.java内容:
...... public class User { private Long id; private String user; private String password; private Integer age; ...... }
文件UserMapper.java内容:
...... public interface UserMapper { User selectByPrimaryKey(Long id); ...... }
文件UserMapper.xml内容:
...... <mapper namespace="com.test.dao.UserMapper" > <resultMap id="BaseResultMap" type="com.test.pojo.User" > <id column="id" property="id" jdbcType="BIGINT" /> <result column="user" property="user" jdbcType="VARCHAR" /> <result column="password" property="password" jdbcType="VARCHAR" /> <result column="age" property="age" jdbcType="INTEGER" /> </resultMap> <sql id="Base_Column_List" > id, user, password, age </sql> <select id="selectByPrimaryKey" resultMap="BaseResultMap" parameterType="java.lang.Long" > select <include refid="Base_Column_List" /> from test_user where id = #{id,jdbcType=BIGINT} </select> ...... </mapper>
主要优势:
主要缺点:
用代码生成代码,就是本身编写代码,按照本身的格式生成代码。下面,以生成基于MyBatis的数据库访问代码为例说明。
首先,咱们要从数据库中拿到咱们生成代码所须要的表和列相关信息。
查询表信息语句:
select t.table_name as '表名称' , t.table_comment as '表备注' from information_schema.tables t where t.table_schema = ? and t.table_type = 'BASE TABLE' and t.table_name = ?;
其中,第1个问号赋值数据库名称,第2个问号赋值表名称。
查询表信息结果:
序号 | 表名称 | 表备注 |
---|---|---|
1 | org_company | 组织公司表 |
查询列信息语句:
select c.column_name as '列名称' , c.column_comment as '列备注' , c.data_type as '数据类型' , c.character_maximum_length as '字符长度' , c.numeric_precision as '数字精度' , c.numeric_scale as '数字范围' , c.column_default as '' , c.is_nullable as '是否可空' , c.column_key as '列键名' from information_schema.columns c where c.table_schema = ? and c.table_name = ? order by c.ordinal_position;
其中,第1个问号赋值数据库名称,第2个问号赋值表名称。
查询列信息结果:
序号 | 列名称 | 列备注 | 数据类型 | 字符长度 | 数字精度 | 数字范围 | 是否可空 | 列键名 |
---|---|---|---|---|---|---|---|---|
1 | id | 公司标识 | bigint | 19 | 0 | NO | PRI | |
2 | name | 公司名称 | varchar | 50 | NO | |||
3 | address | 联系地址 | varchar | 200 | YES | |||
4 | description | 公司描述 | text | 65535 | YES |
/** 生成模型类文件函数 */ private void generateModelClassFile(File dir, Table table, List<Column> columnList) throws Exception { try (PrintWriter writer = new PrintWriter(new File(dir, className + "DO.java"))) { String className = getClassName(table.getTableName()); String classComments = getClassComment(table.getTableComment()); writer.println("package " + groupName + "." + systemName + ".database;"); ...... writer.println("/** " + classComments + "DO类 */"); writer.println("@Getter"); writer.println("@Setter"); writer.println("@ToString"); writer.println("public class " + className + "DO {"); for (Column column : columnList) { String fieldType = getFieldType(column); String fieldName = getFieldName(column.getColumnName()); String fieldComment = getFieldComment(column); writer.println("\t/** " + fieldComment + " */"); writer.println("\tprivate " + fieldType + " " + fieldName + ";"); } writer.println("}"); } }
/** 生成DAO接口文件函数 */ private void generateDaoInterfaceFile(File dir, Table table, List<Column> columnList, List<Column> pkColumnList) throws Exception { try (PrintWriter writer = new PrintWriter(new File(dir, className + "DAO.java"))) { String className = getClassName(table.getTableName()); String classComments = getClassComment(table.getTableComment()); writer.println("package " + groupName + "." + systemName + ".database;"); ...... writer.println("/** " + classComments + "DAO接口 */"); writer.println("public interface " + className + "DAO {"); writer.println("\t/** 获取" + classComments + "函数 */"); writer.print("\tpublic " + className + "DO get("); boolean isFirst = true; for (Column pkColumn : pkColumnList) { if (!isFirst) { writer.print(", "); } else { isFirst = false; } String fieldType = getFieldType(pkColumn); String fieldName = getFieldName(pkColumn.getColumnName()); writer.print("@Param(\"" + fieldName + "\") " + fieldType + " " + fieldName); } writer.println(");"); ...... writer.println("}"); } }
/** 生成DAO映射文件函数 */ private void generateDaoMapperFile(File dir, Table table, List<Column> columnList, List<Column> pkColumnList) throws Exception { try (PrintWriter writer = new PrintWriter(new File(dir, className + "DAO.xml"))) { String className = getClassName(table.getTableName()); String classComments = getClassComment(table.getTableComment()); writer.println("<?xml version=\"1.0\" encoding=\"UTF-8\"?>"); ...... writer.println("<!-- " + classComments + "映射 -->"); writer.println("<mapper namespace=\"" + groupName + "." + systemName + ".database." + className + "DAO\">"); writer.println("\t<!-- 全部字段语句 -->"); writer.println("\t<sql id=\"fields\">"); if (CollectionUtils.isNotEmpty(columnList)) { boolean isFirst = true; String columnName = getColumnName(pkColumn.getColumnName()); for (Column column : columnList) { if (isFirst) { isFirst = false; writer.println("\t\t" + columnName); } else { writer.println("\t\t, " + columnName); } } } writer.println("\t</sql>"); writer.println("\t<!-- 获取" + classComments + "函数语句 -->"); writer.println("\t<select id=\"get\" resultType=\"" + groupName + "." + systemName + ".database." + className + "DO\">"); writer.println("\t\tselect"); writer.println("\t\t<include refid=\"fields\"/>"); writer.println("\t\tfrom " + table.getTableName()); boolean isFirst = true; for (Column pkColumn : pkColumnList) { String columnName = getColumnName(pkColumn.getColumnName()); String fieldName = getFieldName(pkColumn.getColumnName()); writer.print("\t\t"); if (isFirst) { writer.print("where"); isFirst = false; } else { writer.print("and"); } writer.println(" " + columnName + " = #{" + fieldName + "}"); } writer.println("\t</select>"); writer.println("</mapper>"); } }
/** 组织公司DO类 */ @Getter @Setter @ToString public class OrgCompanyDO { /** 公司标识 */ private Long id; /** 公司名称 */ private String name; /** 联系地址 */ private String address; /** 公司描述 */ private String description; }
/** 组织公司DAO接口 */ public interface OrgCompanyDAO { /** 获取组织公司函数 */ public OrgCompanyDO get(@Param("id") Long id); }
<!-- 组织公司映射 --> <mapper namespace="xxx.database.OrgCompanyDAO"> <!-- 全部字段语句 --> <sql id="fields"> id , name , address , description </sql> <!-- 获取组织公司函数语句 --> <select id="get" resultType="xxx.database.OrgCompanyDO"> select <include refid="fields"/> from org_company where id = #{id} </select> </mapper>
主要优势:
主要缺点:
编码的终极方法,是否是直接对着电脑说需求,而后电脑就自动生成代码了?将来科技发展到必定水平后,这种状况或许会变成现实。可是,目前这种状况是不现实的。现实中,想要作到"大口一张、代码就来",除非你是老板、产品经理或者技术管理者。
编码的终极方法是“无招胜有招”,"无招"并非不讲究"招式",而是不拘泥于某一"招式",信手拈来合适的"招式"为宜。本文中列举的各类编码方法,没有高低优劣之分,只有合不合适之说。因此,灵活地运用各类编码方法,就是编码的终极方法。
在上面的各类编码方法中,不少方法都须要手工编写样例代码。若是你的代码不遵循代码规范,就很难发现代码之间的共性,并抽象出可以做为标准的样例代码;若是做为标准的样例代码不知足代码规范,必然致使生成的代码也不知足代码规范,因而把这些不规范放大了十倍、百倍甚至千倍。 因此,代码规范化是编码的重中之重。
请参考阿里集团的开发规约:
我写的一些代码规范化建议:
在构思这篇文章的时候,在网上看见这么一个梗:一位网友讽刺一位阿里人的简历,满篇都是"沉淀了一套XX方法论,为XX业务赋能。",用了流行语"赋能"显得很"高大上"。姑且不论他的简历如何,可以从方法论上着手的人,必定有值得咱们学习的地方。这里,我也来蹭一下这个梗,就取一个高大上的名字《编码方法论,赋能你我他》。
原文连接 本文为云栖社区原创内容,未经容许不得转载。