android的GC内存泄露问题

 更新时间:2016年11月18日 17:14:54   作者:Devin Zhang  
本篇文章主要介绍了android GC内存泄露问题,具有一定的参考价值,有需要的可以了解一下。

1. android内存泄露概念

不少人认为JAVA程序,因为有垃圾回收机制,应该没有内存泄露。其实如果我们一个程序中,已经不再使用某个对象,但是因为仍然有引用指向它,垃圾回收器就无法回收它,当然该对象占用的内存就无法被使用,这就造成了内存泄露。如果我们的java运行很久,而这种内存泄露不断的发生,最后就没内存可用了。当然java的,内存泄漏和C/C++是不一样的。如果java程序完全结束后,它所有的对象就都不可达了,系统就可以对他们进行垃圾回收,它的内存泄露仅仅限于它本身,而不会影响整个系统的。C/C++的内存泄露就比较糟糕了,它的内存泄露是系统级,即使该C/C++程序退出,它的泄露的内存也无法被系统回收,永远不可用了,除非重启机器。

Android的一个应用程序的内存泄露对别的应用程序影响不大。为了能够使得Android应用程序安全且快速的运行,Android的每个应用程序都会使用一个专有的Dalvik虚拟机实例来运行,也就是说每个应用程序都是在属于自己的进程中运行的。Android为不同类型的进程分配了不同的内存使用上限,如果程序在运行过程中出现了内存泄漏的而造成应用进程使用的内存超过了这个上限,则会被系统视为内存泄漏,从而被kill掉,这使得仅仅自己的进程被kill掉,而不会影响其他进程(如果是system_process等系统进程出问题的话,则会引起系统重启)。

内存泄露示例:

/*此时,所有的Object对象都没有被释放,因为变量v引用这些对象。实际上这些对象已经是无用的,但还被引用,GC就无能为力了(事实上GC认为它还有用),这一点是导致内存泄漏最重要的原因。*/
Vector v = new Vector(10);   
for (int i = 1; i < 100; i++)   {    
  Object o = new Object();    
  v.add(o);    
  o = null;   
 }

循环申请Object对象,并将所申请的对象放入一个Vector中,如果仅仅释放对象本身,但因为Vector仍然引用该对象,所以这个对象对GC来说是不可回收的。因此,如果对象加入到Vector后,还必须从Vector中删除,最简单的方法就是将Vector对象设置为null。

总的来说,内存管理中的内存泄漏产生的主要原因:保留下来却永远不再使用的对象引用。

2.引起内存泄露的情况

1)资源对象没关闭造成的内存泄露

资源性对象比如(Cursor,File文件等)往往都用了一些缓冲,我们在不使用的时候,应该及时关闭它们,以便它们的缓冲及时回收内存。它们的缓冲不仅存在于java虚拟机内,还存在于java虚拟机外。如果我们仅仅是把它的引用设置为null,而不关闭它们,往往会造成内存泄露。因为有些资源性对象,比如SQLiteCursor(在析构函数finalize(),如果我们没有关闭它,它自己会调close()关闭),如果我们没有关闭它,系统在回收它时也会关闭它,但是这样的效率太低了。因此对于资源性对象在不使用的时候,应该调用它的close()函数,将其关闭掉,然后才置为null.在我们的程序退出时一定要确保我们的资源性对象已经关闭。

程序中经常会进行查询数据库的操作,但是经常会有使用完毕Cursor后没有关闭的情况。如果我们的查询结果集比较小,对内存的消耗不容易被发现,只有在常时间大量操作的情况下才会复现内存问题,这样就会给以后的测试和问题排查带来困难和风险。

2)一些不良代码成内存压力

有些代码并不造成内存泄露,但是它们,或是对没使用的内存没进行有效及时的释放,或是没有有效的利用已有的对象而是频繁的申请新内存,对内存的回收和分配造成很大影响的,容易迫使虚拟机不得不给该应用进程分配更多的内存,造成不必要的内存开支。

a.Bitmap没调用recycle()

Bitmap对象在不使用时,我们应该先调用recycle()释放内存,然后才它设置为null。

虽然recycle()从源码上看,调用它应该能立即释放Bitmap的主要内存,但是测试结果显示它并没能立即释放内存。但是我它应该还是能大大的加速Bitmap的主要内存的释放。

b.构造Adapter时,没有使用缓存的convertView

以构造ListView的BaseAdapter为例,在BaseAdapter中提共了方法:

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

来向ListView提供每一个item所需要的view对象。初始时ListView会从BaseAdapter中根据当前的屏幕布局实例化一定数量的view对象,同时ListView会将这些view对象缓存起来。当向上滚动ListView时,原先位于最上面的list item的view对象会被回收,然后被用来构造新出现的最下面的list item。这个构造过程就是由getView()方法完成的,getView()的第二个形参View convertView就是被缓存起来的list item的view对象(初始化时缓存中没有view对象则convertView是null)。

由此可以看出,如果我们不去使用convertView,而是每次都在getView()中重新实例化一个View对象的话,即浪费时间,也造成内存垃圾,给垃圾回收增加压力,如果垃圾回收来不及的话,虚拟机将不得不给该应用进程分配更多的内存,造成不必要的内存开支。

layout.xml

 <?xml version="1.0" encoding="utf-8" ?> 
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="fill_parent" android:layout_height="60dip">
 <ImageView android:id="@+id/icon" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_alignParentLeft="true" android:paddingLeft="9px" /> 
 <TextView android:id="@+id/text" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_toRightOf="@id/icon" android:textAppearance="?android:attr/textAppearanceMedium" android:paddingLeft="9px" /> 
 </RelativeLayout>

能引起内存泄露的代码: BadAdapter.java

public class BadAdapter extends BaseAdapter {
  ......

  @Override
  public View getView(int position, View convertView, ViewGroup parent) {
    Log.d("MyAdapter", "Position:" + position + "---"
        + String.valueOf(System.currentTimeMillis()));
    final LayoutInflater inflater = (LayoutInflater) mContext
        .getSystemService(Context.LAYOUT_INFLATER_SERVICE);
    View v = inflater.inflate(R.layout.list_item_icon_text, null);
    ((ImageView) v.findViewById(R.id.icon)).setImageResource(R.drawable.icon);
    ((TextView) v.findViewById(R.id.text)).setText(mData[position]);
    return v;
  }
}

修正优化示例代码示例代码:GoodAdapter.java

public class GoodAdapter extends BaseAdapter {

  ......

  @Override
  public View getView(int position, View convertView, ViewGroup parent) {
    Log.d("MyAdapter", "Position:" + position + "---"
        + String.valueOf(System.currentTimeMillis()));
    ViewHolder holder;
    if (convertView == null) {
      final LayoutInflater inflater = (LayoutInflater) mContext
          .getSystemService(Context.LAYOUT_INFLATER_SERVICE);
      convertView = inflater.inflate(R.layout.list_item_icon_text, null);
      holder = new ViewHolder();
      holder.icon = (ImageView) convertView.findViewById(R.id.icon);
      holder.text = (TextView) convertView.findViewById(R.id.text);
      convertView.setTag(holder);
    } else {
      holder = (ViewHolder) convertView.getTag();
    }
    holder.icon.setImageResource(R.drawable.icon);
    holder.text.setText(mData[position]);
    return convertView;
  }

  static class ViewHolder {
    ImageView icon;
    TextView text;
  }
}

MainActivity.java

public class MainActivity extends ListActivity {
  private BadAdapter/GoodAdapter mAdapter;

  private String[] mArrData;

  /** Called when the activity is first created. */
  @Override
  public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.main);
    mArrData = new String[1000];
    for (int i = 0; i < 1000; i++) {
      mArrData[i] = "Google IO Adapter";
    }
    mAdapter = new BadAdapter/GoodAdapter(this, mArrData);
    setListAdapter(mAdapter);
  }
}

3)ThreadLocal使用不当

如果我们粗暴的把ThreadLocal设置null,而不调用remove()方法或set(null),那么就可能造成ThreadLocal绑定的对象长期也能被回收,因而产出内存泄露。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • 解决android studio 3.0 加载项目过慢问题--maven仓库选择

    解决android studio 3.0 加载项目过慢问题--maven仓库选择

    这篇文章主要介绍了android studio 3.0 加载项目过慢问题解决方案---maven仓库选择,非常不错,具有参考借鉴价值,需要的朋友可以参考下
    2017-11-11
  • Android之EditText控制禁止输入空格和回车

    Android之EditText控制禁止输入空格和回车

    本文主要介绍了Android中使用EditText控制禁止输入空格和回车的实现代码。具有很好的参考价值。下面跟着小编一起来看下吧
    2017-04-04
  • Android实现GPS定位代码实例

    Android实现GPS定位代码实例

    这篇文章主要介绍了Android实现GPS定位实例,对关键操作部份给出代码示例并做了一定的注释,需要的朋友可以参考下
    2014-07-07
  • Kotlin lambda表达式入门指南

    Kotlin lambda表达式入门指南

    在kotlin当中,Lambda表达式是最高等级的,Lambda表达式可以理解为一种匿名函数,是一种高效的类似于函数式编程的表达式,本文介绍Kotlin lambda表达式入门指南,感兴趣的朋友一起看看吧
    2024-03-03
  • 详解kotlin中::双冒号的使用

    详解kotlin中::双冒号的使用

    在 Kotlin 中 , :: 双冒号操作符 的作用是获取类,对象,函数,属性的 类型对象引用,这篇文章主要介绍了详解kotlin中::双冒号的使用,需要的朋友可以参考下
    2023-04-04
  • Android实现加载对话框

    Android实现加载对话框

    这篇文章主要为大家详细介绍了Android实现加载对话框,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-01-01
  • Android 超简易Zxing框架 生成二维码+扫码功能

    Android 超简易Zxing框架 生成二维码+扫码功能

    这篇文章主要介绍了Android 超简易Zxing框架 生成二维码+扫码功能,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-09-09
  • Android IPC机制绑定Service实现本地通信

    Android IPC机制绑定Service实现本地通信

    本文主要介绍Android IPC机制绑定Service 实现本地通信,通过图解,代码等方式给大家解释Android IPC机制,需要参考的同学可以看一下
    2016-07-07
  • Android Notification的多种用法总结

    Android Notification的多种用法总结

    这篇文章主要介绍了Android Notification的多种用法总结的相关资料,需要的朋友可以参考下
    2017-06-06
  • Android数据库SD卡创建和图片存取操作

    Android数据库SD卡创建和图片存取操作

    这篇文章主要介绍了Android数据库SD卡创建和图片存取操作的相关资料,需要的朋友可以参考下
    2017-04-04

最新评论