成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Android內存泄漏的各種原因詳解

移動開發(fā) Android
在Android開發(fā)過程中,最為讓我們頭疼的就是內存的泄露問題了,很可能你很小的一個錯誤都會引起內存的泄露,下面將為大家奉上引起內存泄露的解決方案。

1.資源對象沒關閉造成的內存泄漏

描述:

資源性對象比如(Cursor,F(xiàn)ile文件等)往往都用了一些緩沖,我們在不使用的時候,應該及時關閉它們,以便它們的緩沖及時回收內存。它們的緩沖不僅存在于 java虛擬機內,還存在于java虛擬機外。如果我們僅僅是把它的引用設置為null,而不關閉它們,往往會造成內存泄漏。因為有些資源性對象,比如 SQLiteCursor(在析構函數(shù)finalize(),如果我們沒有關閉它,它自己會調close()關閉),如果我們沒有關閉它,系統(tǒng)在回收它時也會關閉它,但是這樣的效率太低了。因此對于資源性對象在不使用的時候,應該調用它的close()函數(shù),將其關閉掉,然后才置為null.在我們的程序退出時一定要確保我們的資源性對象已經(jīng)關閉。

程序中經(jīng)常會進行查詢數(shù)據(jù)庫的操作,但是經(jīng)常會有使用完畢Cursor后沒有關閉的情況。如果我們的查詢結果集比較小,對內存的消耗不容易被發(fā)現(xiàn),只有在常時間大量操作的情況下才會復現(xiàn)內存問題,這樣就會給以后的測試和問題排查帶來困難和風險。

示例代碼:

  1. Cursor cursor = getContentResolver().query(uri...); 
  2.  
  3. if (cursor.moveToNext()) { 
  4.  
  5. ... ... 
  6.  

修正示例代碼:

  1. Cursor cursor = null
  2.  
  3. try { 
  4.  
  5. cursor = getContentResolver().query(uri...); 
  6.  
  7. if (cursor != null &&cursor.moveToNext()) { 
  8.  
  9. ... ... 
  10.  
  11.  
  12. finally { 
  13.  
  14. if (cursor != null) { 
  15.  
  16. try { 
  17.  
  18. cursor.close(); 
  19.  
  20. catch (Exception e) { 
  21.  
  22. //ignore this 
  23.  
  24.  
  25.  

2.構造Adapter時,沒有使用緩存的convertView

描述:

以構造ListView的BaseAdapter為例,在BaseAdapter中提供了方法:

public View getView(int position, ViewconvertView, ViewGroup parent)

來向ListView提供每一個item所需要的view對象。初始時ListView會從BaseAdapter中根據(jù)當前的屏幕布局實例化一定數(shù)量的 view對象,同時ListView會將這些view對象緩存起來。當向上滾動ListView時,原先位于最上面的list item的view對象會被回收,然后被用來構造新出現(xiàn)的最下面的list item。這個構造過程就是由getView()方法完成的,getView()的第二個形參View convertView就是被緩存起來的list item的view對象(初始化時緩存中沒有view對象則convertView是null)。由此可以看出,如果我們不去使用 convertView,而是每次都在getView()中重新實例化一個View對象的話,即浪費資源也浪費時間,也會使得內存占用越來越大。 ListView回收list item的view對象的過程可以查看:

android.widget.AbsListView.java --> voidaddScrapView(View scrap) 方法。

示例代碼:

  1. public View getView(int position, ViewconvertView, ViewGroup parent) { 
  2.  
  3. View view = new Xxx(...); 
  4.  
  5. ... ... 
  6.  
  7. return view; 
  8.  

修正示例代碼:

  1. public View getView(int position, ViewconvertView, ViewGroup parent) { 
  2.  
  3. View view = null
  4.  
  5. if (convertView != null) { 
  6.  
  7. view = convertView; 
  8.  
  9. populate(view, getItem(position)); 
  10.  
  11. ... 
  12.  
  13. else { 
  14.  
  15. view = new Xxx(...); 
  16.  
  17. ... 
  18.  
  19.  
  20. return view; 
  21.  

3.Bitmap對象不在使用時調用recycle()釋放內存

描述:

有時我們會手工的操作Bitmap對象,如果一個Bitmap對象比較占內存,當它不在被使用的時候,可以調用Bitmap.recycle()方法回收此對象的像素所占用的內存,但這不是必須的,視情況而定。可以看一下代碼中的注釋:

  1. /** 
  2.  
  3. •Free up the memory associated with thisbitmap's pixels, and mark the 
  4.  
  5. •bitmap as "dead", meaning itwill throw an exception if getPixels() or 
  6.  
  7. •setPixels() is called, and will drawnothing. This operation cannot be 
  8.  
  9. •reversed, so it should only be called ifyou are sure there are no 
  10.  
  11. •further uses for the bitmap. This is anadvanced call, and normally need 
  12.  
  13. •not be called, since the normal GCprocess will free up this memory when 
  14.  
  15. •there are no more references to thisbitmap. 
  16.  
  17. */ 

4.試著使用關于application的context來替代和activity相關的context

這是一個很隱晦的內存泄漏的情況。有一種簡單的方法來避免context相關的內存泄漏。最顯著地一個是避免context逃出他自己的范圍之外。使用Application context。這個context的生存周期和你的應用的生存周期一樣長,而不是取決于activity的生存周期。如果你想保持一個長期生存的對象,并且這個對象需要一個context,記得使用application對象。你可以通過調用 Context.getApplicationContext() or Activity.getApplication()來獲得。更多的請看這篇文章如何避免

Android內存泄漏。

5.注冊沒取消造成的內存泄漏

一些Android程序可能引用我們的Anroid程序的對象(比如注冊機制)。即使我們的Android程序已經(jīng)結束了,但是別的引用程序仍然還有對我們的Android程序的某個對象的引用,泄漏的內存依然不能被垃圾回收。調用registerReceiver后未調用unregisterReceiver。

比如:假設我們希望在鎖屏界面(LockScreen)中,監(jiān)聽系統(tǒng)中的電話服務以獲取一些信息(如信號強度等),則可以在LockScreen中定義一個 PhoneStateListener的對象,同時將它注冊到TelephonyManager服務中。對于LockScreen對象,當需要顯示鎖屏界面的時候就會創(chuàng)建一個LockScreen對象,而當鎖屏界面消失的時候LockScreen對象就會被釋放掉。

但是如果在釋放 LockScreen對象的時候忘記取消我們之前注冊的PhoneStateListener對象,則會導致LockScreen無法被垃圾回收。如果不斷的使鎖屏界面顯示和消失,則最終會由于大量的LockScreen對象沒有辦法被回收而引起OutOfMemory,使得system_process 進程掛掉。

雖然有些系統(tǒng)程序,它本身好像是可以自動取消注冊的(當然不及時),但是我們還是應該在我們的程序中明確的取消注冊,程序結束時應該把所有的注冊都取消掉。

6.集合中對象沒清理造成的內存泄漏

我們通常把一些對象的引用加入到了集合中,當我們不需要該對象時,并沒有把它的引用從集合中清理掉,這樣這個集合就會越來越大。如果這個集合是static的話,那情況就更嚴重了。

責任編輯:張葉青 來源: eoe Android開發(fā)者社區(qū)
相關推薦

2016-12-15 21:47:11

Android內存泄漏

2024-07-03 11:28:15

2024-08-05 10:40:58

2016-07-05 14:09:02

AndroidJAVA內存

2016-03-21 10:31:25

Android內存泄露

2016-12-22 17:21:11

Android性能優(yōu)化內存泄漏

2023-10-31 16:40:38

LeakCanary內存泄漏

2013-02-20 16:02:02

Android開發(fā)內存泄露

2015-03-30 11:18:50

內存管理Android

2011-06-01 12:50:41

Android 內存

2014-07-30 14:22:41

AndroidWebView內存泄漏

2019-01-30 18:24:14

Java內存泄漏編程語言

2023-10-24 09:30:49

Java內存

2013-08-02 09:52:14

AndroidApp內存泄漏

2009-06-16 11:17:49

內存泄漏

2024-03-11 08:22:40

Java內存泄漏

2023-12-18 10:45:23

內存泄漏計算機服務器

2012-06-19 15:12:20

Java內存泄露

2024-02-21 08:00:55

WindowsDWM進程

2012-02-22 21:28:58

內存泄漏
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产精品99久 | 国产美女视频黄a视频免费 国产精品福利视频 | 久久精品国产亚洲a | 久久国产精品-国产精品 | 国产精品日韩欧美一区二区 | 久久国产传媒 | 国产91在线 | 欧美 | 欧美色综合 | 人人爽人人爽人人片av | 91精品国产91久久久久久 | 国内精品视频在线观看 | 99re视频在线观看 | 亚洲午夜久久久 | 欧美视频1区 | 国产一区高清 | 狠狠综合久久av一区二区老牛 | 日本不卡一区二区三区 | 国产一区二区在线视频 | 毛片网在线观看 | 日韩成人在线视频 | 国产精品视频一 | 欧美日韩a | 欧美一级片黄色 | 精品国产一区二区三区久久久久久 | 亚洲精品大全 | 91就要激情 | 在线成人免费视频 | 亚洲欧洲视频 | 免费亚洲成人 | 久久精品国产精品青草 | 亚洲免费三级 | 五月婷婷导航 | 亚洲精品一区在线观看 | 最新黄色在线观看 | 国产精品一区二区av | 一区二区三区国产精品 | 国产在线观看不卡一区二区三区 | 一区二区三区在线免费观看 | a免费在线| 一本一道久久a久久精品蜜桃 | 91视频进入 |