編輯:關於Android編程
之前使用OnSharedPreferenceChangeListener,遇到了點小問題,就是有些時候OnSharedPreferenceChangeListener沒有被觸發。最近花了點時間研究了一下,小做整理。本文將會介紹監聽器不被觸發的原因,解決方法,以及其中隱含的一些技術細節。
問題再現
OnSharedPreferenceChangeListener是Android中SharedPreference文件發生變化的監聽器。通常我們想要進行監聽,會實現如下的代碼。
protected void onCreate(Bundle savedInstanceState) { PreferenceManager.getDefaultSharedPreferences(getApplicationContext()) .registerOnSharedPreferenceChangeListener(new OnSharedPreferenceChangeListener() { @Override public void onSharedPreferenceChanged( SharedPreferences sharedPreferences, String key) { Log.i(LOGTAG, "testOnSharedPreferenceChangedWrong key =" + key); } }); }
這種寫法看上去沒有什麼問題,而且很多時候開始幾次onSharedPreferenceChanged方法也可以被調用。但是過一段時間(簡單demo不容易出現,但是使用DDMS中的gc會立刻導致接下來的問題),你會發現前面的方法突然不再被調用,進而影響到程序的處理。
原因剖析
簡而言之,就是你注冊的監聽器被移除掉了。
首先我們先了解一下registerOnSharedPreferenceChangeListener注冊的實現。
private final WeakHashMap<OnSharedPreferenceChangeListener, Object> mListeners = new WeakHashMap<OnSharedPreferenceChangeListener, Object>(); //some code goes here public void More ...registerOnSharedPreferenceChangeListener(OnSharedPreferenceChangeListener listener) { synchronized(this) { mListeners.put(listener, mContent); } }
從上面的代碼可以得知,一個OnSharedPreferenceChangeListener對象實際上是放到了一個WeakHashMap的容器中,執行完示例中的onCreate方法,這個監聽器對象很快就會成為垃圾回收的目標,由於放在WeakHashMap中作為key不會阻止垃圾回收,所以當監聽器對象被回收之後,這個監聽器也會從mListeners中移除。所以就造成了onSharedPreferenceChanged不會被調用。
關於WeakHashMap相關,請閱讀譯文:理解Java中的弱引用進而更多了解。
如何解決
改為對象成員變量(推薦)
將監聽器作為Activity的一個成員變量,在Activity的onResume進行注冊,在onPause時進行注銷。推薦在這兩個Activity生命周期中進行處理,尤其是當SharedPreference值發生變化後,對Activity展示的UI進行處理操作的情況。這種方法是最推薦的解決方案。
private OnSharedPreferenceChangeListener mListener = new OnSharedPreferenceChangeListener() { @Override public void onSharedPreferenceChanged( SharedPreferences sharedPreferences, String key) { Log.i(LOGTAG, "instance variable key=" + key); } }; @Override protected void onResume() { PreferenceManager.getDefaultSharedPreferences(getApplicationContext()).registerOnSharedPreferenceChangeListener(mListener); super.onResume(); } @Override protected void onPause() { PreferenceManager.getDefaultSharedPreferences(getApplicationContext()).unregisterOnSharedPreferenceChangeListener(mListener); super.onPause(); }
改為靜態變量(不推薦)
如下,將一個指向匿名的內部類對象的變量sListener使用static修飾,這個內部類對象則不會持有外部類的引用。
但是這種做法並不推薦,因為一個靜態變量和與外部實例不相關,我們很難和外部實例進行一些操作。
private static OnSharedPreferenceChangeListener sListener = new OnSharedPreferenceChangeListener() { @Override public void onSharedPreferenceChanged( SharedPreferences sharedPreferences, String key) { Log.i(LOGTAG, "static variable key=" + key); } };
為什麼這樣設計
可能會有人認為這是系統設計的貓膩或者bug,其實不然,這正是Android設計人員的高明之處。
正如我們示例的代碼一樣,將一個(隱式的)局部變量添加到監聽器容器中,如果該容器只是一個普通的HashMap,這樣會導致內存洩露,因為該容器還有局部變量指向的對象,該對象又隱式持有外部Activity的對象,導致Activity無法被銷毀。關於非靜態內部類持有隱式持有外部類引用,請參考細話Java:”失效”的private修飾符
除此之外,因為局部變量無法在其所在方法外部訪問,這樣就導致了我們只可以使用方法中使用局部變量就行注冊,在合適的時機卻無法使用局部變量進行注銷。
以上就是對 Android OnSharedPreferenceChangeListener的介紹,以及出現問題解決辦法,謝謝大家對本站的支持!
Fragment Android是在Android 3.0 (API level 11)開始引入Fragment的。 可以把Fragment想成Activity中的模塊,
intent主要包括隱式意圖和顯式意圖。顯式意圖通常主要是啟動本應用中的Activity之間的數據,而隱式意圖則常見於啟動系統中的某些特定的動作,比如打電話,發短信,或者
引言從GraphicsLab Project項目立項以來,一直都在忙著搭建Shader的實驗環境,現在基本的實驗環境已經搭建完畢,所以就試著使用它來編寫一些效果。本篇文章
在開發中UI布局是我們都會遇到的問題,隨著UI越來越多,布局的重復性、復雜度也會隨之增長。Android官方給了幾個優化的方法,但是網絡上的資料基本上都是對官方資料的翻