Nota : nella maggior parte dei casi, ti consigliamo di utilizzare lo Libreria Glide per recuperare, decodificare visualizzare le bitmap nella tua app. Glide astrae la maggior parte delle a una maggiore complessità nella gestione altre attività relative all'utilizzo di bitmap e altre immagini su Android. Per informazioni sull'utilizzo e sul download di Glide, visita il Repository Glide su GitHub.
Oltre ai passaggi descritti nella sezione Memorizzazione dei bitmap nella cache,
ci sono azioni specifiche che puoi fare per facilitare la garbage collection
e il riutilizzo delle bitmap. La strategia consigliata dipende dalle versioni
di Android che hai scelto come target. L'app di esempio BitmapFun
inclusa in
questo corso ti mostra come progettare un'app in modo che funzioni in modo
diverse versioni di Android.
Per gettare le basi di questa lezione, ecco come la gestione delle la memoria bitmap si è evoluta:
- Su Android 2.2 (livello API 8) e versioni precedenti, quando viene raccolta, i thread dell'app vengono arrestati. Questo causa un ritardo che può peggiorare le prestazioni. Android 2.3 aggiunge la garbage collection simultanea, il che significa la memoria viene recuperata subito dopo che non viene più fatto riferimento a una bitmap.
- Su Android 2.3.3 (livello API 10) e versioni precedenti, i dati dei pixel di supporto per un viene archiviata nella memoria nativa. È separato dalla bitmap stessa, che è archiviata nell'heap Dalvik. I dati dei pixel nella memoria nativa sono non rilasciate in modo prevedibile, causando potenzialmente un'applicazione per superare brevemente i limiti di memoria e arrestarsi in modo anomalo. Da Android 3.0 (livello API 11) ad Android 7.1 (livello API 25), i dati dei pixel vengono archiviati Elvik insieme alla bitmap associata. In Android 8.0 (livello API 26), e superiori, i dati dei pixel bitmap vengono archiviati nell'heap nativo.
Le seguenti sezioni descrivono come ottimizzare la memoria bitmap per le varie versioni di Android.
Gestisci la memoria su Android 2.3.3 e versioni precedenti
Su Android 2.3.3 (livello API 10) e versioni precedenti, utilizzando
recycle()
è consigliato. Se nella tua app vengono visualizzati
grandi quantità di dati bitmap,
incontrerai
OutOfMemoryError
errore. La
Il metodo recycle()
consente a un'app
per recuperare la memoria il prima possibile.
Attenzione: devi usare
recycle()
solo quando hai la certezza che
non viene più usato bitmap. Se chiami recycle()
e successivamente prova a tracciare la bitmap, verrà visualizzato l'errore:
"Canvas: trying to use a recycled bitmap"
.
Il seguente snippet di codice fornisce un esempio di chiamata
recycle()
. Usa il conteggio dei riferimenti
(nelle variabili mDisplayRefCount
e mCacheRefCount
) da monitorare
se una bitmap è attualmente visualizzata o nella cache. La
ricicla la bitmap quando queste condizioni sono soddisfatte:
- Il conteggio dei riferimenti sia per
mDisplayRefCount
che permCacheRefCount
è 0. - La bitmap non è
null
e non è stata ancora riciclata.
Kotlin
private var cacheRefCount: Int = 0 private var displayRefCount: Int = 0 ... // Notify the drawable that the displayed state has changed. // Keep a count to determine when the drawable is no longer displayed. fun setIsDisplayed(isDisplayed: Boolean) { synchronized(this) { if (isDisplayed) { displayRefCount++ hasBeenDisplayed = true } else { displayRefCount-- } } // Check to see if recycle() can be called. checkState() } // Notify the drawable that the cache state has changed. // Keep a count to determine when the drawable is no longer being cached. fun setIsCached(isCached: Boolean) { synchronized(this) { if (isCached) { cacheRefCount++ } else { cacheRefCount-- } } // Check to see if recycle() can be called. checkState() } @Synchronized private fun checkState() { // If the drawable cache and display ref counts = 0, and this drawable // has been displayed, then recycle. if (cacheRefCount <= 0 && displayRefCount <= 0 && hasBeenDisplayed && hasValidBitmap() ) { getBitmap()?.recycle() } } @Synchronized private fun hasValidBitmap(): Boolean = getBitmap()?.run { !isRecycled } ?: false
Java
private int cacheRefCount = 0; private int displayRefCount = 0; ... // Notify the drawable that the displayed state has changed. // Keep a count to determine when the drawable is no longer displayed. public void setIsDisplayed(boolean isDisplayed) { synchronized (this) { if (isDisplayed) { displayRefCount++; hasBeenDisplayed = true; } else { displayRefCount--; } } // Check to see if recycle() can be called. checkState(); } // Notify the drawable that the cache state has changed. // Keep a count to determine when the drawable is no longer being cached. public void setIsCached(boolean isCached) { synchronized (this) { if (isCached) { cacheRefCount++; } else { cacheRefCount--; } } // Check to see if recycle() can be called. checkState(); } private synchronized void checkState() { // If the drawable cache and display ref counts = 0, and this drawable // has been displayed, then recycle. if (cacheRefCount <= 0 && displayRefCount <= 0 && hasBeenDisplayed && hasValidBitmap()) { getBitmap().recycle(); } } private synchronized boolean hasValidBitmap() { Bitmap bitmap = getBitmap(); return bitmap != null && !bitmap.isRecycled(); }
Gestione della memoria su Android 3.0 e versioni successive
Android 3.0 (livello API 11) introduce la
BitmapFactory.Options.inBitmap
. Se questa opzione è impostata, i metodi di decodifica che prendono
Options
oggetto
tenterà di riutilizzare una bitmap esistente durante il caricamento dei contenuti. Ciò significa
che la memoria della bitmap viene riutilizzata, con un conseguente miglioramento delle prestazioni, e
rimuovendo sia l'allocazione che la de-allocazione della memoria. Esistono, però, alcune limitazioni relative al modo in cui
È possibile utilizzare inBitmap
. In particolare, prima di Android
4.4 (livello API 19), sono supportate solo bitmap di dimensioni uguali. Per maggiori dettagli, consulta
documentazione di inBitmap
.
Salva una bitmap per un utilizzo futuro
Lo snippet seguente mostra come una bitmap esistente viene archiviata per
in un secondo momento nell'app di esempio. Se un'app è installata su Android 3.0 o versioni successive e
una bitmap viene rimossa da LruCache
,
viene posizionato un riferimento soft alla bitmap
in HashSet
, per un eventuale riutilizzo in un secondo momento con
inBitmap
:
Kotlin
var reusableBitmaps: MutableSet<SoftReference<Bitmap>>? = null private lateinit var memoryCache: LruCache<String, BitmapDrawable> // If you're running on Honeycomb or newer, create a // synchronized HashSet of references to reusable bitmaps. if (Utils.hasHoneycomb()) { reusableBitmaps = Collections.synchronizedSet(HashSet<SoftReference<Bitmap>>()) } memoryCache = object : LruCache<String, BitmapDrawable>(cacheParams.memCacheSize) { // Notify the removed entry that is no longer being cached. override fun entryRemoved( evicted: Boolean, key: String, oldValue: BitmapDrawable, newValue: BitmapDrawable ) { if (oldValue is RecyclingBitmapDrawable) { // The removed entry is a recycling drawable, so notify it // that it has been removed from the memory cache. oldValue.setIsCached(false) } else { // The removed entry is a standard BitmapDrawable. if (Utils.hasHoneycomb()) { // We're running on Honeycomb or later, so add the bitmap // to a SoftReference set for possible use with inBitmap later. reusableBitmaps?.add(SoftReference(oldValue.bitmap)) } } } }
Java
Set<SoftReference<Bitmap>> reusableBitmaps; private LruCache<String, BitmapDrawable> memoryCache; // If you're running on Honeycomb or newer, create a // synchronized HashSet of references to reusable bitmaps. if (Utils.hasHoneycomb()) { reusableBitmaps = Collections.synchronizedSet(new HashSet<SoftReference<Bitmap>>()); } memoryCache = new LruCache<String, BitmapDrawable>(cacheParams.memCacheSize) { // Notify the removed entry that is no longer being cached. @Override protected void entryRemoved(boolean evicted, String key, BitmapDrawable oldValue, BitmapDrawable newValue) { if (RecyclingBitmapDrawable.class.isInstance(oldValue)) { // The removed entry is a recycling drawable, so notify it // that it has been removed from the memory cache. ((RecyclingBitmapDrawable) oldValue).setIsCached(false); } else { // The removed entry is a standard BitmapDrawable. if (Utils.hasHoneycomb()) { // We're running on Honeycomb or later, so add the bitmap // to a SoftReference set for possible use with inBitmap later. reusableBitmaps.add (new SoftReference<Bitmap>(oldValue.getBitmap())); } } } .... }
Usa una bitmap esistente
Nell'app in esecuzione, i metodi di decoder verificano se è presente bitmap che possono utilizzare. Ad esempio:
Kotlin
fun decodeSampledBitmapFromFile( filename: String, reqWidth: Int, reqHeight: Int, cache: ImageCache ): Bitmap { val options: BitmapFactory.Options = BitmapFactory.Options() ... BitmapFactory.decodeFile(filename, options) ... // If we're running on Honeycomb or newer, try to use inBitmap. if (Utils.hasHoneycomb()) { addInBitmapOptions(options, cache) } ... return BitmapFactory.decodeFile(filename, options) }
Java
public static Bitmap decodeSampledBitmapFromFile(String filename, int reqWidth, int reqHeight, ImageCache cache) { final BitmapFactory.Options options = new BitmapFactory.Options(); ... BitmapFactory.decodeFile(filename, options); ... // If we're running on Honeycomb or newer, try to use inBitmap. if (Utils.hasHoneycomb()) { addInBitmapOptions(options, cache); } ... return BitmapFactory.decodeFile(filename, options); }
Lo snippet successivo mostra il metodo addInBitmapOptions()
chiamato nello snippet
sopra lo snippet. Cerca una bitmap esistente da impostare come valore per
inBitmap
. Tieni presente che
imposta solo un valore per inBitmap
se trova una corrispondenza adatta (il tuo codice non deve mai dare per scontato che verrà trovata una corrispondenza):
Kotlin
private fun addInBitmapOptions(options: BitmapFactory.Options, cache: ImageCache?) { // inBitmap only works with mutable bitmaps, so force the decoder to // return mutable bitmaps. options.inMutable = true // Try to find a bitmap to use for inBitmap. cache?.getBitmapFromReusableSet(options)?.also { inBitmap -> // If a suitable bitmap has been found, set it as the value of // inBitmap. options.inBitmap = inBitmap } } // This method iterates through the reusable bitmaps, looking for one // to use for inBitmap: fun getBitmapFromReusableSet(options: BitmapFactory.Options): Bitmap? { mReusableBitmaps?.takeIf { it.isNotEmpty() }?.let { reusableBitmaps -> synchronized(reusableBitmaps) { val iterator: MutableIterator<SoftReference<Bitmap>> = reusableBitmaps.iterator() while (iterator.hasNext()) { iterator.next().get()?.let { item -> if (item.isMutable) { // Check to see it the item can be used for inBitmap. if (canUseForInBitmap(item, options)) { // Remove from reusable set so it can't be used again. iterator.remove() return item } } else { // Remove from the set if the reference has been cleared. iterator.remove() } } } } } return null }
Java
private static void addInBitmapOptions(BitmapFactory.Options options, ImageCache cache) { // inBitmap only works with mutable bitmaps, so force the decoder to // return mutable bitmaps. options.inMutable = true; if (cache != null) { // Try to find a bitmap to use for inBitmap. Bitmap inBitmap = cache.getBitmapFromReusableSet(options); if (inBitmap != null) { // If a suitable bitmap has been found, set it as the value of // inBitmap. options.inBitmap = inBitmap; } } } // This method iterates through the reusable bitmaps, looking for one // to use for inBitmap: protected Bitmap getBitmapFromReusableSet(BitmapFactory.Options options) { Bitmap bitmap = null; if (reusableBitmaps != null && !reusableBitmaps.isEmpty()) { synchronized (reusableBitmaps) { final Iterator<SoftReference<Bitmap>> iterator = reusableBitmaps.iterator(); Bitmap item; while (iterator.hasNext()) { item = iterator.next().get(); if (null != item && item.isMutable()) { // Check to see it the item can be used for inBitmap. if (canUseForInBitmap(item, options)) { bitmap = item; // Remove from reusable set so it can't be used again. iterator.remove(); break; } } else { // Remove from the set if the reference has been cleared. iterator.remove(); } } } } return bitmap; }
Infine, questo metodo determina se una bitmap candidata
soddisfi i criteri delle dimensioni da utilizzare
inBitmap
:
Kotlin
private fun canUseForInBitmap(candidate: Bitmap, targetOptions: BitmapFactory.Options): Boolean { return if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { // From Android 4.4 (KitKat) onward we can re-use if the byte size of // the new bitmap is smaller than the reusable bitmap candidate // allocation byte count. val width = ceil((targetOptions.outWidth * 1.0f / targetOptions.inSampleSize).toDouble()).toInt() val height = ceil((targetOptions.outHeight * 1.0f / targetOptions.inSampleSize).toDouble()).toInt() val byteCount: Int = width * height * getBytesPerPixel(candidate.config) byteCount <= candidate.allocationByteCount } else { // On earlier versions, the dimensions must match exactly and the inSampleSize must be 1 candidate.width == targetOptions.outWidth && candidate.height == targetOptions.outHeight && targetOptions.inSampleSize == 1 } } /** * A helper function to return the byte usage per pixel of a bitmap based on its configuration. */ private fun getBytesPerPixel(config: Bitmap.Config): Int { return when (config) { Bitmap.Config.ARGB_8888 -> 4 Bitmap.Config.RGB_565, Bitmap.Config.ARGB_4444 -> 2 Bitmap.Config.ALPHA_8 -> 1 else -> 1 } }
Java
static boolean canUseForInBitmap( Bitmap candidate, BitmapFactory.Options targetOptions) { if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) { // From Android 4.4 (KitKat) onward we can re-use if the byte size of // the new bitmap is smaller than the reusable bitmap candidate // allocation byte count. int width = (int) Math.ceil(targetOptions.outWidth * 1.0f / targetOptions.inSampleSize); int height = (int) Math.ceil(targetOptions.outHeight * 1.0f / targetOptions.inSampleSize); int byteCount = width * height * getBytesPerPixel(candidate.getConfig()); return byteCount <= candidate.getAllocationByteCount(); } // On earlier versions, the dimensions must match exactly and the inSampleSize must be 1 return candidate.getWidth() == targetOptions.outWidth && candidate.getHeight() == targetOptions.outHeight && targetOptions.inSampleSize == 1; } /** * A helper function to return the byte usage per pixel of a bitmap based on its configuration. */ static int getBytesPerPixel(Config config) { if (config == Config.ARGB_8888) { return 4; } else if (config == Config.RGB_565) { return 2; } else if (config == Config.ARGB_4444) { return 2; } else if (config == Config.ALPHA_8) { return 1; } return 1; }