从《阿里巴巴Java开发手册》看Java中的坑

本文精选自《阿里巴巴Java开发手册》,聚焦于提升代码质量和避免潜在陷阱的规范。内容涵盖命名风格、OOP规约、集合处理等关键领域,并提供实用案例解析。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

 

简介

为什么会有代码规范?一个很重要的原因是,加强代码的可阅读性,避免歧义。还有一个重要的原因是,有许多语法允许,但是你这么做了,在特定情况下就会坑你一下你还不知道怎么回事的用法,应该避免使用。

今天,我们就扒一扒《阿里巴巴Java开发手册》中的此类规范。

本文可以看作《阿》的简化版,只筛选出了上述定义的规范,对于一些只是增强代码可读性的规范,还请参考《阿》原文

命名风格

11554693-d6464c54896b96bf.png

Java中的内部类是以Father&Son.class命名的,特定情况下,使用$会产生编译错误

11554693-bc51706fd9b3a394.png

OOP规约

11554693-bb413689addcad33.png

直接通过类名引用静态变量,会在编译时期编译成字面量,放到类自身的常量池中

11554693-591b6d04fadb050f.png

11554693-2cd2e17b71f5a959.png

名副其实的大坑,笔者曾经调了好久才找到这个bug,记忆犹新

11554693-df3e07e36afe1d4d.png

11554693-e29bdb21dbec2406.png

一般都很少注意这个值吧,或者项目中用json传递数据,不受这个影响

11554693-3f3fc44467ed8d5c.png

构造函数中的业务逻辑,会在子类的构造函数中调用,应该手动init()

11554693-9db0bbae4d9088c8.png

笔者碰到的生产问题:如果接到的报文是一个xml,只有其中一个值有用,通常会做简单处理,即:
xml.split("<a>|</a>");
标签<a>在中间和结尾处返回的数组大小是不一样的,原因同上

11554693-9f8a082039480f2f.png

11554693-b0faf6df319dcad8.png

大部分朋友唯一用到final的地方就是常量,其实还有这么多的场景可以使用,让程序更加易读

集合处理

11554693-de6ed0dc0b3ca043.png

不按本条规则,你的HashMap最终可能只是一颗红黑树(JDK1.8起)

11554693-98afe4b724317988.png

11554693-2fe0b6b6b530d783.png

11554693-32523b25e57e0270.png

11554693-3939a909c1ca80ae.png

看似很绕,其实很容易理解,<? extents T> ,集合内部都是T的子类,add的时候不能保证类型一致,同理,<? super T> get的时候,不知道返回的是什么类型

11554693-6ce9622d2d36f8b7.png

一试便知

11554693-4de535cdb4430f99.png

Comparator源码中返回值的定义如下,等于时返回0
* @return a negative integer, zero, or a positive integer as the
* first argument is less than, equal to, or greater than the
* second.

11554693-b5a311eb1ff8d4a5.png

并发处理

11554693-0cfebdaf7d67cd13.png

11554693-c9db4cc16b95fc75.png

11554693-dc12c8f1a36f20e2.png

11554693-abacd4af7613451d.png

11554693-a6ba5179d6da0081.png

11554693-03e0a4b90d11b3a5.png

问题在于helper=new Helper();
该操作需要三步:
1. 分配对象的内存空间
2. 初始化对象
3. 设置helper指向刚分配的内存地址
其中2和3可能会被虚拟机重排序,导致其他线程看到一个还未被初始化的helper,从而出现问题。
在标红处加上volatile可以避免重排序
参考《Java并发编程的艺术》3.8章 “双重检查锁定与延迟初始化”

11554693-b90c19cf0ff2c7b9.png

volatile仅仅是解决了内存可见问题,线程在更新volatile时,会更新到主内存(这里指堆中的线程共享空间,与TLAB(ThreadLocalAllocationBuffer对应)),和锁、原子性没有任何关系

11554693-1d6e6f1827c62053.png

控制语句

11554693-dbe860143d79fa73.png

虽然这是一个提高代码可读性的规范,但实在忍不住推荐出来,如果所有的复杂if都采用这条规范,那么世界将是多么的美好

其他

11554693-1a5b1172a8f59273.png

定义在类中,用static修饰,可以参考我的这篇文章《JAVA中final、static、volatile在字节码文件中的表现》,帮助理解satic的意义

11554693-6fe76c2c29bee124.png

11554693-835aa0dc26c00a3e.png

异常处理

11554693-d949195392a2d767.png

11554693-3facb500dbabf4c6.png

catch完,起码要打个log吧?catch住什么都不做,出错的时候会让人抓狂

索引规约

11554693-cf80810e1465c3c6.png

11554693-9c7a79ffe1f47c3b.png

11554693-574824a5815ff2ad.png

11554693-da3fe6ddcaeb499e.png

总结

本文筛选了《阿里巴巴Java开发手册》中一部分比较有意思的规范,强烈建议读者下载完整电子版通读

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值