android 管理Bitmap內存 - 開發文檔翻譯

由於本人英文能力實在有限,不足之初敬請諒解

本博客只要沒有註明“轉”,那麼均爲原創,轉貼請註明本博客鏈接鏈接

 

Managing Bitmap Memory

管理Bitmap內存

 

In addition to the steps described in Caching Bitmaps, there are specific things you can do to facilitate garbage collection and bitmap reuse. 

The recommended strategy depends on which version(s) of Android you are targeting. 

The BitmapFun sample app included with this class shows you how to design your app to work efficiently across different versions of Android.

除了在Caching Bitmaps裏描述的,還有一些具體的事情有助於垃圾回收和bitmap的重用

建議的策略取決於你針對的Android版本。

BitmapFun樣本應用中包括這個類,它展示給你如何設計你的應用使得跨版本工作更有效率

 

To set the stage for this lesson, here is how Android's management of bitmap memory has evolved:

爲課程打基礎,下面是android的bitmap內存管理是如何演化的

 

On Android Android 2.2 (API level 8) and lower, when garbage collection occurs, your app's threads get stopped. 

This causes a lag that can degrade performance. 

Android 2.3 adds concurrent garbage collection, which means that the memory is reclaimed soon after a bitmap is no longer referenced.

在Android2.2(API 8)以及更低的版本中,當發生垃圾回收時,你的應用線程會停止。

這會導致延遲,使得性能降低

Android2.3添加了併發垃圾收集,這意爲着一個bitmap不再被引用的時候,內存很快就被回收

 

On Android 2.3.3 (API level 10) and lower, the backing pixel data for a bitmap is stored in native memory. 

It is separate from the bitmap itself, which is stored in the Dalvik heap. 

The pixel data in native memory is not released in a predictable manner, potentially causing an application to briefly exceed its memory limits and crash. 

As of Android 3.0 (API Level 11), the pixel data is stored on the Dalvik heap along with the associated bitmap.

在Android2.3.3(API 10)和更低的版本中,bitmap的像素數據是存儲在native內存中的

它獨立於bitmap本身,bitmap是存儲在Dalvik堆中的

在native內存中的bitmap的像素數據不會在一個可預測的行爲之釋放,潛在的導致應用內存超過限制並且崩潰

在android3.0(API 11)中,bitmap的像素數據存儲在Dalvik堆中於bitmap相關聯

 

The following sections describe how to optimize bitmap memory management for different Android versions.

下面章節講述在不同的android版本中,如何最優化bitmap內存管理

 

 

Manage Memory on Android 2.3.3 and Lower

在android2.3.3以及更低的版本中管理內存

 

On Android 2.3.3 (API level 10) and lower, using recycle() is recommended. 

If you're displaying large amounts of bitmap data in your app, you're likely to run into OutOfMemoryError errors. 

The recycle() method allows an app to reclaim memory as soon as possible.

在android2.3.3以及更低的版本中,建議使用recycle()

如果你在你的應用中大量的顯示bitmap數據,你很可能得到OutOfMemoryError錯誤

recycle()方法允許一個應用盡快回收內存

 

Caution: You should use recycle() only when you are sure that the bitmap is no longer being used. 

If you call recycle() and later attempt to draw the bitmap, you will get the error: "Canvas: trying to use a recycled bitmap".

注意:僅當你確定這個bitmap不會再被使用的時候,你才應該使用recycle()

如果你調用了recycle(),之後又試圖繪製這個bitmap,你會得到 錯誤:“Canvas: trying to use a recycled bitmap”

 

The following code snippet gives an example of calling recycle(). 

It uses reference counting (in the variables mDisplayRefCount and mCacheRefCount) to track whether a bitmap is currently being displayed or in the cache. 

The code recycles the bitmap when these conditions are met:

下面的代碼片斷給出了一個調用recycle()例子

它使用引用計數(在變量mDisplayRefCount 和 mCacheRefCount中)來跟蹤一個bitmap當前正在被顯示還是在緩存中

代碼回收bitmap需要的條件是:

 

The reference count for both mDisplayRefCount and mCacheRefCount is 0.

The bitmap is not null, and it hasn't been recycled yet.

引用計數mDisplayRefCount和mCacheRefCount都要=0

bitmap不爲null,並且它還沒有被回收

private int mCacheRefCount = 0;
private int mDisplayRefCount = 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) {
            mDisplayRefCount++;
            mHasBeenDisplayed = true;
        } else {
            mDisplayRefCount--;
        }
    }
    // 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) {
            mCacheRefCount++;
        } else {
            mCacheRefCount--;
        }
    }
    // 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 (mCacheRefCount <= 0 && mDisplayRefCount <= 0 && mHasBeenDisplayed
            && hasValidBitmap()) {
        getBitmap().recycle();
    }
}
 
private synchronized boolean hasValidBitmap() {
    Bitmap bitmap = getBitmap();
    return bitmap != null && !bitmap.isRecycled();
}

Manage Memory on Android 3.0 and Higher

在android3.0以及更高的版本中管理內存

 

Android 3.0 (API Level 11) introduces the BitmapFactory.Options.inBitmap field. 

If this option is set, decode methods that take the Options object will attempt to reuse an existing bitmap when loading content. 

This means that the bitmap's memory is reused, resulting in improved performance, and removing both memory allocation and de-allocation. 

There are some caveats in using inBitmap:

Android3.0(API 11)引入了BitmapFactory.Options.inBitmap

如果設置了這個選項,當加載內容的時候,使用Options對象的解碼方法將會嘗試複用一個存在的bitmap

這意爲着bitmap的內存被複用了,導致性能提升,並且無須分配與重新分配內存

使用inBitmap有一些需要注意的地方:

 

 

The reused bitmap must be of the same size as the source content (to make sure that the same amount of memory is used), and in JPEG or PNG format (whether as a resource or as a stream).

重用的bitmap必須與源內容尺寸一致(保證相同的內存被使用),並且是JPEG或者PNG格式(作爲resource或者stream)

 

The configuration of the reused bitmap overrides the setting of inPreferredConfig, if set.

You should always use the returned bitmap of the decode method, because you can't assume that reusing the bitmap worked (for example, if there is a size mismatch).

Save a bitmap for later use

如果設置了inPreferredConfig,重用的bitmap的配置就重寫了inPreferredConfig

你應該總數使用解碼函數返回的bitmap,因爲你不能假定重用的bitmap能工作(例如,如果尺寸不匹配)

爲之後使用保存一個bitmap

 

The following snippet demonstrates how an existing bitmap is stored for possible later use in the sample app. 

When an app is running on Android 3.0 or higher and a bitmap is evicted from the LruCache, a soft reference to the bitmap is placed in a HashSet, for possible reuse later with inBitmap:

下面的代碼片斷示範在樣本應用中,一個存在的圖片如何爲之後可能的使用而存儲

當一個應用運行在andoid3.0或者更高版本中,並且LruCache已經回收了bitmap,爲了之後使用inBitmap重用bitmap,在HashSet中存放bitmap的一個軟引用。

HashSet<SoftReference<Bitmap>> mReusableBitmaps;
private LruCache<String, BitmapDrawable> mMemoryCache;
 
// If you're running on Honeycomb or newer, create
// a HashSet of references to reusable bitmaps.
if (Utils.hasHoneycomb()) {
    mReusableBitmaps = new HashSet<SoftReference<Bitmap>>();
}
 
mMemoryCache = new LruCache<String, BitmapDrawable>(mCacheParams.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.
                mReusableBitmaps.add
                        (new SoftReference<Bitmap>(oldValue.getBitmap()));
            }
        }
    }
....
}

Use an existing bitmap

使用已經存在的bitmap

In the running app, decoder methods check to see if there is an existing bitmap they can use. 

For example:

在運行中的應用中,解碼方法檢查是否有已經存在的bitmap可供使用

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);
}

The next snippet shows the addInBitmapOptions() method that is called in the above snippet. 

It looks for an existing bitmap to set as the value for inBitmap. 

Note that this method only sets a value for inBitmap if it finds a suitable match (your code should never assume that a match will be found):

下面的代碼片斷展示上面代碼中調用的addInBitmapOptions()方法

它尋找一個存在的bitmap作爲inBitmap的值

注意,這個方法只設置inBitmap,如果找到了合適的匹配的話(你的代碼不該假設這個匹配一定會找到

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 (mReusableBitmaps != null && !mReusableBitmaps.isEmpty()) {
        final Iterator<SoftReference<Bitmap>> iterator
                = mReusableBitmaps.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;
}

Finally, this method determines whether a candidate bitmap satisfies the size criteria to be used for inBitmap:

最後,這個方法決定一個備選bitmap是否滿足設置inBitmap的尺寸標準

private static boolean canUseForInBitmap(
        Bitmap candidate, BitmapFactory.Options targetOptions) {
    int width = targetOptions.outWidth / targetOptions.inSampleSize;
    int height = targetOptions.outHeight / targetOptions.inSampleSize;
 
    // Returns true if "candidate" can be used for inBitmap re-use with
    // "targetOptions".
    return candidate.getWidth() == width && candidate.getHeight() == height;
}

 

原文地址如下,英文水平實在有限,希望拍磚同時能給予指正。

http://developer.android.com/training/displaying-bitmaps/manage-memory.html

 

 

轉貼請保留以下鏈接

本人blog地址

http://su1216.iteye.com/

http://blog.csdn.net/su1216/

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章