这个Runnable是否可以防止内存泄漏?

Alp*_*ran 7 java android memory-leaks weak-references runnable

我是Java的初学者,并创建了一个简单的Java Android代码片段,在1.5秒后我在Runnable中将TextViewfrom 更改Hello WorldHola Mundo.它完美无缺,基本上WeakReference应该防止这种内存泄漏发生吗?我怀疑是否在设备方向发生时绝对没有内存泄漏.我很想检查这个,但无法在我模拟的Android中改变方向.

这是代码:

package com.example.helloworld;

import android.app.Activity;
import android.os.Bundle;
import android.os.Handler;
import android.widget.TextView;
import android.util.Log;
import java.lang.ref.WeakReference;

public class HelloWorldActivity extends Activity
{
    private Handler h = new Handler();
    private static TextView txtview;
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState)
    {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        txtview = (TextView) findViewById(R.id.mainview);

        h.postDelayed(new WeakRunnable(txtview),1500);
    }

    private static final class WeakRunnable implements Runnable {
        private final WeakReference<TextView> mtextview;

        protected WeakRunnable(TextView textview){
            mtextview = new WeakReference<TextView>(textview);
        }

            @Override
            public void run() {
                TextView textview = mtextview.get();
                if (textview != null) {
                    txtview.setText("Hola Mundo");
                    textview = null; // No idea if setting to null afterwards is a good idea
                }
                Log.d("com.example.helloworld", "" + textview);
            }
    }           

}
Run Code Online (Sandbox Code Playgroud)

编辑

内存泄漏是安全的,但是一些答案也涉及UI线程阻塞.实际上,此代码在主(UI)线程中运行Handler.为了生成一个新线程,我手动生成一个线程,如下所示:

package com.example.helloworld;

import android.app.Activity;
import android.os.Bundle;
import android.os.Handler;
import android.widget.TextView;
import android.util.Log;
import java.lang.ref.WeakReference;

public class HelloWorldActivity extends Activity
{

    private static TextView txtview;
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState)
    {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        txtview = (TextView) findViewById(R.id.mainview);

        Thread t = new Thread(new WeakRunnable(txtview));
        t.start();
    }

    private static final class WeakRunnable implements Runnable {
        private final WeakReference<TextView> mtextview;

        protected WeakRunnable(TextView textview){
            mtextview = new WeakReference<TextView>(textview);
        }

            @Override
            public void run() {
                TextView textview = mtextview.get();
                if (textview != null) {
                    /*
                    try {
                        Thread.sleep(1500);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    */
                    txtview.setText("Hola Mundo");
                    textview = null;
                }
                Log.d("com.example.helloworld", "" + Thread.currentThread().getName()); // Outputs "Thread-<num>" if not running on UI thread
            }
    }           

}
Run Code Online (Sandbox Code Playgroud)

现在的问题是我似乎无法以任何方式延迟产生的线程,否则它的工作原理.

这个:

try {
    Thread.sleep(1500);
} catch (InterruptedException e) {
    e.printStackTrace();
}
Run Code Online (Sandbox Code Playgroud)

使应用程序自行退出,我不明白为什么.有些东西告诉我,我是以错误的方式推迟它.

EDIT2

感谢链接@EugenMatynov给我:从android中的另一个线程更新ui我理解为什么应用程序退出了.这一切都归结为您无法从主线程以外的线程调用UI方法的原因.从另一个线程更新UI是不好的做法.

Bla*_*elt 6

我怀疑是否在设备方向发生时绝对没有内存泄漏.

它可能是.持续1.5秒.清空队列后,可以对处理程序进行垃圾回收,也可以使用旧的Activity.要安全地覆盖onPause,并调用handler.removeCallbacks(null);以清除Handler的队列

  • 在我的例子中,我不得不调用`handler.removeCallbacksAndMessages(null);`,因为从android版本23实现开始,`removeCallbacks();`当你将null作为参数传递时什么都不做(参见MessageQueue的`removeMessages(); `). (4认同)

mml*_*loo 4

我认为如果您使用以下代码,您的代码就不会泄漏:

private static Handler h = new Handler(); 
Run Code Online (Sandbox Code Playgroud)

或者

txtview.postDelayed(new WeakRunnable(txtview),1500);
Run Code Online (Sandbox Code Playgroud)

因为您已将视图存储为 WeakReference。方法:

txtview.postDelayed(new WeakRunnable(txtview),1500);
Run Code Online (Sandbox Code Playgroud)

只需调用 UI 线程的主处理程序,这样如果活动被销毁,视图将为 null,并且可运行对象不会执行任何操作。

另外,由于弱引用,活动可以被垃圾收集,因为没有对它的强引用。