首页 > 解决方案 > 疯狂的 Android Studio 内存使用

问题描述

这是我最后的手段 - 非常希望有人有线索。

我创建了以下自定义OnboardingGuideView。当我将它添加到要显示的布局时,计算机(Mac 或 Windows)开始大喊大叫,内存达到顶部,Android Studio(版本 3.6.1)冻结,好像有一些递归元素阻塞了 IDE。

我已经尝试逐个注释代码片段,直到所有内容都被注释掉。仍然当我启动 Android Studio(不做任何其他事情)时,“派对”开始,一分钟后 AS 被冻结。

如果<dk.tdc.selfapp.ui.common.widgets.onboardingguide.OnboardingGuideView组件被注释,一切正常。

我的王国寻求解决方案!

视图添加到布局:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    style="@style/RowStyle"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">

    <dk.tdc.selfapp.ui.common.widgets.onboardingguide.OnboardingGuideView
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_margin="16dp"
        android:background="@color/errorRed10Color"
        app:descriptionPrefix="tdc_esim_finalize_guide_description"
        app:imagePrefix="esim_finalize_guide_image"
        app:titlePrefix="tdc_esim_finalize_guide_title" />

...

风景:

import android.content.Context
import android.text.Html
import android.util.AttributeSet
import android.view.LayoutInflater
import android.widget.FrameLayout
import android.widget.LinearLayout
import android.widget.TextView
import dk.firm.oldapp.R
import kotlinx.android.synthetic.main.view_onboarding_guide.view.*

class OnboardingGuideView(context: Context, attrs: AttributeSet?, defStyleAttr: Int) : FrameLayout(context, attrs, defStyleAttr) {

    private var itemCount = 0
    private var titlePrefix = ""
    private var descriptionPrefix = ""
    private var imagePrefix = ""

    init {
        LayoutInflater.from(context)
                .inflate(R.layout.view_onboarding_guide, this, true)
        setAttributes(attrs)
    }

    private fun setAttributes(attrs: AttributeSet?) {
        attrs?.let { a ->
            val typedArray = context.obtainStyledAttributes(a,
                    R.styleable.OnboardingGuideView, 0, 0)
            titlePrefix = typedArray.getString(R.styleable.OnboardingGuideView_titlePrefix).toString()
            descriptionPrefix = typedArray.getString(R.styleable.OnboardingGuideView_descriptionPrefix).toString()
            imagePrefix = typedArray.getString(R.styleable.OnboardingGuideView_imagePrefix).toString()

            typedArray.recycle()
        }

        setup()
    }

    private fun setup() {
        itemCount = countResources(titlePrefix, "string")
        val adapter = OnboardingGuideAdapter(context, itemCount, titlePrefix, descriptionPrefix, imagePrefix)
        itemPager.adapter = adapter

        addDotsIndicator()
    }

    private fun countResources(prefix: String, type: String): Int {
        var id: Long = -1
        var count = -1

        while (id != 0L) {
            count++
            id = context.resources.getIdentifier(String.format("%s_%s", prefix, (count + 1)),
                    type, context.packageName).toLong()
        }

        return count
    }

    private fun addDotsIndicator() {
        indicatorView.removeAllViews()

        for (i in 0 until itemCount) {
            val dot = TextView(context)
            dot.text = Html.fromHtml("&#8226;")
            dot.textSize = 35F
            dot.setTextColor(resources.getColor(R.color.appColorBlueLight))

            indicatorView.addView(dot)
        }
    }
}

查看布局 - view_onboarding_guide

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:orientation="vertical">

    <androidx.viewpager.widget.ViewPager
        android:id="@+id/itemPager"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />

    <LinearLayout
        android:id="@+id/indicatorView"
        android:layout_width="match_parent"
        android:layout_height="50dp"
        android:background="@color/status_text_color_red"
        android:orientation="horizontal" />
</LinearLayout>

适配器:

import android.content.Context
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import android.widget.ImageView
import android.widget.LinearLayout
import android.widget.TextView
import androidx.viewpager.widget.PagerAdapter
import dk.firm.oldapp.R
import dk.firm.selfapp.util.Utils

class OnboardingGuideAdapter(
        val context: Context,
        private val itemCount: Int,
        private val titlePrefix: String,
        private val descriptionPrefix: String,
        private val imagePrefix: String) : PagerAdapter() {

    override fun getCount(): Int {
        return itemCount
    }

    override fun isViewFromObject(view: View, o: Any): Boolean {
        return view == o as LinearLayout
    }

    override fun instantiateItem(container: ViewGroup, position: Int): Any {
        val view = LayoutInflater.from(context).inflate(R.layout.view_onboarding_guide_item, container, false)
        val ordinal = position + 1

        // Instantiate views
        val titleView = view.findViewById<TextView>(R.id.onBoardingItemTitle)
        val descriptionView = view.findViewById<TextView>(R.id.onBoardingItemDescription)
        val imageView = view.findViewById<ImageView>(R.id.onBoardingItemImage)

        // Create resource names
        val title = String.format("%s_%s", titlePrefix, ordinal)
        val description = String.format("%s_%s", descriptionPrefix, ordinal)
        val image = String.format("%s_%s", imagePrefix, ordinal)

        // Populate views
        titleView.text = Utils.getStringFromResId(context.resources.getIdentifier(title, "string", context.packageName))
        descriptionView.text = Utils.getStringFromResId(context.resources.getIdentifier(description, "string", context.packageName))
        imageView.setImageResource(context.resources.getIdentifier(image, "drawable", context.packageName))

        container.addView(view)

        return view
    }

    override fun destroyItem(container: ViewGroup, position: Int, o: Any) {
        container.removeView(o as LinearLayout)
    }
}

项目布局 - view_onboarding_guide_item

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:orientation="vertical">

    <androidx.appcompat.widget.AppCompatTextView
        android:id="@+id/onBoardingItemTitle"
        style="@style/TextAppearance.Subtitle1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content" />

    <androidx.appcompat.widget.AppCompatTextView
        android:id="@+id/onBoardingItemDescription"
        style="@style/TextAppearance.Body1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginTop="16dp" />

    <androidx.appcompat.widget.AppCompatImageView
        android:id="@+id/onBoardingItemImage"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />
</LinearLayout>

标签: androidandroid-studio

解决方案


众所周知,Android Studio 会占用内存,这是从 IntelliJ IDE 继承而来的特征。您可以在屏幕的右下角查看 AS 正在使用和可用的内存量:

内存使用指示器和周围的图标

(这可能不会一直显示,如果不显示可以在应用程序设置中打开它的设置)

首先,您可以通过单击此读数告诉 AS 清除未使用的内存!

其次,AS 冻结通常表明它需要比分配的多一点。幸运的是,您可以自己增加限制。如 ProAndroidDev 所示,您可以编辑工作室的 VM 选项:

转到Help > Edit Custom VM Options并编辑以下两个属性:

Xmx: 默认是1280m, PAD 建议把这个改成4096m

XX:MaxPermSize: 默认是350m, PAD 建议把这个改成1024m

有关这些(和其他)值的详细信息,请查看有关自定义 VM 选项和加速 Android Studio 的这篇文章


推荐阅读