Утечка памяти в Android при просмотре текста - LeakCanary (утечка может быть проигнорирована)

У меня возникает следующая утечка памяти, отображаемая LeakCanary, когда я перехожу с заставки на основной экран. Я понимаю, что это ожидаемая утечка из-за ошибки в самой ОС Android, но есть ли способ, которым я могу избежать этого (устанавливая специфику некоторого TextView где-нибудь?)

D/LeakCanary﹕ * LEAK CAN BE IGNORED.
D/LeakCanary﹕ * com.gmspartnersltd.earthmiles.views.ActivitySignUp_ has leaked:
D/LeakCanary﹕ * GC ROOT static android.text.TextLine.sCached
D/LeakCanary﹕ * references array android.text.TextLine[].[1]
D/LeakCanary﹕ * references android.text.TextLine.mCharacterStyleSpanSet
D/LeakCanary﹕ * references android.text.SpanSet.spans
D/LeakCanary﹕ * references array android.text.style.CharacterStyle[].[1]
D/LeakCanary﹕ * references com.gmspartnersltd.earthmiles.views.ActivitySignUp$2.this$0 (anonymous class extends android.text.style.ClickableSpan)
D/LeakCanary﹕ * leaks com.gmspartnersltd.earthmiles.views.ActivitySignUp_ instance
D/LeakCanary﹕ [ 05-22 08:54:52.160 13969:18091 D/LeakCanary ]
    * Reference Key: bb8124a9-2829-4ff3-8ded-13cf35f80f54
D/LeakCanary﹕ * Device: Genymotion generic Google Nexus 5 - 5.0.0 - API 21 - 1080x1920 vbox86p
D/LeakCanary﹕ * Android Version: 5.0 API: 21 LeakCanary: 1.3.1
D/LeakCanary﹕ * Durations: watch=10898ms, gc=137ms, heap dump=5529ms, analysis=9193ms
D/LeakCanary﹕ [ 05-22 08:54:52.160 13969:18091 D/LeakCanary ]
    * Details:
D/LeakCanary﹕ * Class android.text.TextLine
D/LeakCanary﹕ |   static $staticOverhead = byte[] [id=0x70622169;length=24;size=40]
D/LeakCanary﹕ |   static sCached = android.text.TextLine[] [id=0x70775010;length=3]
D/LeakCanary﹕ |   static DEBUG = false
D/LeakCanary﹕ |   static TAB_INCREMENT = 20
D/LeakCanary﹕ * Array of android.text.TextLine[]
D/LeakCanary﹕ |   [0] = android.text.TextLine [id=0x1309a2e0]
D/LeakCanary﹕ |   [1] = android.text.TextLine [id=0x12eed650]
D/LeakCanary﹕ |   [2] = null
D/LeakCanary﹕ * Instance of android.text.TextLine
D/LeakCanary﹕ |   static $staticOverhead = byte[] [id=0x70622169;length=24;size=40]
D/LeakCanary﹕ |   static sCached = android.text.TextLine[] [id=0x70775010;length=3]
D/LeakCanary﹕ |   static DEBUG = false
D/LeakCanary﹕ |   static TAB_INCREMENT = 20
D/LeakCanary﹕ |   mCharacterStyleSpanSet = android.text.SpanSet [id=0x12e32d80]
D/LeakCanary﹕ |   mChars = null
D/LeakCanary﹕ |   mDirections = null
D/LeakCanary﹕ |   mMetricAffectingSpanSpanSet = android.text.SpanSet [id=0x12e32d60]
D/LeakCanary﹕ |   mPaint = null
D/LeakCanary﹕ |   mReplacementSpanSpanSet = android.text.SpanSet [id=0x12e32da0]
D/LeakCanary﹕ |   mSpanned = android.text.SpannedString [id=0x132dabe0]
D/LeakCanary﹕ |   mTabs = null
D/LeakCanary﹕ |   mText = null
D/LeakCanary﹕ |   mWorkPaint = android.text.TextPaint [id=0x1300f5c0]
D/LeakCanary﹕ |   mCharsValid = false
D/LeakCanary﹕ |   mDir = 1
D/LeakCanary﹕ |   mHasTabs = false
D/LeakCanary﹕ |   mLen = 1
D/LeakCanary﹕ |   mStart = 0
D/LeakCanary﹕ * Instance of android.text.SpanSet
D/LeakCanary﹕ |   classType = java.lang.Class [id=0x703bb448;name=android.text.style.CharacterStyle]
D/LeakCanary﹕ |   spanEnds = int[] [id=0x1309fd60;length=2;size=24]
D/LeakCanary﹕ |   spanFlags = int[] [id=0x1309fda0;length=2;size=24]
D/LeakCanary﹕ |   spanStarts = int[] [id=0x1309fd20;length=2;size=24]
D/LeakCanary﹕ |   spans = android.text.style.CharacterStyle[] [id=0x1309fce0;length=2]
D/LeakCanary﹕ |   numberOfSpans = 1
D/LeakCanary﹕ * Array of android.text.style.CharacterStyle[]
D/LeakCanary﹕ |   [0] = null
D/LeakCanary﹕ |   [1] = com.gmspartnersltd.earthmiles.views.ActivitySignUp$2 [id=0x130952c0]
D/LeakCanary﹕ * Instance of com.gmspartnersltd.earthmiles.views.ActivitySignUp$2
D/LeakCanary﹕ |   this$0 = com.gmspartnersltd.earthmiles.views.ActivitySignUp_ [id=0x13361800]
D/LeakCanary﹕ * Instance of com.gmspartnersltd.earthmiles.views.ActivitySignUp_
D/LeakCanary﹕ |   onViewChangedNotifier_ = org.androidannotations.api.view.OnViewChangedNotifier [id=0x13052dc0]
D/LeakCanary﹕ |   birthday = null
D/LeakCanary﹕ |   buttonNext = android.support.v7.widget.AppCompatButton [id=0x13459c00]
D/LeakCanary﹕ |   confirmPassword = java.lang.String [id=0x132f01a0]
D/LeakCanary﹕ |   editTextConformPassword = android.support.v7.widget.AppCompatEditText [id=0x13458400]
D/LeakCanary﹕ |   editTextEmail = android.support.v7.widget.AppCompatEditText [id=0x1339b000]
D/LeakCanary﹕ |   editTextFirstName = android.support.v7.widget.AppCompatEditText [id=0x13396c00]
D/LeakCanary﹕ |   editTextLastName = android.support.v7.widget.AppCompatEditText [id=0x13398800]
D/LeakCanary﹕ |   editTextPassword = android.support.v7.widget.AppCompatEditText [id=0x13456c00]
D/LeakCanary﹕ |   email = java.lang.String [id=0x132f0040]
D/LeakCanary﹕ |   facebook = com.facebook.android.Facebook [id=0x1307ee00]
D/LeakCanary﹕ |   fbUserId = null
D/LeakCanary﹕ |   firstName = java.lang.String [id=0x132f0080]
D/LeakCanary﹕ |   gender = null
D/LeakCanary﹕ |   lastName = java.lang.String [id=0x132f00e0]
D/LeakCanary﹕ |   location = null
D/LeakCanary﹕ |   mAsyncRunner = com.facebook.android.AsyncFacebookRunner [id=0x130952a0]
D/LeakCanary﹕ |   password = java.lang.String [id=0x132f0140]
D/LeakCanary﹕ |   termsOfUse = android.support.v7.widget.AppCompatTextView [id=0x1345a000]
D/LeakCanary﹕ |   text = android.text.SpannableString [id=0x1310eaa0]
D/LeakCanary﹕ |   fromFacebook = false
D/LeakCanary﹕ |   etHelpMessage = null
D/LeakCanary﹕ |   mProgressHUD = null
D/LeakCanary﹕ |   positiveAction = null
D/LeakCanary﹕ |   showBusyAnimationRequesterCount = 0
D/LeakCanary﹕ |   mDelegate = android.support.v7.app.AppCompatDelegateImplV11 [id=0x12ecfd80]
D/LeakCanary﹕ |   mAllLoaderManagers = android.support.v4.util.SimpleArrayMap [id=0x131d80a0]
D/LeakCanary﹕ |   mContainer = android.support.v4.app.FragmentActivity$2 [id=0x13052db0]
D/LeakCanary﹕ |   mFragments = android.support.v4.app.FragmentManagerImpl [id=0x12f7cf60]
D/LeakCanary﹕ |   mHandler = android.support.v4.app.FragmentActivity$1 [id=0x1310ea80]
D/LeakCanary﹕ |   mLoaderManager = null
D/LeakCanary﹕ |   mCheckedForLoaderManager = true
D/LeakCanary﹕ |   mCreated = true
D/LeakCanary﹕ |   mLoadersStarted = false
D/LeakCanary﹕ |   mOptionsMenuInvalidated = false
D/LeakCanary﹕ |   mReallyStopped = true
D/LeakCanary﹕ |   mResumed = false
D/LeakCanary﹕ |   mRetaining = false
D/LeakCanary﹕ |   mStopped = true
D/LeakCanary﹕ |   mActionBar = null
D/LeakCanary﹕ |   mActivityInfo = android.content.pm.ActivityInfo [id=0x12db0180]
D/LeakCanary﹕ |   mActivityTransitionState = android.app.ActivityTransitionState [id=0x1304b600]
D/LeakCanary﹕ |   mAllLoaderManagers = android.util.ArrayMap [id=0x131c9d00]
D/LeakCanary﹕ |   mApplication = com.gmspartnersltd.earthmiles.globalstate.App [id=0x12c6e8c0]
D/LeakCanary﹕ |   mComponent = android.content.ComponentName [id=0x12f64150]
D/LeakCanary﹕ |   mContainer = android.app.Activity$1 [id=0x13052d70]
D/LeakCanary﹕ |   mCurrentConfig = android.content.res.Configuration [id=0x12f97520]
D/LeakCanary﹕ |   mDecor = null
D/LeakCanary﹕ |   mDefaultKeySsb = null
D/LeakCanary﹕ |   mEmbeddedID = null
D/LeakCanary﹕ |   mEnterTransitionListener = android.app.SharedElementCallback$1 [id=0x70765ba8]
D/LeakCanary﹕ |   mExitTransitionListener = android.app.SharedElementCallback$1 [id=0x70765ba8]
D/LeakCanary﹕ |   mFragments = android.app.FragmentManagerImpl [id=0x12f7cef0]
D/LeakCanary﹕ |   mHandler = android.os.Handler [id=0x1310ea60]
D/LeakCanary﹕ |   mInstanceTracker = android.os.StrictMode$InstanceTracker [id=0x13052d90]
D/LeakCanary﹕ |   mInstrumentation = android.app.Instrumentation [id=0x12c33f70]
D/LeakCanary﹕ |   mIntent = android.content.Intent [id=0x12f3b300]
D/LeakCanary﹕ |   mLastNonConfigurationInstances = null
D/LeakCanary﹕ |   mLoaderManager = null
D/LeakCanary﹕ |   mMainThread = android.app.ActivityThread [id=0x12c2b100]
D/LeakCanary﹕ |   mManagedCursors = java.util.ArrayList [id=0x1310ea40]
D/LeakCanary﹕ |   mManagedDialogs = null
D/LeakCanary﹕ |   mMenuInflater = null
D/LeakCanary﹕ |   mParent = null
D/LeakCanary﹕ |   mResultData = null
D/LeakCanary﹕ |   mSearchManager = null
D/LeakCanary﹕ |   mTitle = java.lang.String [id=0x12e6d7e0]
D/LeakCanary﹕ |   mToken = android.os.BinderProxy [id=0x12fe86a0]
D/LeakCanary﹕ |   mTranslucentCallback = null
D/LeakCanary﹕ |   mUiThread = java.lang.Thread [id=0x73b43540]
D/LeakCanary﹕ |   mVoiceInteractor = null
D/LeakCanary﹕ |   mWindow = com.android.internal.policy.impl.PhoneWindow [id=0x12e5d580]
D/LeakCanary﹕ |   mWindowManager = android.view.WindowManagerImpl [id=0x1310ed20]
D/LeakCanary﹕ |   mCalled = true
D/LeakCanary﹕ |   mChangeCanvasToTranslucent = false
D/LeakCanary﹕ |   mChangingConfigurations = false
D/LeakCanary﹕ |   mCheckedForLoaderManager = true
D/LeakCanary﹕ |   mConfigChangeFlags = 0
D/LeakCanary﹕ |   mDefaultKeyMode = 0
D/LeakCanary﹕ |   mDestroyed = true
D/LeakCanary﹕ |   mDoReportFullyDrawn = false
D/LeakCanary﹕ |   mEnableDefaultActionBarUp = false
D/LeakCanary﹕ |   mFinished = true
D/LeakCanary﹕ |   mIdent = 24993652
D/LeakCanary﹕ |   mLoadersStarted = false
D/LeakCanary﹕ |   mResultCode = 0
D/LeakCanary﹕ |   mResumed = false
D/LeakCanary﹕ |   mStartedActivity = false
D/LeakCanary﹕ |   mStopped = true
D/LeakCanary﹕ |   mTemporaryPause = false
D/LeakCanary﹕ |   mTitleColor = 0
D/LeakCanary﹕ |   mTitleReady = true
D/LeakCanary﹕ |   mVisibleBehind = false
D/LeakCanary﹕ |   mVisibleFromClient = true
D/LeakCanary﹕ |   mVisibleFromServer = false
D/LeakCanary﹕ |   mWindowAdded = true
D/LeakCanary﹕ |   mInflater = com.android.internal.policy.impl.PhoneLayoutInflater [id=0x13152580]
D/LeakCanary﹕ |   mOverrideConfiguration = null
D/LeakCanary﹕ |   mResources = android.content.res.Resources [id=0x12c33f20]
D/LeakCanary﹕ |   mTheme = android.content.res.Resources$Theme [id=0x1310ed40]
D/LeakCanary﹕ |   mThemeResource = 2131689670
D/LeakCanary﹕ |   mBase = android.app.ContextImpl [id=0x12c81100]

1 ответ

Решение

От AndroidExcludedRefs.java:

  // TextLine.sCached is a pool of 3 TextLine instances. TextLine.recycle() has had at least two
  // bugs that created memory leaks by not correctly clearing the recycled TextLine instances.
  // The first was fixed in android-5.1.0_r1:
  // https://github.com/android/platform_frameworks_base/commit
  // /893d6fe48d37f71e683f722457bea646994a10bf

  // The second was fixed, not released yet:
  // https://github.com/android/platform_frameworks_base/commit
  // /b3a9bc038d3a218b1dbdf7b5668e3d6c12be5ee4

  // Hack: to fix this, you could access TextLine.sCached and clear the pool every now and then
  // (e.g. on activity destroy).

Шаг 1: Доступ TextLine.sCached

public static class Utils {
    private static final Field TEXT_LINE_CACHED;

    static {
        Field textLineCached = null;
        try {
            textLineCached = Class.forName("android.text.TextLine").getDeclaredField("sCached");
            textLineCached.setAccessible(true);
        } catch (Exception ex) {
            ex.printStackTrace();
        }
        TEXT_LINE_CACHED = textLineCached;
    }

    public static void clearTextLineCache() {
        // If the field was not found for whatever reason just return.
        if (TEXT_LINE_CACHED == null) return;

        Object cached = null;
        try {
            // Get reference to the TextLine sCached array.
            cached = TEXT_LINE_CACHED.get(null);
        } catch (Exception ex) {
            //
        }
        if (cached != null) {
            // Clear the array.
            for (int i = 0, size = Array.getLength(cached); i < size; i ++) {
                Array.set(cached, i, null);
            }
        }
    }

    private Utils() {}
}

Шаг 2: Очистить бассейн

Вызов Utils.clearTextLineCache() при необходимости.

Другие вопросы по тегам