Summarize some suggestions from Effective Java that can help Android development

Summarize some suggestions from Effective Java that can help Android development

Effective Java is considered by many to be one of the most important books on writing Java code that is both maintainable and efficient over the long term. Android uses the Java language, so this means that all the advice given in this book must apply to Android, right? The answer is: not necessarily.

Some people think that most of the advice given in this book is not applicable to Android development. In my opinion, this is not the case. I think some parts of the book are not applicable because not all Java features are optimized for Android (such as enumerations, serialization, etc.) and because mobile devices have their own limitations (such as Dalvik/ART behaves differently from the JVM). Despite this, most of the examples in this book can be used without or with minimal modification and can help to build a healthier, cleaner, and maintainable code base.

This article focuses on the key points in the book that I think are essential for Android development. For those who have read the book, this article can serve as a review of the principles/knowledge mentioned in the book. For those who have not read it (yet), this article can give them a chance to try the book.

Forced not to be instantiable

If you do not want to use the new keyword to create an object, then force the use of a private constructor. This is more useful for utility classes that only contain static methods.

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

Static Factory

Instead of using the _new_ keyword and constructors, use static factory methods (and a private constructor). These factory methods are named, do not need to return a new object instance every time, and can return different subtypes as needed.

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

[Update] Reader @stsdema28 made a useful suggestion: using a static factory can make testing difficult. If this is the case, consider using a non-static factory that can be mocked during testing (or a factory interface that can be implemented).

Builders

When you need more than three construction parameters, use Builder to construct the object. It may be a bit verbose to write, but it is very scalable and readable. If you are creating a value type, consider 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. }
  20. // Use like this:
  21. Movie matrix = Movie.newBuilder().withTitle( "The Matrix" ).build();

Avoid mutability

Immutable objects remain unchanged throughout their lifetime. All required data is provided when the object is created. This approach has many advantages such as simplicity, thread safety, and shareability.

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

It may be difficult to make every class immutable. In this case, make your classes immutable as much as possible (e.g. use private final fields, final class declarations). Object creation is more expensive on mobile devices, so don't overdo it.

Static member class

If you define an inner class that does not depend on the outer class, don't forget to define it as static. Failure to do so will result in each instance of the inner class having a reference to the outer class.

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

Generics are (almost) everywhere

Java provides type safety, and we should be grateful for it (see JS). Try to avoid using raw types or object types. Generics provide a mechanism to make your code type safe at compile time in most cases.

  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);

Don't forget that you can use generics in function parameters and return values.

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

For more flexibility you can use bounded wildcards to expand the range of acceptable types.

  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.   
  9. // Write stuff to collection - use "super"  
  10. void writeList(List<? super Movie> movieList) {
  11. movieList. add (Movie. create ( "Se7en" ));
  12. [...]
  13. }

Returns empty (list/set)

Avoid using null when you must return an empty list/set. Returning an empty set results in a simpler interface (no need to document and declare that the function returns null) and avoids unexpected NullPointerExceptions. *** Return the same empty set instead of creating a new one.

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

Don't use + to concatenate Strings

If you want to concatenate a few strings, the + operator might work. But never use it to do a large number of string concatenations, as the performance will be terrible. Use StringBuilder instead.

  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. }

Recoverable exceptions

I don't like throwing exceptions to indicate errors, but if you do, make sure the exception is checked and the catcher of the exception can recover from the error.

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

Summarize

This is not a complete list of advice given in the book, nor is it a brief in-depth review. It is just a list of useful suggestions :).

<<:  In-depth understanding of the Android Instant Run operating mechanism

>>:  Convolutional neural networks cannot process "graph" structured data? This article tells you the answer

Recommend

WeChat dedicated input method is here

According to the latest news, WeChat input method...

Android phones have exceeded 20G of RAM! How much RAM is enough?

How much running memory is enough? Today, the edi...

PMS APP installation process analysis

PMS (PackageManagerService) is the core of the An...

Architecture design based on dynamic routing on mobile terminals

I haven't written an article for a long time....

Zhihu, forced to live stream

On September 30, a Zhihu user said that he had re...

Marketing Promotion: How does fission marketing achieve market “fission”?

Fission marketing is like cell division, from one...