1 java引用類型
? 4種引用類型:強引用(StrongReference),軟引用(SoftReference),弱引用(WeakReference),幽靈引用(PhantomReference)。
強引用是最普遍的引用,如果一個對象具有強引用,那垃圾回收器絕不會回收它。
軟引用強度上弱于強引用,它的作用是告訴回收器哪些對象是不那么重要,在內存不足時可以暫時回收。當JVM內存不足時,垃圾回收器會釋放那些只被軟引用所指向的對象。如果全部釋放完這些對象之后,內存還不足,才會拋出OutOfMemory錯誤。軟引用非常適合于創建緩存。當系統內存不足的時候,緩存中的內容是可以被釋放的。通過類SoftReference來表示。
弱引用在強度上弱于軟引用,通過類WeakReference來表示。它的作用是引用一個對象,但是并不阻止該對象被回收。如果使用一個強引用的話,只要該引用存在,那么被引用的對象是不能被回收的。弱引用則沒有這個問題。在垃圾回收器運行的時候,如果一個對象的所有引用都是弱引用的話,該對象會被回收。弱引用的作用在于解決強引用所帶來的對象之間在存活時間上的耦合關系。弱引用最常見的用處是在集合類中,尤其在哈希表中。哈希表的接口允許使用任何Java對象作為鍵來使用。當一個鍵值對被放入到哈希表中之后,哈希表對象本身就有了對這些鍵和值對象的引用。如果這種引用是強引用的話,那么只要哈希表對象本身還存活,其中所包含的鍵和值對象是不會被回收的。如果某個存活時間很長的哈希表中包含的鍵值對很多,最終就有可能消耗掉JVM中全部的內存。
幽靈引用,要先介紹Java提供的對象終止化機制(finalization)。在Object類里面有個finalize方法,其設計的初衷是在一個對象被真正回收之前,可以用來執行一些清理的工作。因為Java并沒有提供類似C++的析構函數一樣的機制,就通過 finalize方法來實現。但是問題在于垃圾回收器的運行時間是不固定的,所以這些清理工作的實際運行時間也是不能預知的。幽靈引用(phantom reference)可以解決這個問題。在創建幽靈引用PhantomReference的時候必須要指定一個引用隊列。當一個對象的finalize方法已經被調用了之后,這個對象的幽靈引用會被加入到隊列中。通過檢查該隊列里面的內容就知道一個對象是不是已經準備要被回收了。
幽靈引用及其隊列的使用情況并不多見,主要用來實現比較精細的內存使用控制,這對于移動設備來說是很有意義的。程序可以在確定一個對象要被回收之后,再申請內存創建新的對象。通過這種方式可以使得程序所消耗的內存維持在一個相對較低的數量。
2 Android多進程
1 AIDL
AIDL 即AndroidInterface Definition Language的縮寫,是專為 Android 中跨進程通信接口的描述語言。優缺點很明顯,優點是穩定,快,Android 專門用于跨進程通信設計的。缺點是比較麻煩,AIDL 是通信的約定,參加通信的雙方都需要把這個 AIDL 文件都加入自己的代碼中,然后創建 Service 來實現訪問和被訪問。
2 ContentProvider
作為 Android 四大基礎組件之一的 ContentProvider 本來它的作用只是提供內容性質的跨進程訪問。但是在 API 11 (Android 3.0) 中,ContentProvider 加入了一個新的方法,可以用來進行跨進程的方法調用,ContentProvider 中這個方法的定義如下:
Bundle call(String method, String arg, Bundle extras)
從易用性來講,這個沒有 AIDL 那么麻煩,而且擴展性更強,也沒有 Broadcast 過于依賴系統,API 11 應該就是主要是缺點了,別的缺點暫時沒發現,歡迎補充。
3 Broadcast
廣播是最簡單的:優點是把分發消息的任務全部交給 Android 系統了;缺點也是因為全交給系統了,很多地方不受控制。缺點:
雖然廣播可以通過指定包名來進行發送指向性消息,但是卻不能驗證消息去向 App 的簽名。
系統重啟之后,在系統的廣播隊列里邊的消息就丟失了。
3 Java多線程之間的通訊
線程間的相互作用:線程之間需要一些協調通信,來共同完成一件任務。
wait()方法
wait()方法使得當前線程必須要等待,等到另外一個線程調用notify()或者notifyAll()方法。
線程調用wait()方法,釋放它對鎖的擁有權,然后等待另外的線程來通知它(通知的方式是notify()或者notifyAll()方法),這樣它才能重新獲得鎖的擁有權和恢復執行。
要確保調用wait()方法的時候擁有鎖,即,wait()方法的調用必須放在synchronized方法或synchronized塊中。
notify()方法
notify()方法會喚醒一個等待當前對象的鎖的線程。
如果多個線程在等待,它們中的一個將會選擇被喚醒。這種選擇是隨意的,和具體實現有關。(線程等待一個對象的鎖是由于調用了wait方法中的一個)。
被喚醒的線程是不能被執行的,需要等到當前線程放棄這個對象的鎖。
被喚醒的線程將和其他線程以通常的方式進行競爭,來獲得對象的鎖。也就是說,被喚醒的線程并沒有什么優先權,也沒有什么劣勢,對象的下一個線程還是需要通過一般性的競爭。
notify()方法應該是被擁有對象的鎖的線程所調用。
(This method should only be called by a thread that is the owner of this object's monitor.)
換句話說,和wait()方法一樣,notify方法調用必須放在synchronized方法或synchronized塊中。
4 Android布局優化
Overdraw就是過度繪制,
Android提供了測量Overdraw的選項,在開發者選項-調試GPU過度繪制(Show GPU Overdraw),打開選項就可以看到當前頁面Overdraw的狀態.
第一招:合理選擇控件容器。
LinearLayout易用,效率高,表達能力有限。RelativeLayout復雜,表達能力強,效率稍遜。
第二招:去掉window的默認背景
第三招:去掉其他不必要的背景
第四招:ClipRect & QuickReject
第五招:ViewStub
第六招:Merge
第七招:善用draw9patch
第八招:慎用Alpha
第九招:避免“OverDesign”
5 NestedScrollView 與CoordinatorLayout 機制
NestedScrollingParent
NestedScrollingChild
這是兩個接口, ?Android 就是通過這兩個接口, 來實現 子View 與父View 之間的嵌套滑動。