Java最佳实践,提升代码可读性与可靠性

发表于 4年以前  | 总阅读数:505 次

从命名变量到设计软件架构,开发人员每天都要做出许多决定,而做出正确决定的最好的办法就是经验。虽然并非每个人都具备丰富的软件开发经验,但每个人都可以从他人身上学习。下面是我对 Java 开发总结的一些技巧,希望可以有助你提高 Java 代码的可读性和可靠性。

1、编程原则

写代码不是够用就好,因为这些代码不仅需要你维护,未来某个时刻还有其他人会加入维护的行列。软件开发的“二八定律”,开发人员 80% 的时间在阅读代码,而仅有 20% 的时间在编写和调试。

请务必编写可读性高的代码

这样的代码不看注释只通过代码就可以了解代码完成的功能。

下面我会列出一些重要的设计原则,可以帮助我们设计出优秀的代码。

  • KISS 原则:在设计当中应当注重简约的原则。与之相对的是,在刚开始编程时就高出复杂、摸棱两可的设计。
  • DRY 原则:不要做重复的工作。把重复的代码和逻辑提取出来。
  • YAGNI 原则:不要过度设计。可以为将来预留扩展点,但不要仅仅因为可能需要就开始工作。
  • 代码整洁更重要:没有必要为了展示聪明和学识而去“打磨”代码。
  • 避免过早优化:过早的优化的问题在于,只有事后才能真正知道真正的瓶颈在哪里。
  • 单一职责:一个类只负责一个功能领域中的相应职责。
  • 组合优于继承:实现具有复杂行为的对象应该通过实现接口而不是继承来添加行为。
  • 对象健身操:对象健身操时一组编程练习,包含了9条规则(也称为“九戒”)。
  • 快速失败**原则**:快速失败原则表示一旦发生任何意外错误,应立即停止当前操作。坚持这一原则通常会带来更稳定的解决方案。

2、Package 最佳实践

  1. 优先考虑按业务领域而非技术层次构组织 package。
  2. 定义 package 时要考虑信息的封装和隐藏,避免按技术实现定义 package 带来的错误使用。
  3. View package 作为严格的 API 对待:不要暴露内部实现。
  4. 不要为只在 package 内使用的类使用 pulic。

3、Class 最佳实践

3.1 静态类

  1. 静态类不允许实例化,为静态类添加一个私有构造函数。
  2. 静态类必须满足:无状态、不可变、不允许继承、线程安全。
  3. 确保使用静态类不会给程序带来副作用。通常静态类会作为工具类提供,例如过滤列表等。

3.2 继承

  1. 优先考虑组合而不是继承。
  2. 不要公开 protected 字段,可以提供一个 protected accessor。
  3. 如果可以使用 final,请把类标记为 final 。
  4. 如果不希望被其他类继承,同样把类标记为 final。
  5. 除非允许子类覆盖方法,否则请把方法标记为 final。
  6. 如果不需要构造函数,请不要创建没有实现逻辑的默认构造函数。Java 会替你创建一个默认构造函数。

4、接口最佳实践

  1. 不要在接口中定义常量,这样不但无法阻止类实现该接口同时还会污染 API。请改为使用静态类。使用静态类还有一个好处,就是可以在 static 代码块中执行更复杂的对象初始化操作。
  2. 不要过度使用接口。
  3. 如果有且只有一个类实现接口,这可能是过度使用接口的表现。这种用法的弊大于利。
  4. 面向接口编程,不要面向实现编程”并不意味着每个业务类都要有接口,这种是过度设计,违反前面提到的 YAGNI 原则。
  5. 保持接口功能小而具体,这样使用的人能快速找到自己感兴趣的功能。可以参考 SOLID 六大设计原则中的 SIP 原则(接口隔离原则)。

5、Finalizer 最佳实践

  1. 请谨慎使用 Object#finalize()。只用作清理资源时 fail-safe 措施(失效安全措施)使用,比如关闭文件。使用资源的时,始终提供显式的清理方法,比如 close()。
  2. 在继承层次结构中,始终在 try 块中调用父类的 finalize(),在 finally 中执行清理操作。
  3. 如果没有显示调用清理方法并且 finalizer 关闭了资源,要记录错误。
  4. 如果没有 logger,请使用线程异常处理程序。最终会转到标准错误并记录日志。

6、通用原则

6.1 断言

断言用来检查程序执行的先决条件,是快速失败原则的一种体现。可以借助断言更快地定位错误根源。

对象的状态:

  • 永远不要创建无效对象或把对象变为无效状态。
  • 在构造函数和方法中,始终检查入参确保符合要求。
  • 不要使用Java assert 关键字,因为它可能会被禁用。
  • 使用 Assertions 类避免冗长的 if-else 检查。

6.2 泛型

下面是开发者应该注意的泛型使用典型场景。

1、尽可能使用类型推断而不是返回基类或接口:

// MySpecialObject o = MyObjectFactory.getMyObject();
public <T extends MyObject> T getMyObject(int type) { 
    return (T) factory.create(type);
}

2、当无法自动推断类型时使用 inline。

public class MySpecialObject extends MyObject<SpecialType> {
  public MySpecialObject() {
   super(Collections.emptyList());   // 这种写法很丑陋,还丢弃了类型
   super(Collections.EMPTY_LIST();    // 这种写法很蠢
   // 推荐写法
   super(new ArrayList<SpecialType>());    
   super(Collections.<SpecialType>emptyList());
  }
}

3、通配符:

只读不可写时用 exends,只写不可读时用 super。如果需要读写,则不要用通配符。

  1. 大家都喜欢 PECS 原则(生产者用 extends,消费者用 super)
  2. T Producer 使用 Foo<? extends T>
  3. T Consumer 使用 Foo<? super T>

7、单例最佳实践

不要原封不动地照抄经典“设计模式”代码实现单例。虽然在 C++ 中有效,但在 Java 中不合适。

1、下面的代码尽管线程安全,但请不要这么做(当心性能瓶颈)。

public final class MySingleton {
  private static MySingleton instance;

  private MySingleton() {
    // 单例
  }

  public static synchronized MySingleton getInstance() {
    if (instance == null) {
      instance = new MySingleton();
    }
    return instance;
  }
}

2、如果确实需要延迟初始化,则可以像下面这样结合使用:

public final class MySingleton {
  private MySingleton() {
    // singleton
  }

  private static final class MySingletonHolder {
    static final MySingleton instance = new MySingleton();
  }

  public static MySingleton getInstance() {
    return MySingletonHolder.instance;
  }
}

3、Spring 中的单例没有性能瓶颈。默认情况下,创建的 bean 会加入 singleton scope,然后提供给所有的使用者。

8、异常最佳实践

1、可恢复的情况下用受检异常,编程中的错误用运行时异常。

例如将字符串转为整数:

  • 错误:NumberFormatException 继承自 RuntimeException,这是一种编程错误。
  • 不要像下面这样做:
// String str = input string
Integer value = null;

try {
   value = Integer.valueOf(str);
} catch (NumberFormatException e) {
    // 非字符串 string
}

if (value == null) {
    // 处理错误的 string
} else {
    // 业务逻辑
}
  • 正确做法:
// String str = input string
// string 只包含数字和开始的负号
if ( (str != null) && str.matches("-?\\d++") ) {  
   Integer value = Integer.valueOf(str);
  // 业务逻辑
} else {
  // 处理错误的 string
}

2、在正确的地方处理异常,通常是业务层。

  • 错误做法:发生数据库异常时,数据对象层不知道如何处理。
class UserDAO{
    public List<User> getUsers(){
        try{
            ps = conn.prepareStatement("SELECT * from users");
            rs = ps.executeQuery();
            //return result
        }catch(Exception e){
            log.error("exception")
            return null
        }finally{
            // 释放资源
        }
    }}
  • 推荐方式:数据层只要抛出异常,接下来交由合适的层次来处理。
// 推荐方法
// 数据层只管抛出异常交由其他层处理
class UserDAO{
    public List<User> getUsers(){
       try{
          ps = conn.prepareStatement("SELECT * from users");
          rs = ps.executeQuery();
          //return result
       }catch(Exception e){
        throw new DataLayerException(e);
       }finally{
          // 释放资源
       }
   }
}

3、通常,异常不应在触发异常时记录,而应当在实际处理时记录。抛出或重新抛出异常时记录日志,往往会让日志文件看起来杂乱无章。

另外请注意,异常堆栈跟踪会捕获生成异常的位置。

4、推荐使用标准异常。

5、推荐使用异常而非返回码。

9、Equals 与 HashCode 最佳实践


实现 Equals 和 HashCode 的时候有许多要注意的地方。简单起见,可以用 java.util.Object 的 equals 10和 hash。

public final class User {
  private final String firstName;
  private final String lastName;
  private final int age;

  ...

  public boolean equals(Object o) {
    if (this == o) {
      return true;
    } else if (!(o instanceof User)) {
      return false;
    }

    User user = (User) o;
    return Objects.equals(getFirstName(), user.getFirstName()) &&                                 
     Objects.equals(getLastName(),user.getLastName()) &&
     Objects.equals(getAge(), user.getAge());
  }

  public int hashCode() {
    return Objects.hash(getFirstName(),getLastName(),getAge());
  }
}

10、资源管理最佳实践

安全释放资源的方法:

  • try-with-resources 语句可确保在语句执行结束关闭所有资源。
  • 实现 java.lang.AutoCloseable 接口(java.io.Closeable)的对象都可以作为资源使用。
private doSomething() {
    try (BufferedReader br = new BufferedReader(new FileReader(path))) {
        try {
        // 业务逻辑
    }
}

11、**提供 Java 关闭钩子**


如果 JVM 正常终止会调用提供的关闭钩子函数。当然了,这无法解决由于断电引起的突然终止。

下面是 finalize() 的一种替代方法,仅在 System.runFinalizersOnExit() 为 true(默认为 false)时调用。

public final class SomeObject {
  var distributedLock = new ExpiringGeneralLock ("SomeObject", "shared");

  public SomeObject() {
    Runtime
      .getRuntime()
      .addShutdownHook(new Thread(new LockShutdown(distributedLock)));
  }

  /** 从不同的服务器上获取锁 */
  ...
  /** 安全地释放分布式锁 */
  private static final class LockShutdown implements Runnable {
    private final ExpiringGeneralLock distributedLock;

    public LockShutdown(ExpiringGeneralLock distributedLock) {
      if (distributedLock == null) {
        throw new IllegalArgumentException("ExpiringGeneralLock is null");
      }
      this.distributedLock = distributedLock;
    }

    public void run() {
      if (isLockAlive()) {
        distributedLock.release();
      }
    }
    /** @return True 如果获得锁成功且没有过期 */
    private boolean isLockAlive() {
      return distributedLock.getExpirationTimeMillis() > System.currentTimeMillis();
    }
  }
}

服务器之间可以共享过期或新创建的资源,这样可以支持从突然终止的情况下恢复(例如断电)。

在上面的示例代码中,使用了在系统之间共享的锁 ExpiringGeneralLock。

12、日期日期最佳实践

Java 8 在 java.time package 中引入了新的日期时间 API,弥补了早期 API 的一些缺点:例如非线程安全、设计不良,时区处理困难等。

13、并发最佳实践

13.1 通用原则

  1. 使用下面这些库的时候请当心,它们是非线程安全的。如果在多个线程之间共享,则务必始终对对象进行同步。
  2. Date(非不可变):推荐使用线程安全的新 Date-time API。
  3. SimpleDateFormat :推荐使用线程安全的新 Date-time API。
  4. 优先使用 java.util.concurrent.atomic 类,而不是把变量标记为 volatile
  5. 原子类的行为对于普通开发人员更显而易见,而使用 volatile 需要了解 Java 内存模型。
  6. 原子类将 volatile 变量包装在一个更友好的接口中。
  7. 了解哪些场合适合使用 volatile 变量。
  8. 当需要受检异常但没有返回类型时,请使用 Callable。由于 Void 无法实例化,因此可以清晰地传达意图,安全地返回 null。

13.2 线程

  1. 可以认为 java.lang.Thread 已过期,推荐用 java.util.concurrent package。后者提供了更干净的解决方案。
  2. 不要继承 java.lang.Thread,而是实现 Runnable 接口并在构造函数中使用实例创建一个新线程(组合优于继承)。
  3. 处理并发处理时,优先选择 executors 和 streams。
  4. 推荐自定义线程工厂,可以更好地控制创建线程时的配置。
  5. 在 Executors 中对非关键线程使用 DaemonThreadFactory,这样在服务器关闭时立即关闭线程池。
this.executor = Executors.newCachedThreadPool((Runnable runnable) -> {
    Thread thread = Executors.defaultThreadFactory().newThread(runnable);
    thread.setDaemon(true);
    return thread;
});
  1. Java同步已经不像以前那么慢了(55–110ns)。不要使用诸如双重检查锁定之类的破坏性技巧来提高效率。
  2. 最好与内部对象(而不是类)同步,因为用户可能会与类或者实例同步。
  3. 始终按照相同的顺序同步多个对象,避免死锁。
  4. 与类同步并不能 100% 可靠阻止访问内部对象。访问资源时,请始终使用相同的锁。
  5. 当心,synchronized 关键字不是方法签名的一部分,因此不会被子类继承。
  6. 要避免过度使用同步,这可能导致性能下降和产生死锁。只需要同步的代码使用 synchronized 关键字。

14、集合最佳实践

  1. 尽可能在多线程代码中使用 Java 5 并发集合,不但安全而且性能高。
  2. 在合适的情况下,推荐使用 CopyOnWriteArrayList 取代 SynchronizedList。
  3. 推荐使用 Collections.unmodifiable list(…) 或在把集合作为参数 new ArrayList(list) 接收时复制集合。避免从 class 以外的地方修改集合。
  4. 始终返回集合的拷贝,避免 new ArrayList(list) 被外部修改集合。
  5. 每个集合都应该包装到自己的类中。这样因此与集合相关的行为就有了归属(例如,filter 方法,可以向每个元素应用规则)。

15、其它原则

  1. 推荐使用 lambda 而非匿名类。
  2. 推荐使用方法引而非 lambda。
  3. 推荐使用枚举而非 int 常量。
  4. 如果需要结果精确,推荐使用 BigDecimal 而非 float 或 double。
  5. 推荐使用原始类型而非装箱类型。
  6. 推荐使用常数,不要在代码中使用“魔数”。
  7. 使用 Optional,不要返回 Null。集合也一样:返回空数组或集合,不要返回 null。
  8. 避免创建不必要的对象,尽可能重用对象,避免产生不必要的 GC。

16、延迟初始化最佳实践

延迟初始化是一种性能优化。处理某些不可避免开销很大的情况。Java 8 支持 Supplier 函数式接口支持延迟初始化。

// 线程安全的延迟初始化
public final class Lazy<T> {
    private volatile T value;

    public T getOrCompute(Supplier<T> supplier) {
        final T result = value; // 执行一次 volatile 读取
        return result == null ? maybeCompute(supplier) : result;
    }

    private synchronized T maybeCompute(Supplier<T> supplier) {
        if (value == null) {
            value = supplier.get();
        }
        return value;
    }
}
Lazy<String> lazyToString= new Lazy<>()
return lazyToString.getOrCompute( () -> "(" + x + ", " + y + ")");

本文由哈喽比特于4年以前收录,如有侵权请联系我们。
文章来源:https://mp.weixin.qq.com/s/yLqc4f73qJNN6I5R4ZBdlg

 相关推荐

刘强东夫妇:“移民美国”传言被驳斥

京东创始人刘强东和其妻子章泽天最近成为了互联网舆论关注的焦点。有关他们“移民美国”和在美国购买豪宅的传言在互联网上广泛传播。然而,京东官方通过微博发言人发布的消息澄清了这些传言,称这些言论纯属虚假信息和蓄意捏造。

发布于:1年以前  |  808次阅读  |  详细内容 »

博主曝三大运营商,将集体采购百万台华为Mate60系列

日前,据博主“@超能数码君老周”爆料,国内三大运营商中国移动、中国电信和中国联通预计将集体采购百万台规模的华为Mate60系列手机。

发布于:1年以前  |  770次阅读  |  详细内容 »

ASML CEO警告:出口管制不是可行做法,不要“逼迫中国大陆创新”

据报道,荷兰半导体设备公司ASML正看到美国对华遏制政策的负面影响。阿斯麦(ASML)CEO彼得·温宁克在一档电视节目中分享了他对中国大陆问题以及该公司面临的出口管制和保护主义的看法。彼得曾在多个场合表达了他对出口管制以及中荷经济关系的担忧。

发布于:1年以前  |  756次阅读  |  详细内容 »

抖音中长视频App青桃更名抖音精选,字节再发力对抗B站

今年早些时候,抖音悄然上线了一款名为“青桃”的 App,Slogan 为“看见你的热爱”,根据应用介绍可知,“青桃”是一个属于年轻人的兴趣知识视频平台,由抖音官方出品的中长视频关联版本,整体风格有些类似B站。

发布于:1年以前  |  648次阅读  |  详细内容 »

威马CDO:中国每百户家庭仅17户有车

日前,威马汽车首席数据官梅松林转发了一份“世界各国地区拥车率排行榜”,同时,他发文表示:中国汽车普及率低于非洲国家尼日利亚,每百户家庭仅17户有车。意大利世界排名第一,每十户中九户有车。

发布于:1年以前  |  589次阅读  |  详细内容 »

研究发现维生素 C 等抗氧化剂会刺激癌症生长和转移

近日,一项新的研究发现,维生素 C 和 E 等抗氧化剂会激活一种机制,刺激癌症肿瘤中新血管的生长,帮助它们生长和扩散。

发布于:1年以前  |  449次阅读  |  详细内容 »

苹果据称正引入3D打印技术,用以生产智能手表的钢质底盘

据媒体援引消息人士报道,苹果公司正在测试使用3D打印技术来生产其智能手表的钢质底盘。消息传出后,3D系统一度大涨超10%,不过截至周三收盘,该股涨幅回落至2%以内。

发布于:1年以前  |  446次阅读  |  详细内容 »

千万级抖音网红秀才账号被封禁

9月2日,坐拥千万粉丝的网红主播“秀才”账号被封禁,在社交媒体平台上引发热议。平台相关负责人表示,“秀才”账号违反平台相关规定,已封禁。据知情人士透露,秀才近期被举报存在违法行为,这可能是他被封禁的部分原因。据悉,“秀才”年龄39岁,是安徽省亳州市蒙城县人,抖音网红,粉丝数量超1200万。他曾被称为“中老年...

发布于:1年以前  |  445次阅读  |  详细内容 »

亚马逊股东起诉公司和贝索斯,称其在购买卫星发射服务时忽视了 SpaceX

9月3日消息,亚马逊的一些股东,包括持有该公司股票的一家养老基金,日前对亚马逊、其创始人贝索斯和其董事会提起诉讼,指控他们在为 Project Kuiper 卫星星座项目购买发射服务时“违反了信义义务”。

发布于:1年以前  |  444次阅读  |  详细内容 »

苹果上线AppsbyApple网站,以推广自家应用程序

据消息,为推广自家应用,苹果现推出了一个名为“Apps by Apple”的网站,展示了苹果为旗下产品(如 iPhone、iPad、Apple Watch、Mac 和 Apple TV)开发的各种应用程序。

发布于:1年以前  |  442次阅读  |  详细内容 »

特斯拉美国降价引发投资者不满:“这是短期麻醉剂”

特斯拉本周在美国大幅下调Model S和X售价,引发了该公司一些最坚定支持者的不满。知名特斯拉多头、未来基金(Future Fund)管理合伙人加里·布莱克发帖称,降价是一种“短期麻醉剂”,会让潜在客户等待进一步降价。

发布于:1年以前  |  441次阅读  |  详细内容 »

光刻机巨头阿斯麦:拿到许可,继续对华出口

据外媒9月2日报道,荷兰半导体设备制造商阿斯麦称,尽管荷兰政府颁布的半导体设备出口管制新规9月正式生效,但该公司已获得在2023年底以前向中国运送受限制芯片制造机器的许可。

发布于:1年以前  |  437次阅读  |  详细内容 »

马斯克与库克首次隔空合作:为苹果提供卫星服务

近日,根据美国证券交易委员会的文件显示,苹果卫星服务提供商 Globalstar 近期向马斯克旗下的 SpaceX 支付 6400 万美元(约 4.65 亿元人民币)。用于在 2023-2025 年期间,发射卫星,进一步扩展苹果 iPhone 系列的 SOS 卫星服务。

发布于:1年以前  |  430次阅读  |  详细内容 »

𝕏(推特)调整隐私政策,可拿用户发布的信息训练 AI 模型

据报道,马斯克旗下社交平台𝕏(推特)日前调整了隐私政策,允许 𝕏 使用用户发布的信息来训练其人工智能(AI)模型。新的隐私政策将于 9 月 29 日生效。新政策规定,𝕏可能会使用所收集到的平台信息和公开可用的信息,来帮助训练 𝕏 的机器学习或人工智能模型。

发布于:1年以前  |  428次阅读  |  详细内容 »

荣耀CEO谈华为手机回归:替老同事们高兴,对行业也是好事

9月2日,荣耀CEO赵明在采访中谈及华为手机回归时表示,替老同事们高兴,觉得手机行业,由于华为的回归,让竞争充满了更多的可能性和更多的魅力,对行业来说也是件好事。

发布于:1年以前  |  423次阅读  |  详细内容 »

AI操控无人机能力超越人类冠军

《自然》30日发表的一篇论文报道了一个名为Swift的人工智能(AI)系统,该系统驾驶无人机的能力可在真实世界中一对一冠军赛里战胜人类对手。

发布于:1年以前  |  423次阅读  |  详细内容 »

AI生成的蘑菇科普书存在可致命错误

近日,非营利组织纽约真菌学会(NYMS)发出警告,表示亚马逊为代表的电商平台上,充斥着各种AI生成的蘑菇觅食科普书籍,其中存在诸多错误。

发布于:1年以前  |  420次阅读  |  详细内容 »

社交媒体平台𝕏计划收集用户生物识别数据与工作教育经历

社交媒体平台𝕏(原推特)新隐私政策提到:“在您同意的情况下,我们可能出于安全、安保和身份识别目的收集和使用您的生物识别信息。”

发布于:1年以前  |  411次阅读  |  详细内容 »

国产扫地机器人热销欧洲,国产割草机器人抢占欧洲草坪

2023年德国柏林消费电子展上,各大企业都带来了最新的理念和产品,而高端化、本土化的中国产品正在不断吸引欧洲等国际市场的目光。

发布于:1年以前  |  406次阅读  |  详细内容 »

罗永浩吐槽iPhone15和14不会有区别,除了序列号变了

罗永浩日前在直播中吐槽苹果即将推出的 iPhone 新品,具体内容为:“以我对我‘子公司’的了解,我认为 iPhone 15 跟 iPhone 14 不会有什么区别的,除了序(列)号变了,这个‘不要脸’的东西,这个‘臭厨子’。

发布于:1年以前  |  398次阅读  |  详细内容 »
 相关文章
Java 中验证时间格式的 4 种方法 2年以前  |  3870次阅读
Java经典面试题答案解析(1-80题) 4年以前  |  3651次阅读
CentOS 配置java应用开机自动启动 4年以前  |  2797次阅读
IDEA依赖冲突分析神器—Maven Helper 4年以前  |  2772次阅读
SpringBoot 控制并发登录的人数教程 4年以前  |  2452次阅读
 目录