AndroidStudio中Handler类的内存溢出风险

package com.test.king.xmlparser;

import android.annotation.SuppressLint;
import android.app.Activity;
import android.content.res.Resources;
import android.os.Handler;
import android.os.Message;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.TextView;

import org.xmlpull.v1.XmlPullParser;
import org.xmlpull.v1.XmlPullParserException;

import java.io.IOException;
import java.lang.ref.WeakReference;
import java.util.ArrayList;
import java.util.List;

public class MainActivity extends AppCompatActivity {
    private static TextView tvContent;
    private static final int MSG_FINISH=0x0001;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        tvContent=findViewById(R.id.tv_content);

    }

    //内部类解决可能存在的内存溢出
    /*
    *、本文描述静态和非静态内部类的区别是非静态内部类持有外部类的引用。
2、内部类实例的持有对象的生命周期大于其外部类对象,那么就有可能导致内存泄露。
比如,要实例化一个超出activity生命周期的内部类对象,
避免使用非静态的内部类。建议使用静态内部类并且在内部类中持有外部类的弱引用。
    *
    * */
    //静态内部类
    private static class MyHandler extends Handler {
        private final WeakReference<MainActivity> mActivity;
        //构造方法
        public MyHandler(MainActivity activity) {
            mActivity = new WeakReference<MainActivity>(activity);//对外部类的弱引用
        }
        @Override
        public void handleMessage(Message msg) {
            MainActivity activity = mActivity.get();
            if (activity != null) {
                switch (msg.what)
                {
                    case MSG_FINISH:
                        List<String> contents=(List<String>)msg.obj;
                        //在主线程中显示
                        for(String content:contents)
                        {
                            tvContent.append(content+"\n");
                        }
                        break;
                }
            }
        }
    }
    //new Handler对象处理消息,下文有引用
    private final MyHandler mHandler = new MyHandler(this);
    //警告
    /*
    private Handler handler=new Handler()
    {
        @Override
        //重写handleMessage方法处理消息
        public void handleMessage(Message msg) {
            switch (msg.what)
            {
                case MSG_FINISH:
                    List<String> contents=(List<String>)msg.obj;
                    //在主线程中显示
                    for(String content:contents)
                    {
                        tvContent.append(content+"\n");
                    }
                    break;
            }

        }
    };*/

    public  void parser(View view) throws IOException, XmlPullParserException {
     //常规方法
     /*
     List<String> contents=getPullParserContent(getResources(),R.xml.words);
     for(String content:contents)
     {
         tvContent.append(content+"\n");
     }*/
     //解析XML可能需要耗费很长的时间,所以这里单独做一个子线程
        new Thread()
        {
            @Override
            //重写run方法
            public void run()
            {
                try {
                    List<String> contents=getPullParserContent(getResources(),R.xml.words);
                    //完成工作,通知主线程
                    Message msg=mHandler.obtainMessage();//也可以new Message
                    msg.what=MSG_FINISH;
                    msg.obj=contents;
                    mHandler.sendMessage(msg);
                } catch (IOException e) {
                    e.printStackTrace();
                } catch (XmlPullParserException e) {
                    e.printStackTrace();
                }
            }
        }.start();

    }
    private List<String> getPullParserContent(Resources res,int id) throws IOException, XmlPullParserException {
        List<String> contents = null;
        String tagName;
        //XmlPullParser
        XmlPullParser parser = res.getXml(id);
        //Pull解析本质是SAX解析
        int eventType = parser.getEventType();
        while (eventType != XmlPullParser.END_DOCUMENT) {
            switch (eventType) {
                case XmlPullParser.START_DOCUMENT:
                    Log.i("Test", "START_DOCUMENT");
                    contents=new ArrayList<String>();
                    break;
                case XmlPullParser.END_DOCUMENT:
                    break;
                case XmlPullParser.START_TAG:
                    tagName = parser.getName();
                    if (tagName.equals("word")) {
                        String value = parser.getAttributeValue(0);
                        contents.add(value);
                        Log.i("Test", "START_TAG:" + tagName + " " + value);
                    }
                    break;
                case XmlPullParser.END_TAG:
                    tagName = parser.getName();
                    Log.i("Test", "END_TAG:" + tagName);
                    break;
            }
            eventType = parser.next();
        }
        return contents;
    }
}

  

原文地址:https://www.cnblogs.com/wxjnew/p/9426778.html

时间: 2024-10-08 18:03:10

AndroidStudio中Handler类的内存溢出风险的相关文章

VS中的类的内存分布(上)

0.序 目前正在学习C++中,对于C++的类及其类的实现原理也挺感兴趣.于是打算通过观察类在内存中的分布更好地理解类的实现.因为其实类的分布是由编译器决定的,而本次试验使用的编译器为VS2015 RC,所以此处的标题为<VS中的类的内存分布>. 1.对无继承类的探索 1.1 空类 我们先一步一步慢慢来,从一个空的类开始. //空类 class test { }; int main(int argc, char *argv[]) { test ts; cout << sizeof(t

Android中Handler引起的内存泄露

在Android常用编程中,Handler在进行异步操作并处理返回结果时经常被使用.通常我们的代码会这样实现. 但是,其实上面的代码可能导致内存泄露,当你使用Android lint工具的话,会得到这样的警告 In Android, Handler classes should be static or leaks might occur, Messages enqueued on the application thread’s MessageQueue also retain their t

Android 中 Handler 引起的内存泄露

在Android常用编程中,Handler在进行异步操作并处理返回结果时经常被使用.其实这可能导致内存泄露,代码中哪里可能导致内存泄露,又是如何导致内存泄露的呢?那我们就慢慢分析一下.http://www.jinhusns.com/Products/Download/?type=xcj 在Android常用编程中,Handler在进行异步操作并处理返回结果时经常被使用.通常我们的代码会这样实现. public class SampleActivity extends Activity {   

【转】内部Handler类引起内存泄露

如果您在Activity中定义了一个内部Handler类,如下代码: public class MainActivity extends Activity {       private  Handler mHandler = new Handler() {         @Override         public void handleMessage(Message msg) {             //TODO handle message...         }       

【转载】java项目中经常碰到的内存溢出问题: java.lang.OutOfMemoryError: PermGen space, 堆内存和非堆内存,写的很好,理解很方便

Tomcat Xms Xmx PermSize MaxPermSize 区别 及 java.lang.OutOfMemoryError: PermGen space 解决 解决方案 在 catalina.bat 里的 蓝色代码前加入: 红色代码 rem ----- Execute The Requested Command --------------------------------------- set JAVA_OPTS=%JAVA_OPTS%-server -Xms800m -Xmx1

IntelliJ IDEA中运行Tomcat报内存溢出(java.lang.OutOfMemoryError: PermGen space)

在Run/Debug Configuration中修改Tomcat的VM options,在里面输入以下内容: -server -XX:PermSize=128M -XX:MaxPermSize=256m 修改位置如下图所示: PermSize和MaxPermSize 也不要设置得太大,会浪费物理内存. 来自为知笔记(Wiz)

Android中Handler导致的内存泄露

http://www.androiddesignpatterns.com/2013/01/inner-class-handler-memory-leak.html Consider the following code: 1 2 3 4 5 6 7 8 9 public class SampleActivity extends Activity { private final Handler mLeakyHandler = new Handler() { @Override public voi

eclipse中启动项目报内存溢出问题通过修改配置解决

先停止运行,双击图片中的指定位置,显示具体配置页面. 点击图片中链接 在“arguments”标签卡中对“VM arguments”进行配置,在第一行添加 -Xms256M -Xmx1024M -XX:PermSize=64M -XX:MaxPermSize=128M 点击“apply”按钮.Tomcat停止运行才可以修改.

[转]Java内存溢出详解及解决方案

原文地址:http://blog.csdn.net/xianmiao2009/article/details/49254391 内存溢出与数据库锁表的问题,可以说是开发人员的噩梦,一般的程序异常,总是可以知道在什么时候或是在什么操作步骤上出现了异常,而且根据堆栈信息也很容易定位到程序中是某处出现了问题.内存溢出与锁表则不然,一般现象是操作一般时间后系统越来越慢,直到死机,但并不能明确是在什么操作上出现的,发生的时间点也没有规律,查看日志或查看数据库也不能定位出问题的代码. 更严重的是内存溢出与数