第一篇 Glide图片和默认图替换过程中,默认图被拉伸了一下")第一篇 Glide图片和默认图替换过程中,默认图被拉伸了一下
现象描述:
今天遇到一个比较坑的问题,用Glide加载图片被拉伸了,找了半天才发现跟Glide设置占位图有关,如果占位图要比原图大,
图片就直接被拉伸了.以下为出问题的代码:
Glide.with(mContext)
.load(imgUrl)
.placeholder(R.drawable.img_defualt)
.into(iv_show);
解决方法:
这是替换过程中一个动画阶段,Glide提供一个方法,果断
Glide.with(mContext)
.load(imgUrl)
.placeholder(R.drawable.img_defualt)
.dontAnimate()
.into(iv_show);
第二篇 Glide加载图片偏绿或者质量偏低")第二篇 Glide加载图片偏绿或者质量偏低
现象描述:
用Glide加载图片感觉便绿色,普通出现这种问题一般都是压缩造成的.
解决方法:
1.缓存原图 2.修改bitmap的编码(Glide默认编码为RGB_565 替换成ARGB_8888),这样一般能解决大部分问题,
(不过这样一来,Glide的加载速度自然就下降了,毕竟增加一倍的解码内存,不是有特殊需求,不要随便使用),以下为示例代码:
//1.缓存原图
Glide.with(mContext)
.load(imgUrl)
.diskCacheStrategy(DiskCacheStrategy.SOURCE)
.into(iv_show);
//修改Bitmap的编码为ARGB_8888
public class GlideConfig implements GlideModule {
@Override
public void applyOptions(Context context, GlideBuilder builder) {
builder.setDecodeFormat(DecodeFormat.PREFER_ARGB_8888);
}
@Override
public void registerComponents(Context context, Glide glide) {
}
}
//在manifest.xml配置上以上信息
<meta-data
android:name="com.xxx.xxx.GlideConfig"
android:value="GlideModule"/>
第三篇 Glide混淆脚本没加导致的Crash
现象描述:
在加了第二篇提到的一些配置,其中GlideModule是需要在manifest中指定的,debug版本一直没事,realease版本各种Crash,报错信息:
java.lang.IllegalArgumentException: Unable to find GlideModule to find GlideModule implementation...
一开始我就想到了混淆脚本,结果遇到了两个坑:
- 网上提供的混淆脚本包名是Glide之前的包名
- Glide内部混淆没关系,但是实现了GlideModule接口的类不能混淆,因为manifest中明确的指明了包名+类名的,混淆了自然就找不到了
解决方法:
1.加上必要的混淆脚本:
-keep public class * implements com.bumptech.glide.module.GlideModule
-keep public enum com.bumptech.glide.load.resource.bitmap.ImageHeaderParser$** {
**[] $VALUES;
public *;
}
-keep class com.bumptech.** {
*;
}
第四篇 Glide无法给Gone的ImageView设置图片
现象描述:
因为有一个需求,其实就是广告,位置是个ImageView,没有拉到广告之前,我设置的是Gone,拉到图片之后,设置上图片,然后设置为可见,结果广告拉到了,但是对应位置没有任何反应,图片地址是可以正确打开的,以下为示例代码:
RequestListener callback = new RequestListener() {
@Override
public boolean onException(Exception e, Object model, Target target, boolean isFirstResource) {
return false;
}
@Override
public boolean onResourceReady(Object resource, Object model, Target target, boolean isFromMemoryCache, boolean isFirstResource) {
iv.setVisibility(View.VISIBLE);
return false;
}
};
iv.setVisibility(View.GONE);//如果你的ImageViewVisibility为Gone,以上回调没有作用,改成INVISIBLE即可
Glide.with(mContext).load(url)
.listener(callback)
.into(iv);
解决方法:
经过我测试,首先两个回调根本没有执行,网上搜到了一个国外程序员的回答
What is the size of the ImageView declared in XML? Glide needs to know how big an image the target wants. And does so using a layout listener if the size is not ready yet. Having a hidden ImageView doesn’t go through a layout to save processor cycles for obvious reasons, so it may not have a size yet.
You can try .override(w, h) to specify the size yourself or use explicit sizing (100dp).
他指出Glide需要知道我们控件的大小,加上override(w,h)去指定控件的大小,在我这里我直接将Visiable设置为可见就OK了
第五篇 Glide为通知栏(Notification设置图片不生效)
现象描述:就是很正常Glide加载bitmap,设置给Remoteview结果不生效(bug代码)
//********错误代码************
SimpleTarget<Bitmap> target = new SimpleTarget<Bitmap>() {
@Override
public void onResourceReady(Bitmap resource, GlideAnimation<? super Bitmap> glideAnimation) {
rv.setImageViewBitmap(R.id.iv_icon, resource);
}
};
Glide.with(mContext).load(icon).asBitmap().into(target);
解决方法:
以下为正确的使用方式
// 1.上下文 2.RemoteView 3.图标控件的ResID ,4 Notification 5 Notification_ID
NotificationTarget notificationTarget = new NotificationTarget(mContext, rv, R.id.iv_icon, notify, NOTIFY_ID);
Glide.with(mContext).load(icon).asBitmap().placeholder(R.mipmap.ic_launcher).error(R.mipmap.ic_launcher).listener(new RequestListener<String, Bitmap>() {
@Override
public boolean onException(Exception e, String model, Target<Bitmap> target, boolean isFirstResource) {
return true;
}
@Override
public boolean onResourceReady(Bitmap resource, String model, Target<Bitmap> target, boolean isFromMemoryCache, boolean isFirstResource) {
return false;
}
}).into(notificationTarget);
第六篇 Glide清除缓存
双清方法:
Glide.get(this).clearDiskCache();//请在非UI线程中调用
Glide.get(this).clearMemory();//请在主线程中调用
第七篇 Glide加载加密的数据
Glide的Load方法是可以接受byte[] 我们当然可以每次解图之前将输入流转换成byte[]进行加载但是,就没有办法很好的使用到Glide内置的缓存策略了
1.自定义GlideModule (注意混淆)
public class CipherGlideModule implements GlideModule {
@Override
public void applyOptions(Context context, GlideBuilder builder) {
}
@Override
public void registerComponents(Context context, Glide glide) {
/**
* String.class 输入源
* InputStream 输出
*/
glide.register(String.class, InputStream.class, new ImageLoader.Factory());
}
}
//mainfest.xml中声明 :
<meta-data
android:name="xxx.xxx.CipherGlideModule"
android:value="GlideModule" />
2.自定义ImageLoader
public class ImageLoader implements ModelLoader<String, InputStream> {
public ImageLoader() {
}
@Override
public DataFetcher<InputStream> getResourceFetcher(String model, int width, int height) {
return new ImageDataFetcher(model);
}
public static class Factory implements ModelLoaderFactory<String, InputStream> {
@Override
public ModelLoader<String, InputStream> build(Context context, GenericLoaderFactory factories) {
return new ImageLoader();
}
@Override
public void teardown() {
}
}
}
3.自定义ImageDataFetcher
public class ImageDataFetcher implements DataFetcher<InputStream> {
private volatile boolean mIsCanceled;
private final String mFilePath;
private InputStream mInputStream;
public ImageDataFetcher(String filePath) {
mFilePath = filePath;
}
/**
* 这个方法是在非UI线程中执行,我们利用此方法来加载我们的加密数据
*
* @param priority
* @throws Exception
*/
@Override
public InputStream loadData(Priority priority) throws Exception {
if (mIsCanceled) {
return null;
}
mInputStream = fetchStream(mFilePath);
return mInputStream;
}
/**
* 处理完成之后清理工作
*/
@Override
public void cleanup() {
if (mInputStream != null) {
try {
mInputStream.close();
} catch (IOException e) {
} finally {
mInputStream = null;
}
}
}
/**
* 该文件的唯一ID
*/
@Override
public String getId() {
return mFilePath;
}
/**
* 在UI线程中调用,取消加载任务
*/
@Override
public void cancel() {
mIsCanceled = true;
}
/**
* 返回解密后的数据流
*
* @param filePath 文件名
* @return inputStream
*/
private InputStream fetchStream(String filePath) {
InputStream inputStream = null;
// 返回解密数据流
return inputStream;
}
}
第八篇 Glide滑动的时候不去加载图片
监听滑动事件,不滑动的时候才去加载图片,遇见一个Crash,就是Acticity onDestory的时候,依然触发了我们的方法
java.lang.IllegalArgumentException: You cannot start a load for a destroyed activity
解决方法
onDestroy的时候,注销监听器
mRecyclerView.addOnScrollListener(new RecyclerView.OnScrollListener() {
@Override
public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
switch (newState) {
case SCROLL_STATE_IDLE:
Glide.with(mActivity).resumeRequests();
break;
case SCROLL_STATE_SETTLING:
case SCROLL_STATE_DRAGGING:
Glide.with(mActivity).pauseRequests();
break;
default:
Glide.with(mActivity).resumeRequests();
break;
}
}
});
网友评论