從Effective Java總結(jié)一些有助安卓開發(fā)的建議
在編寫長期運(yùn)行下既易于維護(hù)又能保持高效的 Java 代碼這方面 ,《Effective Java》 被很多人看做最重要的書之一。 Android 使用的是 Java 語言,這就意味著這本書中所有給出的建議一定都是適用與 Android 的,對(duì)嗎?答案是:不一定。
有些人認(rèn)為這本書給出的大多數(shù)建議都不適用于 Android 開發(fā)。在我看來,也并非如此。我認(rèn)為這本書中有一部分是不適用的,因?yàn)椴皇撬械?Java 特性都是為了用于 Android 而優(yōu)化的(比如枚舉,序列化等等),也因?yàn)橐苿?dòng)設(shè)備存在自身的限制(比如 Dalvik/ART 表現(xiàn)和 JVM 不同)。盡管如此,這本書中的大多數(shù)范例還是能夠不加修改或者少量修改地被使用,并能夠有助于建立一個(gè)更加健康、干凈和可維護(hù)的代碼庫。
本文關(guān)注于這本書中我認(rèn)為對(duì)于 Android 開發(fā)至關(guān)重要的知識(shí)點(diǎn)。對(duì)于閱讀過這本書的人來說,本文可以作為書中提到的原則/知識(shí)的回顧。對(duì)于那些(目前)還沒有讀過的人來說,本文可以給他們一個(gè)機(jī)會(huì)去嘗試下這本書。
強(qiáng)制不可實(shí)例化
如果你不希望使用 new 關(guān)鍵字來創(chuàng)建一個(gè)對(duì)象,那就強(qiáng)制使用私有構(gòu)造器(private constructor)。這對(duì)于僅包含靜態(tài)方法的工具類來說更加有用。
- List<Movie> latestMovies() {
- if (db.query().isEmpty()) {
- return Collections.emptyList();
- }
- [...]
- }
靜態(tài)工廠
不要使用 _new_ 關(guān)鍵字和構(gòu)造器,使用靜態(tài)工廠方法(以及一個(gè)私有的構(gòu)造器)。這些工廠方法是命名的,不需要每次都返回一個(gè)新的對(duì)象實(shí)例,而且可以根據(jù)需要返回不同的子類型。
- class Movie {
- [...]
- public static Movie create(String title) {
- return new Movie(title);
- }
- }
【更新】讀者 @ stsdema28 提出了一個(gè)有用的建議:使用靜態(tài)工廠會(huì)使得測(cè)試變得困難。如果這樣的話,不妨考慮使用一個(gè)非靜態(tài)的工廠用于在測(cè)試時(shí)進(jìn)行模擬(或者一個(gè)能夠?qū)崿F(xiàn)的工廠接口)。
Builders
當(dāng)你有需要三個(gè)以上的構(gòu)造參數(shù)時(shí),使用 Builder 去構(gòu)造這個(gè)對(duì)象。寫起來可能有點(diǎn)啰嗦但是這樣伸縮性和可讀性都很好。如果你創(chuàng)建的是值類型,考慮 AutoValue。
- class Movie {
- static Builder newBuilder() {
- return new Builder();
- }
- static class Builder {
- String title;
- Builder withTitle(String title) {
- this.title = title;
- return this;
- }
- Movie build() {
- return new Movie(title);
- }
- }
- private Movie(String title) {
- [...]
- }
- }
- // Use like this:
- Movie matrix = Movie.newBuilder().withTitle("The Matrix").build();
避免可變性
不可變對(duì)象在其整個(gè)生命周期中都保持不變。所有需要的數(shù)據(jù)都在對(duì)象創(chuàng)建時(shí)提供。這種方式有著多種優(yōu)點(diǎn),如簡單,線程安全以及可共享。
- class Movie {
- [...]
- Movie sequel() {
- return Movie.create(this.title + " 2");
- }
- }
- // Use like this:
- Movie toyStory = Movie.create("Toy Story");
- Movie toyStory2 = toyStory.sequel();
或許很難做到每個(gè)類都不可變。對(duì)于這種情況,盡可能使你的類變得不可變(比如使用 private final 字段,final 類聲明)。在移動(dòng)設(shè)備上創(chuàng)建對(duì)象代價(jià)更加昂貴,因此不要過度使用。
靜態(tài)成員類
如果你定義了一個(gè)不依賴于外部類的內(nèi)部類,別忘了將其定義為靜態(tài)的。不這么做的話會(huì)導(dǎo)致每個(gè)內(nèi)部類的實(shí)例都會(huì)擁有對(duì)外部類的引用。
- class Movie {
- [...]
- static class MovieAward {
- [...]
- }
- }
泛型(幾乎)無處不在
Java 提供了類型安全的特性,我們應(yīng)對(duì)此心懷感激(看看JS吧)。盡量避免使用原始類型 (raw types)或者對(duì)象類型。泛型大多數(shù)情況下都提供了讓你的代碼在編譯時(shí)類型安全的機(jī)制。
- // DON'T
- List movies = Lists.newArrayList();
- movies.add("Hello!");
- [...]
- String movie = (String) movies.get(0);
- // DO
- List<String> movies = Lists.newArrayList();
- movies.add("Hello!");
- [...]
- String movie = movies.get(0);
別忘了你能在函數(shù)的參數(shù)和返回值中使用泛型。
- // DON'T
- List sort(List input) {
- [...]
- }
- // DO
- <T> List<T> sort(List<T> input) {
- [...]
- }
為了更加靈活你可以使用 bounded wildcards 來拓展可接受的類型的范圍。
- // Read stuff from collection - use "extends"
- void readList(List<? extends Movie> movieList) {
- for (Movie movie : movieList) {
- System.out.print(movie.getTitle());
- [...]
- }
- }
- // Write stuff to collection - use "super"
- void writeList(List<? super Movie> movieList) {
- movieList.add(Movie.create("Se7en"));
- [...]
- }
返回空(列表/集合)
當(dāng)必須返回空的列表/集合時(shí),避免使用null。返回一個(gè)空的集合會(huì)產(chǎn)生一個(gè)更簡單的接口(不需要去進(jìn)行文檔注釋并聲明函數(shù)返回值為 null),還能避免意外的空指針異常。***返回一個(gè)相同的空集合而不是創(chuàng)建一個(gè)新的。
- List<Movie> latestMovies() {
- if (db.query().isEmpty()) {
- return Collections.emptyList();
- }
- [...]
- }
不要用 + 連接 String
如果要連接幾個(gè)字符串,+ 操作符或許可以。但永遠(yuǎn)不要使用它來進(jìn)行大量的字符串連接,那樣性能會(huì)十分糟糕。***使用 StringBuilder 來代替。
- String latestMovieOneLiner(List<Movie> movies) {
- StringBuilder sb = new StringBuilder();
- for (Movie movie : movies) {
- sb.append(movie);
- }
- return sb.toString();
- }
可恢復(fù)的異常
我不喜歡通過拋出異常來指明錯(cuò)誤,但如果你這樣做的話,就要確保異常被檢查并且異常的捕獲者能夠從錯(cuò)誤中恢復(fù)。
- List<Movie> latestMovies() throws MoviesNotFoundException {
- if (db.query().isEmpty()) {
- throw new MoviesNotFoundException();
- }
- [...]
- }
總結(jié)
列舉的這些不是這本書中所給出的完整建議,也不是簡短說明深入評(píng)價(jià)。只是這些有用的建議的一紙小抄而已 :)。