android - 疯狂的 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("•")
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>
解决方案
众所周知,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 的这篇文章。