Search code examples
androidandroid-spinnerleakcanary

Spinner is leaking when I have ZERO code inside my Activity


I have zero code inside my MainActivity, only the XML layout I created and the Spinner is leaking, not sure why this is happening or how I can fix this. Does anyone know why? I'm using the LeakCanary Library

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }
}

My XML layout for activity_main: ‎‎‎‎‎‎‎‎‎‎‎‎‎‎‎‎‎‎‎‎‎‎

Added nothing else in my mainactivity.java

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="MainActivity">

    <androidx.appcompat.widget.Toolbar
        android:id="@+id/toolbar"
        android:layout_width="match_parent"
        android:layout_height="?attr/actionBarSize"
        android:background="@color/colorPrimary"
        android:elevation="4dp"
        app:contentInsetStart="0dp"
        app:layout_collapseMode="pin"
        app:layout_constraintTop_toTopOf="parent">

        <RelativeLayout
            android:layout_width="match_parent"
            android:layout_height="match_parent">

            <ImageView
                android:id="@+id/toolbar_icon"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_centerVertical="true"
                android:background="?attr/actionBarItemBackground"
                android:paddingStart="12dp"
                android:paddingEnd="3dp"
                app:srcCompat="@drawable/ic_white_printer" />

            <TextView
                android:id="@+id/toolbar_app_name"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_centerVertical="true"
                android:layout_marginEnd="16dp"
                android:layout_toStartOf="@+id/toolbar_more"
                android:layout_toEndOf="@+id/toolbar_icon"
                android:ellipsize="end"
                android:maxLines="2"
                android:text="Test app"
                android:textColor="@color/colorWhite"
                android:textSize="24sp"
                android:textStyle="bold" />

            <ImageView
                android:id="@+id/toolbar_more"
                android:layout_width="wrap_content"
                android:layout_height="match_parent"
                android:layout_alignParentEnd="true"
                android:background="?attr/actionBarItemBackground"
                android:paddingStart="16dp"
                android:paddingEnd="16dp"
                app:srcCompat="@drawable/ic_white_more_vertical" />

        </RelativeLayout>
    </androidx.appcompat.widget.Toolbar>

    <Spinner
        android:id="@+id/spinner_printer_type"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:dropDownVerticalOffset="35dp"
        android:gravity="end"
        android:layout_margin="16dp"
        app:backgroundTint="@color/standardBlue"
        app:layout_constraintTop_toBottomOf="@+id/toolbar"
        app:layout_constraintStart_toStartOf="parent"/>

    <Spinner
        android:id="@+id/spinner_roll_type"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:dropDownVerticalOffset="35dp"
        android:gravity="end"
        android:layout_marginStart="16dp"
        app:layout_constraintHorizontal_bias="1"
        app:backgroundTint="@color/standardBlue"
        app:layout_constraintStart_toEndOf="@+id/spinner_printer_type"
        app:layout_constraintBottom_toBottomOf="@+id/spinner_printer_type"
        app:layout_constraintTop_toTopOf="@+id/spinner_printer_type"/>


    <Button
        android:id="@+id/btn_upload"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_margin="16dp"
        android:background="@color/colorAccent"
        android:text="upload"
        android:textColor="@color/colorWhite"
        app:layout_constraintBottom_toBottomOf="parent" />


</androidx.constraintlayout.widget.ConstraintLayout>

enter image description here

You can try it yourself using

debugImplementation 'com.squareup.leakcanary:leakcanary-android:2.6'

The logs:

┬───
│ GC Root: Global variable in native code
│
├─ android.view.RenderNodeAnimator instance
│    Leaking: NO (MaterialButton↓ is not leaking)
│    ↓ RenderNodeAnimator.mTarget
├─ android.view.RenderNode instance
│    Leaking: NO (MaterialButton↓ is not leaking)
│    ↓ RenderNode.mOwningView
├─ com.google.android.material.button.MaterialButton instance
│    Leaking: NO (ConstraintLayout↓ is not leaking and View attached)
│    View is part of a window view hierarchy
│    View.mAttachInfo is not null (view attached)
│    View.mID = R.id.btn_upload
│    View.mWindowAttachCount = 1
│    mContext instance of com.example.smartlabels.MainActivity with
│    mDestroyed = false
│    ↓ View.mParent
├─ androidx.constraintlayout.widget.ConstraintLayout instance
│    Leaking: NO (AppCompatSpinner↓ is not leaking and View attached)
│    View is part of a window view hierarchy
│    View.mAttachInfo is not null (view attached)
│    View.mWindowAttachCount = 1
│    mContext instance of com.example.smartlabels.MainActivity with
│    mDestroyed = false
│    ↓ ViewGroup.mChildren
├─ android.view.View[] array
│    Leaking: NO (AppCompatSpinner↓ is not leaking)
│    ↓ View[].[1]
├─ androidx.appcompat.widget.AppCompatSpinner instance
│    Leaking: NO (View attached)
│    View is part of a window view hierarchy
│    View.mAttachInfo is not null (view attached)
│    View.mID = R.id.spinner_printer_type
│    View.mWindowAttachCount = 1
│    mPopupContext instance of com.example.smartlabels.MainActivity
│    with mDestroyed = false
│    mContext instance of com.example.smartlabels.MainActivity with
│    mDestroyed = false
│    ↓ AppCompatSpinner.mPopup
│                       ~~~~~~
├─ androidx.appcompat.widget.AppCompatSpinner$DropdownPopup instance
│    Leaking: UNKNOWN
│    Retaining 14.8 kB in 245 objects
│    mContext instance of com.example.smartlabels.MainActivity with
│    mDestroyed = false
│    ↓ ListPopupWindow.mPopup
│                      ~~~~~~
├─ androidx.appcompat.widget.AppCompatPopupWindow instance
│    Leaking: UNKNOWN
│    Retaining 8.8 kB in 154 objects
│    mContext instance of com.example.smartlabels.MainActivity with
│    mDestroyed = false
│    ↓ PopupWindow.mEnterTransition
│                  ~~~~~~~~~~~~~~~~
├─ android.transition.TransitionSet instance
│    Leaking: UNKNOWN
│    Retaining 2.9 kB in 76 objects
│    ↓ Transition.mTargets
│                 ~~~~~~~~
├─ java.util.ArrayList instance
│    Leaking: UNKNOWN
│    Retaining 60 B in 2 objects
│    ↓ ArrayList.elementData
│                ~~~~~~~~~~~
├─ java.lang.Object[] array
│    Leaking: UNKNOWN
│    Retaining 40 B in 1 objects
│    ↓ Object[].[0]
│               ~~~
├─ android.widget.PopupWindow$PopupBackgroundView instance
│    Leaking: UNKNOWN
│    Retaining 4.2 kB in 35 objects
│    View not part of a window view hierarchy
│    View.mAttachInfo is null (view detached)
│    View.mWindowAttachCount = 1
│    mContext instance of com.example.smartlabels.MainActivity with
│    mDestroyed = false
│    ↓ View.mParent
│           ~~~~~~~
╰→ android.widget.PopupWindow$PopupDecorView instance
​     Leaking: YES (ObjectWatcher was watching this because android.widget.
​     PopupWindow$PopupDecorView received View#onDetachedFromWindow() callback)
​     Retaining 2.3 kB in 23 objects
​     key = 8914b38a-5678-4920-89b9-160635defdc7
​     watchDurationMillis = 552488
​     retainedDurationMillis = 547487
​     View not part of a window view hierarchy
​     View.mAttachInfo is null (view detached)
​     View.mWindowAttachCount = 1
​     mContext instance of com.example.smartlabels.MainActivity with
​     mDestroyed = false

METADATA

Build.VERSION.SDK_INT: 27
Build.MANUFACTURER: Google
LeakCanary version: 2.6
App process name: com.example.smartlabels
Stats: LruCache[maxSize=3000,hits=1393,misses=26800,hitRate=4%]
RandomAccess[bytes=1377952,reads=26800,travel=7146276921,range=9328169,size=1209
4081]
Heap dump reason: user request
Analysis duration: 3014 ms

Solution

  • This is a known issue in LeakCanary 2.6, which has been fixed in LeakCanary 2.7

    https://square.github.io/leakcanary/changelog/#finer-grained-root-view-watching

    In version 2.6, LeakCanary added detection of root views retained after View.onDetachedFromWindow(). This helps find more leaks, but unfortunately some Android widgets keep a detached root view around to reattach it later (e.g. spinner). App developers also sometimes do the same with dialogs, keeping a single instance around and calling show() and hide() as needed. As a result, LeakCanary would report leaks that were actually not leaks.

    In version 2.7, the default behavior changed: LeakCanary will continue to detect leaks of toasts, but will ignore root views created by a PopupWindow (which is what Android widgets use). It will also ignore root views created by a dialog by default, and you can turn this back on by setting the leak_canary_watcher_watch_dismissed_dialogs resource boolean to true: