从Effective Java总结一些有助安卓开发的建议

简介:

从Effective Java总结一些有助安卓开发的建议


在编写长期运行下既易于维护又能保持高效的 Java 代码这方面 ,《Effective Java》 被很多人看做最重要的书之一。 Android 使用的是 Java 语言,这就意味着这本书中所有给出的建议一定都是适用与 Android 的,对吗?答案是:不一定。

有些人认为这本书给出的大多数建议都不适用于 Android 开发。在我看来,也并非如此。我认为这本书中有一部分是不适用的,因为不是所有的 Java 特性都是为了用于 Android 而优化的(比如枚举,序列化等等),也因为移动设备存在自身的限制(比如 Dalvik/ART 表现和 JVM 不同)。尽管如此,这本书中的大多数范例还是能够不加修改或者少量修改地被使用,并能够有助于建立一个更加健康、干净和可维护的代码库。

本文关注于这本书中我认为对于 Android 开发至关重要的知识点。对于阅读过这本书的人来说,本文可以作为书中提到的原则/知识的回顾。对于那些(目前)还没有读过的人来说,本文可以给他们一个机会去尝试下这本书。

强制不可实例化

如果你不希望使用 new 关键字来创建一个对象,那就强制使用私有构造器(private constructor)。这对于仅包含静态方法的工具类来说更加有用。


 
 
  1. List<Movie> latestMovies() { 
  2.     if (db.query().isEmpty()) { 
  3.         return Collections.emptyList(); 
  4.     } 
  5.     [...] 
  6. }  

静态工厂

不要使用 _new_ 关键字和构造器,使用静态工厂方法(以及一个私有的构造器)。这些工厂方法是命名的,不需要每次都返回一个新的对象实例,而且可以根据需要返回不同的子类型。


 
 
  1. class Movie { 
  2.     [...] 
  3.     public static Movie create(String title) { 
  4.         return new Movie(title); 
  5.     } 
  6. }  

【更新】读者 @ stsdema28 提出了一个有用的建议:使用静态工厂会使得测试变得困难。如果这样的话,不妨考虑使用一个非静态的工厂用于在测试时进行模拟(或者一个能够实现的工厂接口)。

Builders

当你有需要三个以上的构造参数时,使用 Builder 去构造这个对象。写起来可能有点啰嗦但是这样伸缩性和可读性都很好。如果你创建的是值类型,考虑 AutoValue。


 
 
  1. class Movie { 
  2.     static Builder newBuilder() { 
  3.         return new Builder(); 
  4.     } 
  5.     static class Builder { 
  6.         String title; 
  7.         Builder withTitle(String title) { 
  8.             this.title = title; 
  9.             return this; 
  10.         } 
  11.         Movie build() { 
  12.             return new Movie(title); 
  13.         } 
  14.     } 
  15.   
  16.     private Movie(String title) { 
  17.     [...]     
  18.     } 
  19. // Use like this: 
  20. Movie matrix = Movie.newBuilder().withTitle("The Matrix").build();  

避免可变性

不可变对象在其整个生命周期中都保持不变。所有需要的数据都在对象创建时提供。这种方式有着多种优点,如简单,线程安全以及可共享。


 
 
  1. class Movie { 
  2.     [...] 
  3.     Movie sequel() { 
  4.         return Movie.create(this.title + " 2"); 
  5.     } 
  6. // Use like this: 
  7. Movie toyStory = Movie.create("Toy Story"); 
  8. Movie toyStory2 = toyStory.sequel();  

或许很难做到每个类都不可变。对于这种情况,尽可能使你的类变得不可变(比如使用 private final 字段,final 类声明)。在移动设备上创建对象代价更加昂贵,因此不要过度使用。

静态成员类

如果你定义了一个不依赖于外部类的内部类,别忘了将其定义为静态的。不这么做的话会导致每个内部类的实例都会拥有对外部类的引用。


 
 
  1. class Movie { 
  2.     [...] 
  3.     static class MovieAward { 
  4.         [...] 
  5.     } 
  6. }  

泛型(几乎)无处不在

Java 提供了类型安全的特性,我们应对此心怀感激(看看JS吧)。尽量避免使用原始类型 (raw types)或者对象类型。泛型大多数情况下都提供了让你的代码在编译时类型安全的机制。


 
 
  1. // DON'T 
  2. List movies = Lists.newArrayList(); 
  3. movies.add("Hello!"); 
  4. [...] 
  5. String movie = (String) movies.get(0); 
  6.   
  7. // DO 
  8. List<String> movies = Lists.newArrayList(); 
  9. movies.add("Hello!"); 
  10. [...] 
  11. String movie = movies.get(0);  

别忘了你能在函数的参数和返回值中使用泛型。


 
 
  1. // DON'T 
  2. List sort(List input) { 
  3.     [...] 
  4.   
  5. // DO 
  6. <T> List<T> sort(List<T> input) { 
  7.     [...] 
  8. }  

为了更加灵活你可以使用 bounded wildcards 来拓展可接受的类型的范围。


 
 
  1. // Read stuff from collection - use "extends" 
  2. void readList(List<? extends Movie> movieList) { 
  3.     for (Movie movie : movieList) { 
  4.         System.out.print(movie.getTitle()); 
  5.         [...] 
  6.     } 
  7.   
  8. // Write stuff to collection - use "super" 
  9. void writeList(List<? super Movie> movieList) { 
  10.     movieList.add(Movie.create("Se7en")); 
  11.     [...] 
  12. }  

返回空(列表/集合)

当必须返回空的列表/集合时,避免使用null。返回一个空的集合会产生一个更简单的接口(不需要去进行文档注释并声明函数返回值为 null),还能避免意外的空指针异常。最好返回一个相同的空集合而不是创建一个新的。


 
 
  1. List<Movie> latestMovies() { 
  2.     if (db.query().isEmpty()) { 
  3.         return Collections.emptyList(); 
  4.     } 
  5.     [...] 
  6. }  

不要用 + 连接 String

如果要连接几个字符串,+ 操作符或许可以。但永远不要使用它来进行大量的字符串连接,那样性能会十分糟糕。最好使用 StringBuilder 来代替。


 
 
  1. String latestMovieOneLiner(List<Movie> movies) { 
  2.     StringBuilder sb = new StringBuilder(); 
  3.     for (Movie movie : movies) { 
  4.         sb.append(movie); 
  5.     } 
  6.     return sb.toString(); 
  7. }  

可恢复的异常

我不喜欢通过抛出异常来指明错误,但如果你这样做的话,就要确保异常被检查并且异常的捕获者能够从错误中恢复。


 
 
  1. List<Movie> latestMovies() throws MoviesNotFoundException { 
  2.     if (db.query().isEmpty()) { 
  3.         throw new MoviesNotFoundException(); 
  4.     } 
  5.     [...] 
  6. }  

总结

列举的这些不是这本书中所给出的完整建议,也不是简短说明深入评价。只是这些有用的建议的一纸小抄而已 :)。


作者:lazy song

来源:51CTO

相关文章
|
1月前
|
Java API Maven
如何使用Java开发抖音API接口?
在数字化时代,社交媒体平台如抖音成为生活的重要部分。本文详细介绍了如何用Java开发抖音API接口,从创建开发者账号、申请API权限、准备开发环境,到编写代码、测试运行及注意事项,全面覆盖了整个开发流程。
108 10
|
29天前
|
监控 Java API
如何使用Java语言快速开发一套智慧工地系统
使用Java开发智慧工地系统,采用Spring Cloud微服务架构和前后端分离设计,结合MySQL、MongoDB数据库及RESTful API,集成人脸识别、视频监控、设备与环境监测等功能模块,运用Spark/Flink处理大数据,ECharts/AntV G2实现数据可视化,确保系统安全与性能,采用敏捷开发模式,提供详尽文档与用户培训,支持云部署与容器化管理,快速构建高效、灵活的智慧工地解决方案。
|
19天前
|
Java 开发者 微服务
Spring Boot 入门:简化 Java Web 开发的强大工具
Spring Boot 是一个开源的 Java 基础框架,用于创建独立、生产级别的基于Spring框架的应用程序。它旨在简化Spring应用的初始搭建以及开发过程。
38 6
Spring Boot 入门:简化 Java Web 开发的强大工具
|
6天前
|
存储 JavaScript 前端开发
基于 SpringBoot 和 Vue 开发校园点餐订餐外卖跑腿Java源码
一个非常实用的校园外卖系统,基于 SpringBoot 和 Vue 的开发。这一系统源于黑马的外卖案例项目 经过站长的进一步改进和优化,提供了更丰富的功能和更高的可用性。 这个项目的架构设计非常有趣。虽然它采用了SpringBoot和Vue的组合,但并不是一个完全分离的项目。 前端视图通过JS的方式引入了Vue和Element UI,既能利用Vue的快速开发优势,
52 13
|
11天前
|
算法 Java API
如何使用Java开发获得淘宝商品描述API接口?
本文详细介绍如何使用Java开发调用淘宝商品描述API接口,涵盖从注册淘宝开放平台账号、阅读平台规则、创建应用并申请接口权限,到安装开发工具、配置开发环境、获取访问令牌,以及具体的Java代码实现和注意事项。通过遵循这些步骤,开发者可以高效地获取商品详情、描述及图片等信息,为项目和业务增添价值。
44 10
|
5天前
|
前端开发 Java 测试技术
java日常开发中如何写出优雅的好维护的代码
代码可读性太差,实际是给团队后续开发中埋坑,优化在平时,没有那个团队会说我专门给你一个月来优化之前的代码,所以在日常开发中就要多注意可读性问题,不要写出几天之后自己都看不懂的代码。
40 2
|
14天前
|
JavaScript 安全 Java
java版药品不良反应智能监测系统源码,采用SpringBoot、Vue、MySQL技术开发
基于B/S架构,采用Java、SpringBoot、Vue、MySQL等技术自主研发的ADR智能监测系统,适用于三甲医院,支持二次开发。该系统能自动监测全院患者药物不良反应,通过移动端和PC端实时反馈,提升用药安全。系统涵盖规则管理、监测报告、系统管理三大模块,确保精准、高效地处理ADR事件。
|
1月前
|
开发框架 Java 关系型数据库
Java哪个框架适合开发API接口?
在快速发展的软件开发领域,API接口连接了不同的系统和服务。Java作为成熟的编程语言,其生态系统中出现了许多API开发框架。Magic-API因其独特优势和强大功能,成为Java开发者优选的API开发框架。本文将从核心优势、实际应用价值及未来展望等方面,深入探讨Magic-API为何值得选择。
42 2
|
1月前
|
监控 前端开发 Java
【技术开发】接口管理平台要用什么技术栈?推荐:Java+Vue3+Docker+MySQL
该文档介绍了基于Java后端和Vue3前端构建的管理系统的技术栈及功能模块,涵盖管理后台的访问、登录、首页概览、API接口管理、接口权限设置、接口监控、计费管理、账号管理、应用管理、数据库配置、站点配置及管理员个人设置等内容,并提供了访问地址及操作指南。
|
1月前
|
IDE Java 编译器
开发 Java 程序一定要安装 JDK 吗
开发Java程序通常需要安装JDK(Java Development Kit),因为它包含了编译、运行和调试Java程序所需的各种工具和环境。不过,某些集成开发环境(IDE)可能内置了JDK,或可使用在线Java编辑器,无需单独安装。
64 1