Предварительная загрузка нескольких изображений с помощью Glide
Мы пытаемся предварительно загрузить изображения в кэш-память, чтобы загрузить их позже (изображения находятся в папке Asset)
Что мы пробовали:
Glide.with(this)
.load(pictureUri)
.diskCacheStrategy(DiskCacheStrategy.ALL);
Glide.with(this)
.load(picture_uri)
.diskCacheStrategy(DiskCacheStrategy.ALL)
.preload();
Проблема: изображения кэшируются только тогда, когда мы пытаемся загрузить/отобразить их:
Они должны быть загружены в память до того, чтобы они отображались быстрее.
Glide.with(this)
.load(picture_uri)
.into(imageView);
Мы также попытались использовать модуль GlideModule для увеличения размера CacheMemory:
public class GlideModule implements com.bumptech.glide.module.GlideModule {
@Override
public void applyOptions(Context context, GlideBuilder
builder.setMemoryCache(new LruResourceCache(100000));
}
@Override
public void registerComponents(Context context, Glide glide) {
}
}
В манифесте:
<meta-data android:name=".GlideModule" android:value="GlideModule"/>
Пока ничего не работает. Любая идея?
Мы пытаемся использовать невидимый 1pp imageView, но результат тот же:
for(Drawing drawing: getDrawingsForTab(tab)){
Glide.with(this)
.load(drawing.getImage().toUri())
.dontAnimate()
.diskCacheStrategy(DiskCacheStrategy.ALL)
.into(mPreloadCacheIv);
for(Picture picture : getPictures()){
Glide.with(this)
.load(picture.getPicture().toUri())
.dontAnimate()
.diskCacheStrategy(DiskCacheStrategy.ALL)
.into(mPreloadCacheIv);
}
}
Ответы
Ответ 1
Наилучшим вариантом является самостоятельное управление кэшированием, оно дает вам больше контроля и должно быть простым, поскольку вы уже знаете, какие битовые карты должны быть загружены.
Сначала: настройте LruCache
LruCache<String, Bitmap> memCache = new LruCache<>(size) {
@Override
protected int sizeOf(String key, Bitmap image) {
return image.getByteCount()/1024;
}
};
Второе: загрузка растровых изображений в LruCache.
Display display = getWindowManager().getDefaultDisplay();
Point size = new Point();
display.getSize(size);
int width = size.x; //width of screen in pixels
int height = size.y;//height of screen in pixels
Glide.with(context)
.load(Uri.parse("file:///android_asset/imagefile"))
.asBitmap()
.fitCenter() //fits given dimensions maintaining ratio
.into(new SimpleTarget(width,height) {
// the constructor SimpleTarget() without (width, height) can also be used.
// as suggested by, An-droid in the comments
@Override
public void onResourceReady(Bitmap resource, GlideAnimation glideAnimation) {
memCache.put("imagefile", resource);
}
});
Третье: использовать кэшированные растровые изображения
Bitmap image = memCache.get("imagefile");
if (image != null) {
//Bitmap exists in cache.
imageView.setImageBitmap(image);
} else {
//Bitmap not found in cache reload it
Glide.with(context)
.load(Uri.parse("file:///android_asset/imagefile"))
.into(imageView);
}
Ответ 2
Используйте следующий код для кэширования изображений без их отображения
-
используя метод downloadOnly
, если вы хотите загрузить изображения из Интернета и сохранить их в diskCache:
FutureTarget<File> future = Glide.with(applicationContext)
.load(yourUrl)
.downloadOnly(500, 500);
-
с помощью метода preload
, если вы хотите загрузить их в кэш памяти.
Glide.with(context)
.load(url)
.preload(500, 500);
Впоследствии вы можете использовать кешированные изображения, используя
Glide.with(yourFragment)
.load(yourUrl)
.into(yourView);
Ответ 3
Glide версия 4.6.1
RequestOptions requestOptions = RequestOptions
.diskCacheStrategy(DiskCacheStrategy.ALL);
Glide.with(appContext)
.asBitmap()
.load(model)
.apply(requestOptions)
.submit();
Ответ 4
//i can give you a solution
/**
* 1.reflect some objects inside Glide
*/
fun prepareGlideObjects() {
if (!enableFunctions || atomicPrepare.getAndSet(true)) {
return
}
var t = LogTime.getLogTime()
glide = Glide.get(BasicConfig.getInstance().appContext)
if (diskCache == null) {
val engine = getObject(glide, "engine")
val diskCacheProvider = getObject(engine, "diskCacheProvider")
val method = diskCacheProvider!!::class.java.getDeclaredMethod("getDiskCache")
method.isAccessible = true
diskCache = method.invoke(diskCacheProvider) as DiskCache
}
if (arrayPool == null) {
arrayPool = getObject(glide, "arrayPool") as ArrayPool
}
if (decoder == null) {
val registry = getObject(glide, "registry") as Registry
val decoderRegistry = getObject(registry, "decoderRegistry") as ResourceDecoderRegistry
val map = getObject(decoderRegistry, "decoders") as HashMap<*, *>
val list = map["Bitmap"] as List<*>
val o = list[0]
decoder = getObject(o, "decoder") as ByteBufferBitmapDecoder
}
Log.debug(TAG, "prepareGlideObjects:" + LogTime.getElapsedMillis(t))
try {
t = LogTime.getLogTime()
//首次打开diskCache 耗时较大,此处是提前打开文件索引
val url = GlideUrl("http://xx.cdn.yy.com/fake_pic.jpg")
val dataCacheKey = DataCacheKey(url, EmptySignature.obtain())
diskCache?.get(dataCacheKey)
Log.debug(TAG, "_load_fake_pic:" + LogTime.getElapsedMillis(t))
} catch (e: Throwable) {
Log.error(TAG, "cold load failed:$e")
}
}
/**
*2.load bitmap-file from diskCache
*/
fun loadBitmap(url: String) {
val gUrl = GlideUrl(url:String)
val dataCacheKey = DataCacheKey(gUrl, EmptySignature.obtain())
val file = diskCache?.get(dataCacheKey)
}
//3.decode bitmap from file
private fun extractBitmapFromFile(url: String, file: File) {
try {
val dimen = getDimensionFromUrl(url)
val result = decoder?.decode(ByteBufferUtil.fromFile(file), dimen[0], dimen[1],
Options()) as BitmapResource?
result!!.initialize()
//simple bitmap cache
bitmapMap[url] = result.get()
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
Log.debug(TAG, "allocationByteCount:" +
bitmapMap[url]!!.allocationByteCount / 1024.0)
}
} catch (e: Throwable) {
Log.error(TAG, "init result failed:$e")
}
}
//4.decode file from http-stream
private fun decodeHttpStream() {
val uri = GlideUrl(call?.request()?.url().toString())
val contentLength = Preconditions.checkNotNull(responseBody).contentLength()
val stream = ContentLengthInputStream.obtain(responseBody.byteStream(),
contentLength)
if (arrayPool == null) {
//re prepare
arrayPool = glide?.arrayPool
}
val encoder = StreamEncoder(arrayPool)
val writer = DataCacheWriter(encoder, stream, Options())
val originalKey = DataCacheKey(uri, EmptySignature.obtain())
//ready,此处可以用来监控文件字节流写入本地文件的时间
diskCache?.put(originalKey, writer)
val file = diskCache?.get(uri)
}
//5.после выполнения этих работ загрузите растровое изображение из diskCache, это займет всего 10 мс。