Glide最简单的使用应该就是下边的代码:
Glide.with(Context).load("url").into(ImageView);
开发者使用起来非常简单,但里面大有文章。
之前看了Retrofit、Volley、OkHttp库源码都感觉自己看下来没有太大问题。难度不算大。但Glide的源码如果你有能力不需要其他资源能看下来的确能力很了得。因为Glide的源码复杂程度绝对在上面说的几个框架之上。
在根据上边范例代码看源码过程中我画了个UML图。基本上都不够画,只能在into部分以最简单的方式呈现出来。上图:
data:image/s3,"s3://crabby-images/733e4/733e4d55267616291a310bf0615daad8bc0dd702" alt=""
大致说一下这个图的三大部分就是:
- .with(Context)
- .load("url")
- .into(ImageView)
.with(Context)
data:image/s3,"s3://crabby-images/9b86c/9b86c3f8122894ccf9b581ce6010246aa758cb4c" alt=""
我们开始讲解with部分:
data:image/s3,"s3://crabby-images/f5767/f5767066e07c4d070ac1747e7296963deb969952" alt=""
Glide的with有5个重载方法。其实都可以归类为当在非主线程调用with(),通通走with(Context);而在主线程调用就看调用哪个一般两步走:
data:image/s3,"s3://crabby-images/0938e/0938eaa7e49b8983667fc1c30eb2cb436e9b28a7" alt=""
哪两步:就是红色长方形框部分get(FragmentActivity)或get(Activity);
而红色圈圈其实最终走getApplicationManager(context)
getApplicationManager(context)
该方法属于工作线程调用with或with里面的参数是Application的Context
data:image/s3,"s3://crabby-images/bf81c/bf81c98bc87e618b57d110efe2107040d052f6fb" alt=""
先讲这个方法是因为它比较简单就是创建一个ResquestManager,传进两个参数:ApplicationLifecycle和EmptyRequestManagerTreeNode
- ApplicationLifecycle 其实看名字大概就知道它是监听Application的
- EmptyRequestManagerTreeNode这个因为是Application所以其实它是什么也没有做。
data:image/s3,"s3://crabby-images/f5cee/f5ceec080d1bb2db815860638135a7fedce1b4b6" alt=""
get((Activity) context);
data:image/s3,"s3://crabby-images/5cd29/5cd2913f21991bb054a8dc6324cf74a55001f09c" alt=""
data:image/s3,"s3://crabby-images/61512/615129c1d682d8bf86b806587661303fcf4325d5" alt=""
只看截图应该知道大概干了什么事情那吧:
- 创建RequestManagerFragment
- 在该fragment获取lifecycle,和RequestManagerTreeNode
其实get(FragmentActivity activity)也是大致做这样的事情:
data:image/s3,"s3://crabby-images/e1a63/e1a63d831ecaf95ecf669e9b2ee676cb70958992" alt=""
为什么要分开做主要是因为在android中有两种fragment,一种是v4和另外一种是非v4包的。
Lifecycle和RequestManagerTreeNode分别干什么
- ActivityFragmentLifecycle 实现 Lifecycle接口
主要就是管理所有实现Lifecycle的接口。
data:image/s3,"s3://crabby-images/28d36/28d36365e1874f77ee76d8056a2b848c421b77e7" alt=""
将所有Lifecycle的接口放进WeakHashMap上,然后分不同情况调用接口的onStart、onStop、onDestroy
- RequestManagerTreeNode
RequestManagerTreeNode有两个实现它接口的类:SupportFragmentRequestManagerTreeNode和FragmentRequestManagerTreeNode
其实作用都是一样就是收集它们层级的RequestManager。因为我们在每个Activity或Fragment调用Glide.with(this);就会创建一个RequestManager。而RequestManagerTreeNode的作用就是获取RequestManager的集合Set。
data:image/s3,"s3://crabby-images/a1350/a1350e9eca50e4042bb0db6fdb98b054e091e061" alt=""
RequestManager的创建过程:
data:image/s3,"s3://crabby-images/14c0a/14c0a6ae8854135920cae2e478214d96f127fa72" alt=""
data:image/s3,"s3://crabby-images/76ca9/76ca92fd8d385e95c161784f8ca58c60c060a03f" alt=""
添加ActivityFragmentLifecycle、RequestManagerTreeNode、RequestTracker、ConnectivityMonitorFactory;
如果在主线程直接把RequestManager放进ActivityFragmentLifecycle,非主线程通过Handler post进去。为了就是和生命周期同步。
这里有两个类:RequestTracker和ConnectivityMonitorFactory
- RequestTracker其实就是有点和ActivityFragmentLifecycle的功能类似就是将Request收集起来,进行统一管理。
data:image/s3,"s3://crabby-images/c40b6/c40b67d9f25b101afe0f16195b2b80cb1b826be7" alt=""
data:image/s3,"s3://crabby-images/f6d70/f6d70fca8c4a790c0634cf581829d781f65b0d9e" alt=""
data:image/s3,"s3://crabby-images/32729/327291e65e0591469dcc1216e564a7adb5647f36" alt=""
- ConnectivityMonitorFactory 的主要作用就是监听网络情况。而监听网络主要通过广播。
ConnectivityMonitorFactory会根据Manifest文件中是否有添加userPermission对网络进行监听;
data:image/s3,"s3://crabby-images/53421/53421783762f83c7e11a336141d5e0ea8f137cd1" alt=""
data:image/s3,"s3://crabby-images/1209c/1209c2f477bd32b64ab441ac6265f9e204eb0dc4" alt=""
data:image/s3,"s3://crabby-images/d95b8/d95b8eacd303abdd724d5cffa61b4a20e37e7f23" alt=""
整个with阶段Glide主要就是做了这些东西:
- 创建一个空的Fragment对生命周期进行监听,在onStart执行request.onStart();
- 通过TreeNode对每个Fragment都进行管理;
- 通过RequestTracker对每个Request进行管理;
- 网络监听,通过广播对网络进行监听;
- Glide对象的创建,会在后面交待清楚;
缓存方面的操作在本文中不打算分析,以后有机会在学习。
下面看看load(url)部分
.load(url)
data:image/s3,"s3://crabby-images/ceae8/ceae806aa82898ce236264e7f9f95d50265592b9" alt=""
RequestManager的load()方法:
data:image/s3,"s3://crabby-images/40885/4088552f274075259337cb838505da6cffded9a0" alt=""
我们就看load(string)
data:image/s3,"s3://crabby-images/fd45b/fd45bd41497a973532aa2d516872134c3f4ad7fc" alt=""
返回一个DrawableTypeRequest<String>
- fromString()
data:image/s3,"s3://crabby-images/0ccbe/0ccbe128639eaa0bd35579e6529d3cdec8b31b11" alt=""
data:image/s3,"s3://crabby-images/65304/65304a8dee476ec9c785a7bef71ff0373c39ce4c" alt=""
可以看到最终是new一个DrawableTypeRequest<T>
这里开始有点复杂了,开始分析源码的时候,我就是卡在这里。后面大概看了两篇博文,解决了我一点疑惑。最后我会贴上两篇博文的链接出来。
这里留意一下泛型,后面会出现四个泛型的对象,到时更晕
data:image/s3,"s3://crabby-images/44af1/44af1b1f25415a5dab82ab896a5a20ed566a8bc9" alt=""
这里的泛型T为String类型。当然主要看它传入什么类型。现在本博文以String为例;
这里有个ModelLoader对象,后面代码会经常出现。
现在返回来看看Glide对象的构建:
data:image/s3,"s3://crabby-images/42e09/42e09e5ab01dce328ff8edb9dbca12da6dac5831" alt=""
data:image/s3,"s3://crabby-images/abb16/abb16d38894baece68ffd6a9360647dc6530507d" alt=""
data:image/s3,"s3://crabby-images/43983/43983a3c8b56c9059a33ff8f250cab3a655970eb" alt=""
data:image/s3,"s3://crabby-images/d286f/d286f3a85510309ca6f4d44b1faae23be5512275" alt=""
代码有点多,前面截图的第一张就不说了,就是准备一些缓存的和BitmapPool,这里Glide创建了一堆loader、transcoder等东西,其实就是为load()方法做铺垫的。Glide框架这么强大,就源于这堆loader、transcoder;当然还有其他的功能;
注册了这么多loader或transcoder怎么管理呢?
主要通过GenericLoaderFactory和ModelLoader
data:image/s3,"s3://crabby-images/5928d/5928d78e4be082a529bae0a589497f406b53c7df" alt=""
data:image/s3,"s3://crabby-images/741ae/741ae395d22e71743932a1f6660ebddf774c5d90" alt=""
这里截了一小部分,有兴趣的研究可以看看源码。其实主要就是通过
ModelLoader里面的两个泛型做key值获取到对应的ModelLoader具体对象。
- 现在看回去刚才的loadGeneric方法:
ModelLoader<T, InputStream> streamModelLoader = Glide.buildStreamModelLoader(modelClass, context);
data:image/s3,"s3://crabby-images/10f97/10f97b4ab82c3acdbd409ae0ae4de26ac6125a7e" alt=""
一路跟下去最终就是调用刚才说的GenericLoaderFactory的buildModelLoader。这里我们可以根据泛型<T><Y>知道对应的ModelLoader:<T>String <Y>InputStream
data:image/s3,"s3://crabby-images/7a6dc/7a6dca191619aaafc40750274992cf122a20d3d1" alt=""
这里需要一直跟下去,其实你就会发现最终是
HttpUrlGlideUrlLoader 非截图的StreamStringLoader;
相同的方法就会发现
ModelLoader<T, ParcelFileDescriptor> fileDescriptorModelLoader =
Glide.buildFileDescriptorModelLoader(modelClass, context);
fileDescriptorModelLoader 是 FileDescriptorUriLoader
最后是new DrawableTypeRequest对象
optionsApplier.apply其实也是返回DrawableTypeRequest,所以看DrawableTypeRequest对象构建:
data:image/s3,"s3://crabby-images/603a9/603a9dc2fc67088033fa267818f6f6950384d097" alt=""
这里传了一大堆参数其实,很多参数是我们之前讲过的,这里就不介绍了。这里主要看buildProvider方法:
它是返回一个FixedLoadProvider,
这里buildProvider主要看:下面部分
data:image/s3,"s3://crabby-images/8c94e/8c94ec76c7ab261e992a3095fc759c79ca63d023" alt=""
transcoder为 GlideBitmapDrawableTranscoder
modelLoader为 ImageVideoModelLoader 通过 HttpUrlGlideUrlLoader和 FileDescriptorUriLoader转换出来的。
dataLoadProvider为ImageVideoGifDrawableLoadProvider
这里记录一下DrawableTypeRequest各个Loader、Provider待会会用得到:
- LoadProvider:FixedLoadProvider
- streamModelLoader :HttpUrlGlideUrlLoader
- fileDescriptorModelLoader:FileDescriptorUriLoader
- ResourceTranscoder : GlideBitmapDrawableTranscoder
- DataLoadProvider : ImageVideoGifDrawableLoadProvider
- modelLoader : ImageVideoModelLoader
基本上load()方法就是根据ModelLoader和传入<T>在Glide注册的Loader,Provider选取各种需要的Loader和Provider。
into(ImageView)
data:image/s3,"s3://crabby-images/df751/df751e4cb44254e5d1f38b9c3ffebe5d7d382b40" alt=""
- DrawableTypeRequest.into(ImageView)
DrawableTypeRequest并没有into方法,它的父类DrawableRequestBuilder有:
data:image/s3,"s3://crabby-images/9412a/9412a3e964bb9f4ef218c4c3dec456271e7b39bb" alt=""
继续看父类GenericRequestBuilder的into方法:
data:image/s3,"s3://crabby-images/42908/42908cbde9f894b4fa43dd189b6fb06807699c66" alt=""
switch语句是设置ImageVIew的ScaleType,看最后一句:
into(glide.buildImageViewTarget(view, transcodeClass));
返回一句Target,Target是一个接口;
先看看glide.buildImageViewTarget(view,transcodeClass);这句代码;
这里的transcodeClass是 GlideDrawable.class
data:image/s3,"s3://crabby-images/5a3f8/5a3f8dbe99c91f19708260894c119cf68c4374fd" alt=""
glide.buildImageViewTarget(view, transcodeClass)
data:image/s3,"s3://crabby-images/7a043/7a0434232790f30c75dc9b0dceb77cf44a9565d9" alt=""
代码一直跟落去会看到上面截图;
返回Target<Z>;
这里其实你一直跟落去GlideDrawableImageViewTarget会发现是通过view获取宽高等信息。关键是看泛型<Z>是什么东西:
其实不难发现泛型Z就是:GlideDrawable.class
这里不多作解释;
即返回的Target为:GlideDrawableImageViewTarget
那么into最终其实就是into(GlideDrawableImageViewTarget);
data:image/s3,"s3://crabby-images/cfe02/cfe021db2e006189677fa8ffb50be5e6b3c0b297" alt=""
data:image/s3,"s3://crabby-images/ba455/ba455a2490f8fffea89f6821f6683ba08d0a2d29" alt=""
关键看红色框框的调用,前面是一些非空判断和清空资源重用资源;
Request request = buildRequest(target);
data:image/s3,"s3://crabby-images/c2d41/c2d4177fd6064486ab9ff2b7c2cb3185e34fd2f6" alt=""
data:image/s3,"s3://crabby-images/f1848/f18482c9e6e5722c4d828f6a5eb08104e7adc306" alt=""
第二个截图是GenericRequestBuilder.buildRequestRecursive()方法调用因为前面 代码是对缩略图的处理可以忽略;
看最后一句:
return obtainRequest(target, sizeMultiplier, priority, parentCoordinator);
private Request obtainRequest(Target<TranscodeType> target, float sizeMultiplier, Priority priority,
RequestCoordinator requestCoordinator) {
return GenericRequest.obtain(
loadProvider,
model,
signature,
context,
priority,
target,
sizeMultiplier,
placeholderDrawable,
placeholderId,
errorPlaceholder,
errorId,
fallbackDrawable,
fallbackResource,
requestListener,
requestCoordinator,
glide.getEngine(),
transformation,
transcodeClass,
isCacheable,
animationFactory,
overrideWidth,
overrideHeight,
diskCacheStrategy);
}
其实最终就是构建一个Request
data:image/s3,"s3://crabby-images/57fad/57fad1bd1a80c0b81384819664150cac894ddb3c" alt=""
Request是什么其实就是加载图片的请求;前面的RequestManager就是负责管理Request的;
太多参数只能截部分图;
你会看到四个泛型分别是<A,T,Z,R>
data:image/s3,"s3://crabby-images/4ee2a/4ee2a68bd93525b60eef60e39af2b6cf9c003c36" alt=""
从我们的例子里,
- A String
- T ImageVideoWrapper
- Z GifBitmapWrapper
- R GlideDrawable
继续往下看
data:image/s3,"s3://crabby-images/5ede3/5ede3fb3a0ff05bdd30d59330ff2db03b9e42aa0" alt=""
Request创建完成后就用target set Request;
然后设置监听器
最后用RequestTracker运行Request;
data:image/s3,"s3://crabby-images/6f1bb/6f1bb68b3333f836b9266f6d8a8d544aac14ddeb" alt=""
看Request.begin()
上面已经说了真正实现Request的是GenericRequest
所以看GenericRequest的begin()
data:image/s3,"s3://crabby-images/6c5d0/6c5d044ba9ff6ef0ee950c1b2b984cf523a1cf28" alt=""
主要看
onSizeReady(overrideWidth, overrideHeight);
所有谜底都在这里
- 包括线程;
- 网络执行
- 回调
data:image/s3,"s3://crabby-images/e44c8/e44c8b5aa442f2e7cb5a09e5a2fc86bab9015575" alt=""
data:image/s3,"s3://crabby-images/3a110/3a110f525643e229014b074020391afc9301423e" alt=""
前面的取宽高其实是缩略图的宽高;
主要看下面的代码:
ModelLoader<A, T> modelLoader = loadProvider.getModelLoader();
final DataFetcher<T> dataFetcher = modelLoader.getResourceFetcher(model, width, height);
这里的modelLoader 即 ImageVideoModelLoader
dataFetcher 即 ImageVideoFetcher
data:image/s3,"s3://crabby-images/3b6a5/3b6a508a511b32aabab2dc3fa994b23c574dc0a9" alt=""
这个截图是上面load(url)阶段时候记录的。现在用得上啦吧。说Glide源码复杂就在这里,太多这些loader,provider的东西了。
继续....
ResourceTranscoder<Z, R> transcoder = loadProvider.getTranscoder();
transcoder 这里的是 GlideBitmapDrawableTranscoder;上面也提到过;
最后阶段了
loadStatus = engine.load(signature, width, height, dataFetcher, loadProvider, transformation, transcoder,
priority, isMemoryCacheable, diskCacheStrategy, this);
public <T, Z, R> LoadStatus load(Key signature, int width, int height, DataFetcher<T> fetcher,
DataLoadProvider<T, Z> loadProvider, Transformation<Z> transformation, ResourceTranscoder<Z, R> transcoder,
Priority priority, boolean isMemoryCacheable, DiskCacheStrategy diskCacheStrategy, ResourceCallback cb) {
Util.assertMainThread();
long startTime = LogTime.getLogTime();
//这里的id其实就是url地址
final String id = fetcher.getId();
EngineKey key = keyFactory.buildKey(id, signature, width, height, loadProvider.getCacheDecoder(),
loadProvider.getSourceDecoder(), transformation, loadProvider.getEncoder(),
transcoder, loadProvider.getSourceEncoder());
EngineResource<?> cached = loadFromCache(key, isMemoryCacheable);
if (cached != null) {
cb.onResourceReady(cached);
if (Log.isLoggable(TAG, Log.VERBOSE)) {
logWithTimeAndKey("Loaded resource from cache", startTime, key);
}
return null;
}
EngineResource<?> active = loadFromActiveResources(key, isMemoryCacheable);
if (active != null) {
//为下次加载做准备,
cb.onResourceReady(active);
if (Log.isLoggable(TAG, Log.VERBOSE)) {
logWithTimeAndKey("Loaded resource from active resources", startTime, key);
}
return null;
}
EngineJob current = jobs.get(key);
if (current != null) {
current.addCallback(cb);
if (Log.isLoggable(TAG, Log.VERBOSE)) {
logWithTimeAndKey("Added to existing load", startTime, key);
}
return new LoadStatus(cb, current);
}
//工作线程
EngineJob engineJob = engineJobFactory.build(key, isMemoryCacheable);
DecodeJob<T, Z, R> decodeJob = new DecodeJob<T, Z, R>(key, width, height, fetcher, loadProvider, transformation,
transcoder, diskCacheProvider, diskCacheStrategy, priority);
EngineRunnable runnable = new EngineRunnable(engineJob, decodeJob, priority);
jobs.put(key, engineJob);
engineJob.addCallback(cb);
engineJob.start(runnable);
if (Log.isLoggable(TAG, Log.VERBOSE)) {
logWithTimeAndKey("Started new load", startTime, key);
}
return new LoadStatus(cb, engineJob);
}
大Boss来了,这里就是核心部分,什么网络加载图片,编码解码的工作都在这里完成。
由于这博文不说缓存部分所以很多代码可以忽略;
主要看:
data:image/s3,"s3://crabby-images/f46ab/f46abc38a24a4b0ffe41453d529f2894e1962478" alt=""
其实看名字都大概知道干什么的;
创建EnginJob主要处理Runnable和回调
DecodeJob就是解码
最终new LoadStatus就是更新Request的状态
这里主要看看Runnable的代码:
data:image/s3,"s3://crabby-images/f323c/f323cd438cc5172b65c905b03c715206d3c37ce4" alt=""
上面截图是EnginRunnable的run方法
data:image/s3,"s3://crabby-images/aa450/aa450796237cc52330a94c363a4f20d33bfc287f" alt=""
data:image/s3,"s3://crabby-images/59b02/59b02692bd393f8d4a0ab1ea48bbf8995951c605" alt=""
data:image/s3,"s3://crabby-images/ffc0d/ffc0d6706df915c6ee65d8b4d2a3641415e2ba00" alt=""
data:image/s3,"s3://crabby-images/0e47c/0e47c147582f1f756a65379fe13010a69cbf0421" alt=""
这里的fetcher即DataFetcher,实现DataFetcher是 ImageVideoModelLoader,而ImageVideoModelLoader的loadData();
data:image/s3,"s3://crabby-images/5bb2a/5bb2ac6ddabde83d79d49b7bfd9a63ba7a55e8a4" alt=""
data:image/s3,"s3://crabby-images/4f6ac/4f6ac60ce3fa5c282f14eaabca6f3c3c84b67048" alt=""
这里的streamFetcher 就是HttpUrlFetcher
private InputStream loadDataWithRedirects(URL url, int redirects, URL lastUrl, Map<String, String> headers)
throws IOException {
if (redirects >= MAXIMUM_REDIRECTS) {
throw new IOException("Too many (> " + MAXIMUM_REDIRECTS + ") redirects!");
} else {
// Comparing the URLs using .equals performs additional network I/O and is generally broken.
// See http://michaelscharf.blogspot.com/2006/11/javaneturlequals-and-hashcode-make.html.
try {
if (lastUrl != null && url.toURI().equals(lastUrl.toURI())) {
throw new IOException("In re-direct loop");
}
} catch (URISyntaxException e) {
// Do nothing, this is best effort.
}
}
urlConnection = connectionFactory.build(url);
for (Map.Entry<String, String> headerEntry : headers.entrySet()) {
urlConnection.addRequestProperty(headerEntry.getKey(), headerEntry.getValue());
}
urlConnection.setConnectTimeout(2500);
urlConnection.setReadTimeout(2500);
urlConnection.setUseCaches(false);
urlConnection.setDoInput(true);
// Connect explicitly to avoid errors in decoders if connection fails.
urlConnection.connect();
if (isCancelled) {
return null;
}
final int statusCode = urlConnection.getResponseCode();
if (statusCode / 100 == 2) {
return getStreamForSuccessfulRequest(urlConnection);
} else if (statusCode / 100 == 3) {
String redirectUrlString = urlConnection.getHeaderField("Location");
if (TextUtils.isEmpty(redirectUrlString)) {
throw new IOException("Received empty or null redirect url");
}
URL redirectUrl = new URL(url, redirectUrlString);
return loadDataWithRedirects(redirectUrl, redirects + 1, url, headers);
} else {
if (statusCode == -1) {
throw new IOException("Unable to retrieve response code from HttpUrlConnection.");
}
throw new IOException("Request failed " + statusCode + ": " + urlConnection.getResponseMessage());
}
}
加载图片资源由HttpUrlFetcher的loadData完成。
最终返回 ImageVideoFetcher<ImageVideoWrapper>
往下看:
data:image/s3,"s3://crabby-images/969c9/969c98bf8dacb0b396d6a300fecfc39a1829d0d8" alt=""
解码部分:loadProvider即ImageVideoGifDrawableLoadProvider
不明白怎么找可以看前面解释;
最终你可以跟到是GifBitmapWrapperResourceDecoder的decode方法:
data:image/s3,"s3://crabby-images/d4368/d43685615f25855fdd98fffa509231545f5db760" alt=""
data:image/s3,"s3://crabby-images/324f8/324f84b66b42984156c7c44ebf61ba19c3c7f20a" alt=""
data:image/s3,"s3://crabby-images/251cf/251cf0512e8bf186bcdf0d41608b2ce3be3de9f7" alt=""
GifBitmapWrapperResourceDecoder这个类就是解释图片的。
data:image/s3,"s3://crabby-images/6538b/6538b4d984a6485553576efbbffd81ce4d8ecdce" alt=""
这里看注解,大概意思是通过输入流的开始部分的2048大小就可以判断这张图是gif还是其他格式的。
如果不是gif就通过ImageVideoDataLoadProvider继续完成解码工作;至于为什么是ImageVideoDataLoadProvider你可以通过之前找provider或dataloader的方法找。
data:image/s3,"s3://crabby-images/0cefc/0cefc5b2db1144c481a0472f201adb898546016d" alt=""
真正解码是通过该类完成。
data:image/s3,"s3://crabby-images/f406b/f406b519ea3de55f47541ce902259ef3e6832ef7" alt=""
这里getStream就是从网络获取到的InputSteam。
最终你可以找到StreamBitmapDecoder的decode()
data:image/s3,"s3://crabby-images/dcaf3/dcaf32ff6263321b3aaf524c7960bfa4d0c26e78" alt=""
downsampler.decode()代码有点多就贴出来了:
@Override
public Bitmap decode(InputStream is, BitmapPool pool, int outWidth, int outHeight, DecodeFormat decodeFormat) {
final ByteArrayPool byteArrayPool = ByteArrayPool.get();
final byte[] bytesForOptions = byteArrayPool.getBytes();
final byte[] bytesForStream = byteArrayPool.getBytes();
final BitmapFactory.Options options = getDefaultOptions();
// Use to fix the mark limit to avoid allocating buffers that fit entire images.
RecyclableBufferedInputStream bufferedStream = new RecyclableBufferedInputStream(
is, bytesForStream);
// Use to retrieve exceptions thrown while reading.
// TODO(#126): when the framework no longer returns partially decoded Bitmaps or provides a way to determine
// if a Bitmap is partially decoded, consider removing.
ExceptionCatchingInputStream exceptionStream =
ExceptionCatchingInputStream.obtain(bufferedStream);
// Use to read data.
// Ensures that we can always reset after reading an image header so that we can still attempt to decode the
// full image even when the header decode fails and/or overflows our read buffer. See #283.
MarkEnforcingInputStream invalidatingStream = new MarkEnforcingInputStream(exceptionStream);
try {
exceptionStream.mark(MARK_POSITION);
int orientation = 0;
try {
orientation = new ImageHeaderParser(exceptionStream).getOrientation();
} catch (IOException e) {
if (Log.isLoggable(TAG, Log.WARN)) {
Log.w(TAG, "Cannot determine the image orientation from header", e);
}
} finally {
try {
exceptionStream.reset();
} catch (IOException e) {
if (Log.isLoggable(TAG, Log.WARN)) {
Log.w(TAG, "Cannot reset the input stream", e);
}
}
}
options.inTempStorage = bytesForOptions;
final int[] inDimens = getDimensions(invalidatingStream, bufferedStream, options);
final int inWidth = inDimens[0];
final int inHeight = inDimens[1];
final int degreesToRotate = TransformationUtils.getExifOrientationDegrees(orientation);
final int sampleSize = getRoundedSampleSize(degreesToRotate, inWidth, inHeight, outWidth, outHeight);
final Bitmap downsampled =
downsampleWithSize(invalidatingStream, bufferedStream, options, pool, inWidth, inHeight, sampleSize,
decodeFormat);
// BitmapFactory swallows exceptions during decodes and in some cases when inBitmap is non null, may catch
// and log a stack trace but still return a non null bitmap. To avoid displaying partially decoded bitmaps,
// we catch exceptions reading from the stream in our ExceptionCatchingInputStream and throw them here.
final Exception streamException = exceptionStream.getException();
if (streamException != null) {
throw new RuntimeException(streamException);
}
Bitmap rotated = null;
if (downsampled != null) {
rotated = TransformationUtils.rotateImageExif(downsampled, pool, orientation);
if (!downsampled.equals(rotated) && !pool.put(downsampled)) {
downsampled.recycle();
}
}
return rotated;
} finally {
byteArrayPool.releaseBytes(bytesForOptions);
byteArrayPool.releaseBytes(bytesForStream);
exceptionStream.release();
releaseOptions(options);
}
}
是Downsampler的decode方法
最终如果是非gif格式的图片返回:GifBitmapWrapper<Bitmap>
最终获取到非gif格式的bitmap放入Imageview上。
由于看Glide源码之前我是看了Volley、Retrofit、OkHttp这三个的源码感觉还好,所以就直接在没有看其他资源的基础上看Glide的源码。但看到with阶段就有困难,卡在Glide是通过什么监听Activity的生命周期,然后看了看郭霖的介绍Glide的第二篇开始部分 Android图片加载框架最全解析(二),从源码的角度理解Glide的执行流程知道是通过构建空的fragment实现的。因为是第一次接触通过这样的方法实现监听Activity生命周期,所以我觉得这个是Glide的一大优点。值得我们学习。
看到load(url)阶段卡住,因为当时没有发现Glide对象的创建注册了很多provider、loader,decoder之类。主要是看源码不够细心。如是看了此篇文章Glide加载图片流程(Part One),这里其实主要看表格部分,它给了我启发。之后,我细致地看Glide对象构建就知道是通过一个接口加对象管理一大堆provider、loader,decoder。
最后想说看源码一定要:
- 细心
- 像郭霖大神所说要有目的地看源码,不然会走进了森林,走不出来。
- 耐心: 人家郭大神看了一个月Glide的源码,所以你懂得~~~哈哈
题外话一下AOSP带给我们很多值得学习的地方如:
- Volley将请求分为缓存队列和网络队列管理;
- Retrofit通过注解、集合对Factory的整理、建造者模式、动态代理实现高度解耦
- OkHttp库的递归迭代实现各个拦截的功能独立,连续调用;静态代码块实现对旧版本代码的调用
网友评论