Effective Java - Prefer lambdas to anonymous classes

這個Item主要提倡使用Lambda去取代Anonymous Class。以書中Sort的例子來說:

Collections.sort(words, new Comparator<String>() {
	public int compare(String s1, String s2) {
		return Integer.compare(s1.length(), s2.length());
	}
});
假如我改用Lambda去寫,程式會簡潔許多:
Collections.sort(words, (s1, s2) -> Integer.compare(s1.length(), s2.length()));
如果再搭配Comparator原本提供的comparingInt,會更簡潔:
Collections.sort(words, Comparator.comparingInt(String::length));
針對這部分的使用,書中有幾點必須注意的:

  1. 除非看不懂,否則可以省略參數Type宣告,讓code簡潔些;如s1和s2,並沒把它宣告成(String s1, String s2)。我的觀點: 省略宣告其實這是讓我有時需要多花點時間看參數種類的原因之一。
  2. 只適用於Functional interface。
  3. 假如method會存取到物件的member或使用this,就不適用這種方法。
  4. 假如要做序列化,就不適用這種方法,可改用private static nested class。
  5. 要使用Lambda寫法的method type必須夠明確,否則編譯器可能無法推斷出type而顯是錯誤。

除了這幾點以外,我認為可測性也是要注意的部分;假如匿名類別內的實作與IO有關,我就不會使用匿名類別。

書中還有提及到把這種做法使用在Operation的Enum上,這部分是我第一次看到的:

public enum Operation {
	PLUS ("+", (x, y) -> x + y),
	MINUS ("-", (x, y) -> x - y),
	TIMES ("*", (x, y) -> x * y),
	DIVIDE("/", (x, y) -> x / y);
	private final String symbol;
	private final DoubleBinaryOperator op;
	Operation(String symbol, DoubleBinaryOperator op) {
		this.symbol = symbol;
		this.op = op;
	}
	@Override public String toString() { return symbol; }
		public double apply(double x, double y) {
		return op.applyAsDouble(x, y);
	}
}
使用這種做法時,method的內容最好別超過3行,避免降低了程式的可讀性。

Effective Java第三版Item 42。

  • Effective Java, 3/e