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中存放。
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地址
android 管理Bitmap内存 - 开发文档翻译的更多相关文章
- Android O Bitmap 内存分配
我们知道,一般认为在Android进程的内存模型中,heap分为两部分,一部分是native heap,一部分是Dalvik heap(实际上也是native heap的一部分). Andro ...
- android <application> 开发文档翻译
由于本人英文能力实在有限,不足之初敬请谅解 本博客仅仅要没有注明"转",那么均为原创.转贴请注明本博客链接链接 <application>语法: <appl ...
- Android系统Bitmap内存分配原理与优化
一.前言 笔者最近致力于vivo游戏中心稳定性维护,在分析线上异常时,发现有相当一部分是由OutOfMemory引起.谈及OOM,我们一般都会想到内存泄漏,其实,往往还有另外一个因素--图片,如果对图 ...
- android 防止bitmap 内存溢出
在android开发过程中经常会处理网络图片发送内存溢出,那么怎么解决这种问题? 思路: 下载到本地 通过网络获取和文件下载存放到手机中目录 代码: // 获取网络 public InputStrea ...
- android 高效显示Bitmap - 开发文档翻译
由于本人英文能力实在有限,不足之初敬请谅解 本博客只要没有注明“转”,那么均为原创,转贴请注明本博客链接链接 Displaying Bitmaps Efficiently 高效显示Bitmap Lea ...
- (转)Android开发:性能最佳实践-管理应用内存
翻自:http://developer.android.com/training/articles/memory.html 在任何软件开发环境中,RAM都是宝贵的资源,但在移动操作系统中更加珍贵.尽管 ...
- Android性能优化:谈话Bitmap内存管理和优化
最近除了那些忙着项目开发的事情,目前正在准备我的论文.短的时间没有写博客,今晚难得想总结.只要有一点时间.因此,为了凑合用,行.唠叨罗嗦,直接进入正题. 从事Android自移动终端的发展,想必是常常 ...
- 【转】Android中的内存管理--不错不错,避免使用枚举类型
原文网址:http://android-performance.com/android/2014/02/17/android-manage-memory.html 本文内容翻译自:http://dev ...
- Android开发文档翻译之-Services
Service是一种能长期在后台运行同一时候不须要与用户进行交互的应用组件.其它组件能够开启service,开启后service能够自行运行及时用户已经切换到其它的应用.此外,组件能够与service ...
随机推荐
- javascript之尺寸,位置,溢出
一.offsetWidth:元素的宽度,包括边框,内容,内边距. 二.offsetHeight:元素的高度,包括边框,内容,内边距. 三.offsetLeft:元素的X坐标(相对于最近已定位的祖先元素 ...
- zend studio使用入门
使用zend studio8建立项目 使用PHP开发工具zend studio8进行PHP网站开发,就需要建立(导入)相应的项目,方法如下:右键左侧Workspace,选择New | PHP Proj ...
- bzoj2018 [Usaco2009 Nov]农场技艺大赛
Description Input 第1行:10个空格分开的整数: N, a, b, c, d, e, f, g, h, M Output 第1行:满足总重量最轻,且用度之和最大的N头奶牛的总体重模M ...
- 【CF 675D Tree Construction】BST
题目链接:http://codeforces.com/problemset/problem/675/D 题意:给一个由n个互异整数组成的序列a[],模拟BST的插入过程,依次输出每插入一个元素a[i] ...
- [Spring boot] web应用返回jsp页面
同事创建了一个spring boot项目,上传到svn.需要我来写个页面.下载下来后,始终无法实现在Controller方法中配置直接返回jsp页面. 郁闷了一下午,终于搞定了问题.在此记录一下. 目 ...
- 用Unity做的一个小游戏,仿照一个样例写的,个人认为文章写的不错,哈哈
- 本地无sqlserver服务下操作数据库 之GSQL
作为程序员无论是我们写的各种MIS系统还是游戏都离不开数据的存取操作,正如我们前几天在VS下做的一MIS系统,现在纠结. 如果是C/S或B/S模型就好了,可是需求不是,没办法,顾客是上帝...他们的需 ...
- 计算机与ARM板通过路由器相连
首先,使用两根网线分别将计算机和ARM板与路由器的LAN口连接. 要想使计算机和ARM板通信,必须使二者在同一网段. 在计算机的终端输入:ifconfig 获取计算机的ip地址,查看eth0,我的ip ...
- sshd安全性能优化
sshd服务是远程登录服务,默认端口为22,对于其优化一是为了增加服务器的安全,避免暴力破解:二是为了加快速度连接,减少不必要的带宽的浪费. sshd服务的配置文件为/etc/ssh/sshd_con ...
- 更改yum网易 阿里云的yum源。
一,鉴于用国外的Yum源,速度比较慢,所以想到将国外的yum源,改为国内的Yum源,著名的有网易 阿里云源.如何更改呢? 二,更改yum源为网易的. 首先备份/etc/yum.repos.d/Cent ...