当前位置: 首页 > news >正文

阿里巴巴编程规范实战(一):编程规约之常量定义代码格式

1.【强制】不允许任何魔法值(即未经预先定义的常量)直接出现在代码中。

反例:String key = "Id#taobao_" + tradeId;

cache.put(key, value);

// 缓存 get 时,由于在代码复制时,漏掉下划线,导致缓存击穿而出现问题

说明:我经历很多的项目中大部分都能看到魔法值,其中包含两家互联网公司,狗东上线前必须代码评审,会规避掉一些,也是没有办法杜绝,CodeReview的时候大家都知道这个点,但是实际开发的过程中,特别是写业务代码,随手一丢,满满的魔法,码处高效啊,所以学习手册还是有很必要的,不然CodeReview会被diss。

2.【强制】 long 或者 Long 赋值时,数值后使用大写的 L,不能是小写的 l,小写容易跟数

字 1 混淆,造成误解。

说明:Long a = 2l; 写的是数字的 21,还是 Long 型的 2

3.【推荐】不要使用一个常量类维护所有常量,要按常量功能进行归类,分开维护。

说明:大而全的常量类,杂乱无章,使用查找功能才能定位到修改的常量,不利于理解和维护。

正例:缓存相关常量放在类 CacheConsts 下;系统配置相关常量放在类 ConfigConsts 下。

以下是实战中CacheKey的定义,使用了Function和BiFunction,看上去还是比较优雅的。

 public final static Function<Long,String>  DEMON_FUNCTION =  demoId -> COMMON_PREFIX.concat("demo:" + demoId);

以下是实战中阿波罗的配置定义

阿波罗中的配置也可以再分类,例如:开关配置(所有的开关都在一个文件中)、数据源配置等等。

 

4.【推荐】常量的复用层次有五层:跨应用共享常量、应用内共享常量、子工程内共享常量、

包内共享常量、类内共享常量。

1) 跨应用共享常量:放置在二方库中,通常是 client.jar 中的 constant 目录下。

2) 应用内共享常量:放置在一方库中,通常是子模块中的 constant 目录下。

反例:易懂变量也要统一定义成应用内共享常量,两位工程师在两个类中分别定义了“YES”的变量:

类 A 中:public static final String YES = "yes";

类 B 中:public static final String YES = "y";

A.YES.equals(B.YES),预期是 true,但实际返回为 false,导致线上问题。

3) 子工程内部共享常量:即在当前子工程的 constant 目录下。

4) 包内共享常量:即在当前包下单独的 constant 目录下。

5) 类内共享常量:直接在类内部 private static final 定义。

5.【推荐】如果变量值仅在一个固定范围内变化用 enum 类型来定义。

说明:如果存在名称之外的延伸属性应使用 enum 类型,下面正例中的数字就是延伸信息,表示一年中的第几个季节。

正例:

public enum SeasonEnum {

SPRING(1), SUMMER(2), AUTUMN(3), WINTER(4);

private int seq;

SeasonEnum(int seq) {

this.seq = seq;

}

public int getSeq() {

return seq;

}

}

以下是代码格式:推荐使用快捷键进行格式化,快捷方便。

6.【强制】如果是大括号内为空,则简洁地写成{}即可,大括号中间无需换行和空格;如果是非

空代码块则:

1) 左大括号前不换行。

2) 左大括号后换行。

3) 右大括号前换行。

4) 右大括号后还有 else 等代码则不换行;表示终止的右大括号后必须换行。

7.【强制】左小括号和字符之间不出现空格;同样,右小括号和字符之间也不出现空格;而左

大括号前需要空格。详见第 5 条下方正例提示。

反例:if (空格 a == b 空格)

8.【强制】if/for/while/switch/do 等保留字与括号之间都必须加空格。

9.【强制】任何二目、三目运算符的左右两边都需要加一个空格。

说明:运算符包括赋值运算符=、逻辑运算符&&、加减乘除符号等。

10.【强制】采用 4 个空格缩进,禁止使用 tab 字符。

说明:如果使用 tab 缩进,必须设置 1 tab 4 个空格。IDEA 设置 tab 4 个空格时,请勿勾选 Use tab character;而在 eclipse 中,必须勾选 insert spaces for tabs

正例: (涉及 1-5 点)

public static void main(String[] args) {

// 缩进 4 个空格

String say = "hello";

// 运算符的左右必须有一个空格

int flag = 0;

// 关键词 if 与括号之间必须有一个空格,括号内的 f 与左括号,0 与右括号不需要空格 if (flag == 0) {

System.out.println(say);

}

// 左大括号前加空格且不换行;左大括号后换行

if (flag == 1) {

System.out.println("world");

// 右大括号前换行,右大括号后有 else,不用换行

} else {

System.out.println("ok");

// 在右大括号后直接结束,则必须换行

}

}

解释:

11.【强制】注释的双斜线与注释内容之间有且仅有一个空格。

正例:

// 这是示例注释,请注意在双斜线之后有一个空格

String param = new String();

12.【强制】在进行类型强制转换时,右括号与强制转换值之间不需要任何空格隔开。

正例:

long first = 1000000000000L;

int second = (int)first + 2;

13.【强制】单行字符数限制不超过 120 个,超出需要换行,换行时遵循如下原则:

1)第二行相对第一行缩进 4 个空格,从第三行开始,不再继续缩进,参考示例。

2)运算符与下文一起换行。

3)方法调用的点符号与下文一起换行。

4)方法调用中的多个参数需要换行时,在逗号后进行。

5)在括号前不要换行,见反例

正例:

StringBuilder sb = new StringBuilder();

// 超过 120 个字符的情况下,换行缩进 4 个空格,点号和方法名称一起换行

sb.append("Jack").append("Ma")...

.append("alibaba")...

.append("alibaba")...

.append("alibaba");

反例:

StringBuilder sb = new StringBuilder();

// 超过 120 个字符的情况下,不要在括号前换行

sb.append("Jack").append("Ma")...append

("alibaba");

// 参数很多的方法调用可能超过 120 个字符,不要在逗号前换行 method(args1, args2, args3, ...

, argsX);

14.【强制】方法参数在定义和传入时,多个参数逗号后边必须加空格。

正例:下例中实参的 args1,后边必须要有一个空格。

method(args1, args2, args3);

15.【强制】IDE 的 text file encoding 设置为 UTF-8; IDE 中文件的换行符使用 Unix 格式,不要使用 Windows 格式。

解释:很多windows格式会有问题,比如记事本开头添加0xefbbbf(十六进制)的字符。会导致文本问题。

16.【推荐】单个方法的总行数不超过 80 行。

说明:除注释之外的方法签名、左右大括号、方法内代码、空行、回车及任何不可见字符的总行数不超过80行。

正例:代码逻辑分清红花和绿叶,个性和共性,绿叶逻辑单独出来成为额外方法,使主干代码更加清晰;共性逻辑抽取成为共性方法,便于复用和维护。

17.【推荐】没有必要增加若干空格来使变量的赋值等号与上一行对应位置的等号对齐。

正例:

int one = 1;

long two = 2L;

float three = 3F;

StringBuilder sb = new StringBuilder();

说明:增加 sb 这个变量,如果需要对齐,则给 one、two、three 都要增加几个空格,在变量比较多的情况下,是非常累赘的事情。

18.【推荐】不同逻辑、不同语义、不同业务的代码之间插入一个空行分隔开来以提升可读性。

说明:任何情形,没有必要插入多个空行进行隔开。

相关文章:

  • 人生苦短 我用Python,零基础运行你的第一行Python代码
  • zabbix案例--zabbix监控nginx状态
  • 《Rust权威指南》读书笔记 - Chapter 1, 2
  • 框架学习——ElasticSearch分布式搜索框架
  • 羊了个羊数据结构分析与代码简单实现
  • Linux驱动开发10 --- 内存和I/O
  • 【大模型迁移 2022】Exploring Visual Prompts for Adapting Large-Scale Models
  • RDKit计算摩根分子描述符
  • 从事网络安全工作一定要科班出身吗?
  • 【BERT-多标签文本分类实战】之四——数据集预处理
  • FreeSWITCH 1.10 源码阅读(2)-xml_curl 模块原理
  • springboot+旅游网站 毕业设计-附源码211713
  • Java:Java和C++哪个更好
  • WebGL编程指南-20 三维视图对象前后关系的处理/隐藏面消除与深度冲突处理
  • Jquery基础入门
  • python3.6+scrapy+mysql 爬虫实战
  • Apache Zeppelin在Apache Trafodion上的可视化
  • es的写入过程
  • java正则表式的使用
  • React-生命周期杂记
  • Spring声明式事务管理之一:五大属性分析
  • 机器学习中为什么要做归一化normalization
  • 入职第二天:使用koa搭建node server是种怎样的体验
  • 三分钟教你同步 Visual Studio Code 设置
  • 视频flv转mp4最快的几种方法(就是不用格式工厂)
  • 云大使推广中的常见热门问题
  • 中国人寿如何基于容器搭建金融PaaS云平台
  • 扩展资源服务器解决oauth2 性能瓶颈
  • !!【OpenCV学习】计算两幅图像的重叠区域
  • #我与Java虚拟机的故事#连载18:JAVA成长之路
  • (1/2)敏捷实践指南 Agile Practice Guide ([美] Project Management institute 著)
  • (c语言)strcpy函数用法
  • (Python第六天)文件处理
  • (二)Pytorch快速搭建神经网络模型实现气温预测回归(代码+详细注解)
  • (附源码)计算机毕业设计ssm电影分享网站
  • (每日持续更新)信息系统项目管理(第四版)(高级项目管理)考试重点整理 第13章 项目资源管理(七)
  • (五)c52学习之旅-静态数码管
  • (一)使用Mybatis实现在student数据库中插入一个学生信息
  • (总结)Linux下的暴力密码在线破解工具Hydra详解
  • ****** 二十三 ******、软设笔记【数据库】-数据操作-常用关系操作、关系运算
  • *Algs4-1.5.25随机网格的倍率测试-(未读懂题)
  • .NET 简介:跨平台、开源、高性能的开发平台
  • .NET 中使用 TaskCompletionSource 作为线程同步互斥或异步操作的事件
  • .NET国产化改造探索(一)、VMware安装银河麒麟
  • .net和jar包windows服务部署
  • @开发者,一文搞懂什么是 C# 计时器!
  • [20161214]如何确定dbid.txt
  • [BZOJ 4129]Haruna’s Breakfast(树上带修改莫队)
  • [C#][opencvsharp]opencvsharp sift和surf特征点匹配
  • [CDOJ 1343] 卿学姐失恋了
  • [Hive] INSERT OVERWRITE DIRECTORY要注意的问题
  • [JavaScript]_[初级]_[关于forof或者for...of循环语句的用法]
  • [jQuery]使用jQuery.Validate进行客户端验证(中级篇-上)——不使用微软验证控件的理由...
  • [k8s系列]:kubernetes·概念入门
  • [LeetCode]Reverse Linked List II