Java开发人员必须掌握的11种干净代码最佳实践

开发 前端
调用 toArray 时使用的是空数组(new String[0]),这种方法避免了计算列表大小的需要,并允许 toArray 方法在内部处理数组大小的调整,从而获得更好的性能和更简洁的代码。

环境:Java17

1. Nulls and Optionals

反例:从方法返回null可能会导致最可怕的NullPointerException或NPE。

public String getValue() {
  // TODO
  return null ;
}

正例:使用Optional可以更明确地处理 null,避免出现此类错误。

public Optional<String> getValue() {
  // TODO
  return Optional.empty() ;
}

2. 使用String.valueOf()优化字符串转换

反例:使用 + 运算符进行字符串连接。

double pi = 3.1415926 ;
String str = "" + pi ;
  • 这里,"+"操作符用于字符串转换,涉及隐式字符串连接
  • 这种方法可能效率低下,特别是在将大量变量转换为字符串时

正例:使用内置方法进行字符串连接。

double pi = 3.1415926 ;
String str = String.valueOf(pi) ;
  • 在这里,我们使用 valueOf() 进行字符串转换和连接
  • 该方法专门用于将其他数据类型转换为字符串,并对性能进行了优化

3. 使用 Arrays.copyOf()复制数组

反例:手动复制数组

int[] source = {1, 2, 3, 4, 5} ;
int [] target = new int[source.length] ;
for (int i = 0, len = source.length; i < len; i++) {
  target[i] = source[i] ;
}

这种方法效率较低,尤其是对于大数组,因为它需要进行多次迭代和元素赋值。

正例:使用 Arrays.copyOf()复制数组

int[] source = {1, 2, 3, 4, 5} ;
int[] target = Arrays.copyOf(source, source.length) ;

4. 使用 isEmpty() 检查空集合

反例:使用 length() 或 size() 检查字符串或集合是否为空。

String text = "Pack" ;
if (text.length() == 0) {
  // TODO
}
Set<String> datas = new HashSet<>() ;
if (datas.size() == 0) {
  // TODO
}
  • 这里,length() 用于检查字符串是否为空,size() 用于检查集合是否为空
  • 这些方法虽然有效,但却降低了代码的可读性

正例:使用 isEmpty() 检查字符串或集合是否为空。

String text = "Pack" ;
if (text.isEmpty()) {
  // TODO
}
Set<String> datas = new HashSet<>() ;
if (datas.isEmpty()) {
  // TODO
}
  • isEmpty() 方法可用于字符串和集合,以检查是否为空
  • 它的时间复杂度为 O(1),因此更高效、更易读

5. 避免并发修改异常

反例:在遍历列表时从列表中删除元素会导致 ConcurrentModificationException 异常。

List<String> datas = new ArrayList<>() ;
datas.add("1") ;
datas.add("2") ;
datas.add("3") ;
for (String s : datas) {
  if ("1".equals(s)) {
    datas.remove(s) ;
  }
}

输出结果

图片图片

正例:使用迭代器的 remove 方法或 removeIf() 方法

List<String> datas = new ArrayList<>() ;
// add(x)
Iterator<String> it = datas.iterator() ;
while (it.hasNext()) {
  String value = it.next() ;
  if ("1".equals(value)) {
    it.remove() ; 
  }
}

你也可以使用 Java 8 中引入的 removeIf() 方法,根据给定条件删除元素。

List<String> datas = new ArrayList<>() ;
// add(x)
datas.removeIf(item -> "1".equals(item)) ;

该方法在内部使用迭代器并移除与条件匹配的元素。这是一种更简洁、可读性更强的方法。

6. 预编译正则表达式

反例:运行时编译正则表达式

String str = "Hello, World" ;
if (str.matches("Hello.*")) {
  System.out.println(true) ;
}
  • 在这里,只要使用正则表达式,就会在运行时对其进行编译
  • 重复使用相同的正则表达式会降低性能

正例:预编译正则表达式

private static final Pattern PATTERN1 = Pattern.compile("Hello.*") ; 
public void validateString() {
  String str = "Hello, World" ;
  if (PATTERN1.matcher(str).matches()) {
    System.out.println(true) ;
  }
}

通过预编译重复使用的正则表达式,并在需要时重复使用它,我们可以避免不必要的编译,并提高性能。

7. 避免在检索前预先检查数据是否存在

反例:先判断是否存在然后在获取数据

public static void process(Map<String, String> params) {
  if (params.containsKey("action")) {
    String value = params.get("action") ;
    // TODO
  }
}
  • 在这里,我们首先检查Map中是否存在指定的key,然后再检索它
  • 这种预先检查是不必要的,因为如果未找到键,则 Map的 get 方法会返回 null

正例:直接获取值判断是否null

String action = params.get("action") ;
if (action != null) {
  // TODO
}

这种方法避免了多余的检查,使代码更简洁、更高效。

8. 将集合高效转换为数组

反例:

List<String> datas = new ArrayList<>() ;
datas.add("1") ;
datas.add("2") ;
datas.add("3") ;
String[] ret = datas.toArray(new String[datas.size()]) ;

在这种方法中,首先计算列表的大小,然后创建一个新数组。这可能会影响性能,尤其是对于大数据集。

正例:

List<String> datas = new ArrayList<>() ;
// add(x)
String[] ret = datas.toArray(new String[0]) ;

在这里,调用 toArray 时使用的是空数组(new String[0]),这种方法避免了计算列表大小的需要,并允许 toArray 方法在内部处理数组大小的调整,从而获得更好的性能和更简洁的代码。

9. 合理使用默认方法

反例:

public interface Logger {
  void log(String message) ;
}
public class FileLogger implements Logger {
  public void log(String message) {
    // TODO
  }
}
public class ConsoleLogger implements Logger {
  public void log(String message) {
    // TODO
  }
}

如果需要在接口中添加 logError 等新方法,则必须修改所有实现类,这可能会导致代码维护问题和潜在错误。

正例:

public interface Logger {
  void log(String message) ;
  default void logError(String error) {
    // TODO
  }
}

在这里,Logger接口定义了一个默认方法(logError),它提供了记录错误的默认实现。这样实现类无需修改即可自动继承该默认实现。

10. 使用Date/Time API

反例:使用传统的Date类

Date birthday = new Date() ;
// TODO

这个类有很多问题,比如可变性和方法不够清晰;该类中的大部分方法,如 getYear()、getMonth() 和 getDay() 已被弃用。

正例:使用Date/Time API(Java 8 及以后版本)中的类

LocalDate date = LocalDate.now() ;
LocalDateTime dateTime = LocalDateTime.now() ;
// TODO

在这里,使用来自Date/Time API的LocalDate、LocalDateTime类;这两个类是不可变的,确保了线程的安全性,并为日期操作提供了清晰直观的方法。

11. 未使用泛型

反例:

List datas = new ArrayList() ;
datas.add(10) ;
datas.add("Hello") ;

不同的数据类型混杂在列表中,可能导致运行时出错。

正例:使用泛型可以确保类型安全,避免此类问题

List<Integer> datas = new ArrayList<>() ;
datas.add(10) ;
// 错误
// datas.add("Hello"

责任编辑:武晓燕 来源: Spring全家桶实战案例源码
相关推荐

2012-04-20 10:10:35

2017-08-18 18:35:11

前端开发网站设计开源项目

2022-08-31 15:45:57

Django数据库优化

2023-10-08 13:59:56

Git开发

2011-04-19 11:07:50

外包开发人员IT

2019-11-01 10:30:37

Reac测试工具开源

2021-07-16 17:05:56

Java 8语言Java 11

2021-07-08 09:57:47

Java 11Java 8开发人员

2018-04-08 10:08:43

开发人员工具

2018-11-21 12:21:33

Ruby框架Web应用

2010-07-06 11:00:33

RationalJazz测试管理

2020-05-26 13:37:43

Android 11开发人员安卓

2022-06-06 10:30:23

容器镜像

2020-04-20 09:57:00

开发工具技术

2012-07-20 10:46:44

Web

2019-11-25 09:41:28

开发者技能工具

2010-11-02 15:26:32

MVC开发工具

2020-10-13 18:20:55

TCPHTTP网络协议

2021-12-10 23:48:19

Java开发技术

2022-12-01 10:05:19

点赞
收藏

51CTO技术栈公众号