Glide加载圆形图片报错:java.lang.AbstractMethodError: abstract method "void com.bumptech.glide.load.Key.updat

本文介绍了Glide的基本用法,并针对在加载圆形图片时遇到的`java.lang.AbstractMethodError`问题进行了深入讨论。Glide是Google推荐的图片加载工具,支持多种图片资源加载。在尝试使用第三方库加载圆形图片时遇到错误,解决方案包括自定义处理方法或修改源码。文章提供了一个可行的处理方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Glide是Google公司在2014年的I/O大会上发布的,到现在有3年左右了,最先版本已经到了4.0以上了。而现在Android上的图片加载框架非常成熟,从最早的老牌图片加载框架UniversalImageLoader,到后来Google推出的Volley,再到后来的新兴军Glide和Picasso,当然还有Facebook的Fresco。而Glide是目前谷歌公司推荐的图片加载工具,好,那我们就现在来了解下Glide。

Glide基本用法

使用前,先添加依赖

dependencies {
    compile 'com.github.bumptech.glide:glide:3.5.2'
    compile 'com.android.support:support-v4:22.0.0'
}

对,上面还要添加v4的依赖,Glide需要依赖Support Library v4,别忘了。其实Support Library v4已经是应用程序的标配了,这不是什么问题。
网络加载图片,可别忘了添加网络权限

<uses-permission android:name="android.permission.INTERNET" />

现在先准备张图片:
http://imgsrc.baidu.com/baike/pic/item/d01373f082025aaf520fb3e9f9edab64034f1ad7.jpg

加载图片,一行代码:


 String url = "http://imgsrc.baidu.com/baike/pic/item/d01373f082025aaf520fb3e9f9edab64034f1ad7.jpg";
        Glide.with(this).load(url).into(imageView);

首先,调用Glide.with()方法用于创建一个加载图片的实例。with()方法可以接收Context、Activity或者Fragment类型的参数。也就是说我们选择的范围非常广,不管是在Activity还是Fragment中调用with()方法,都可以直接传this。那如果调用的地方既不在Activity中也不在Fragment中呢?也没关系,我们可以获取当前应用程序的ApplicationContext,传入到with()方法当中。注意with()方法中传入的实例会决定Glide加载图片的生命周期,如果传入的是Activity或者Fragment的实例,那么当这个Activity或Fragment被销毁的时候,图片加载也会停止。如果传入的是ApplicationContext,那么只有当应用程序被杀掉的时候,图片加载才会停止。

接下来看一下load()方法,这个方法用于指定待加载的图片资源。Glide支持加载各种各样的图片资源,包括网络图片、本地图片、应用资源、二进制流、Uri对象等等。因此load()方法也有很多个方法重载,除了我们刚才使用的加载一个字符串网址之外,你还可以这样使用load()方法:

// 加载本地图片
File file = new File(getExternalCacheDir() + "/image.jpg");
Glide.with(this).load(file).into(imageView);

// 加载应用资源
int resource = R.drawable.image;
Glide.with(this).load(resource).into(imageView);

// 加载二进制流
byte[] image = getImageBytes();
Glide.with(this).load(image).into(imageView);

// 加载Uri对象
Uri imageUri = getImageUri();
Glide.with(this).load(imageUri).into(imageView);

当然Glide还有其他一些常用的方法,比如:

Glide.with(this)
     .load(url)
     .asBitmap()//以图片的形式展示,还有asGif()方法,加载gif图片
     .placeholder(R.drawable.loading)//占位图
     .error(R.drawable.error)//加载出错时的图片
     .override(100, 100)//指定设置图片大小
     .diskCacheStrategy(DiskCacheStrategy.NONE)//缓存策略
     .into(imageView);

其中需要注意的是asBitmap()方法,这个方法的意思就是说这里只允许加载静态图片,不需要Glide去帮我们自动进行图片格式的判断了。其他图片加载设置项方法,可以通过RequestOptions类去进行设置。

设置监听器

Glide.with(this).load(internetUrl).listener(new RequestListener<String, GlideDrawable>() {   
        @Override   
        public boolean onException(Exception e, String model,   Target<GlideDrawable> target, boolean isFirstResource) {    
                  return false;  
 }   
   //这个用于监听图片是否加载完成   
   @Override   
    public boolean onResourceReady(GlideDrawable resource String model, Target<GlideDrawable> target, boolean isFromMemoryCache, boolean isFirstResource) {    
                    return false;   
    }  
}).into(imageView);

缓存策略

//Glide支持多种磁盘缓存策略:
DiskCacheStrategy.NONE :不缓存图片 
DiskCacheStrategy.SOURCE :缓存图片源文件 
DiskCacheStrategy.RESULT:缓存修改过的图片 
DiskCacheStrategy.ALL:缓存所有的图片,默认
//清理缓存
Glide.get(context).clearDiskCache();

好了,上面的基础先到这里了。

下面是本文的重点,是我在使用中报的一个错,就是在使用加载圆形图的时候的出错。这里说下,加载圆形图的时候,有2个选择,一个是自定义写的,一个是利用老外封装的一个依赖包,但是这个依赖包现在报的就是这个错。先说说是如何使用的吧。
1. 添加依赖

 // Glide图形转换工具
    compile 'jp.wasabeef:glide-transformations:2.0.1'

2.使用

// 使用构造方法 CropCircleTransformation(Context context)
     Glide.with(this)
      .load(url)
      .bitmapTransform(new CropCircleTransformation(this))
      .into(mImageView2);

听从网上的一些建议,用上面的依赖库,使用时出错了。这时候就会报出下面的错:

java.lang.AbstractMethodError: abstract method "void com.bumptech.glide.load.Key.updateDiskCacheKey(java.security.MessageDigest)"

需要在里面实现一个抽象方法,但是目前因为是依赖包,你无法进行修改源码,所以那有两种方式,一是下载源码,重新编写,一种是自己写个处理方法。
下面的这个处理方法,经过测试,可行。

/**
*
* des: 将图片转化为圆形
*/
public class GlideCircleTransform extends BitmapTransformation {
    public GlideCircleTransform(Context context) {
        super(context);
    }

    @Override protected Bitmap transform(BitmapPool pool, Bitmap toTransform, int outWidth, int outHeight) {
        return circleCrop(pool, toTransform);
    }

    private static Bitmap circleCrop(BitmapPool pool, Bitmap source) {
        if (source == null) return null;

        int size = Math.min(source.getWidth(), source.getHeight());
        int x = (source.getWidth() - size) / 2;
        int y = (source.getHeight() - size) / 2;

        // TODO this could be acquired from the pool too
        Bitmap squared = Bitmap.createBitmap(source, x, y, size, size);

        Bitmap result = pool.get(size, size, Bitmap.Config.ARGB_8888);
        if (result == null) {
            result = Bitmap.createBitmap(size, size, Bitmap.Config.ARGB_8888);
        }

        Canvas canvas = new Canvas(result);
        Paint paint = new Paint();
        paint.setShader(new BitmapShader(squared, BitmapShader.TileMode.CLAMP, BitmapShader.TileMode.CLAMP));
        paint.setAntiAlias(true);
        float r = size / 2f;
        canvas.drawCircle(r, r, r, paint);
        return result;
    }

    @Override public String getId() {
        return getClass().getName();
    }
}

使用

 Glide.with(this).glideRequet.load(url).transform(new GlideCircleTransform(context)).into(imageView);
<think>我们正在处理一个关于RecyclerView中SliderAdapter报错NullPointerException的问题,具体错误信息是"Argumentmustnotbenull",并且涉及Glide加载图片到ImageView时context为空的情况。根据错误信息,我们可以推测在SliderAdapter中,可能是在使用Glide加载图片时,传入的context参数为null,导致抛出空指针异常。分析可能的原因:1.在Adapter的构造函数中,没有正确初始化context,或者传入的context本身就是null。2.在绑定ViewHolder时,试图从一个已经销毁的context(如Activity)中获取资源,但该context已经被销毁。3.在ViewHolder中,ImageView的context不可用。解决方案:1.确保在创建SliderAdapter时传入的context不为null。通常我们传入Activity的context或者Application的context。在RecyclerView中使用时,建议传入Activity的context,因为Application的context在某些情况下可能无法正确加载资源(比如需要主题相关的资源)。2.在Adapter中,使用传入的context时,检查是否为null。3.考虑使用Application的context来避免内存泄漏,但要注意Application的context不能用于需要主题的资源。对于图片加载,通常Application的context是可以的,但如果需要和Activity生命周期绑定(例如在Activity销毁时取消请求),则最好使用Activity的context。Glide可以自动使用ImageView的context,所以我们可以尝试在Adapter中不直接使用传入的context,而是使用ImageView的context。具体到代码,我们可以这样处理:在Adapter中,我们可以这样写:```javapublicclassSliderAdapterextendsRecyclerView.Adapter<SliderAdapter.SliderViewHolder>{privateList<String>imageUrls;//如果不需要Activity特定的功能,可以不用存储context//因为Glide可以使用ImageView的contextpublicSliderAdapter(List<String>imageUrls){this.imageUrls=imageUrls;}@NonNull@OverridepublicSliderViewHolderonCreateViewHolder(@NonNullViewGroupparent,intviewType){Viewview=LayoutInflater.from(parent.getContext()).inflate(R.layout.slider_item,parent,false);returnnewSliderViewHolder(view);}@OverridepublicvoidonBindViewHolder(@NonNullSliderViewHolderholder,intposition){StringimageUrl=imageUrls.get(position);//使用holder.itemView的context,它是从parent的context中来的,也就是RecyclerView所在的ActivityGlide.with(holder.itemView.getContext()).load(imageUrl).into(holder.imageView);}//...其他代码staticclassSliderViewHolderextendsRecyclerView.ViewHolder{ImageViewimageView;publicSliderViewHolder(@NonNullViewitemView){super(itemView);imageView=itemView.findViewById(R.id.imageView);}}}```这样,我们就不需要在Adapter中保存context,避免了context为null的情况,因为holder.itemView.getContext()在绑定的时候一定是非空的(只要RecyclerView还附着在Activity上)。但是,如果Activity已经被销毁,此时RecyclerView也会被销毁,所以正常情况下不会在已销毁的Activity上调用绑定。如果遇到在Activity销毁后仍然尝试加载图片Glide会自己处理,不会导致崩溃。另外,错误信息中提到了'Argumentmustnotbenull',这可能是Glide在检查到传入的context为null时抛出的。因此,使用holder.itemView.getContext()可以确保每次绑定都使用当前的context。如果问题仍然存在,请检查:-在onBindViewHolder中,holder.itemView.getContext()是否为null?理论上不应该为null,除非ViewHolder的itemView没有关联context。-检查传入的imageUrl是否为null,如果为null,Glide可能会抛出异常,但错误信息可能不同。根据引用[1]中的内容,Glide加载失败时会调用onException方法,并设置错误占位图。如果model(也就是图片URL)为null,会尝试获取fallbackDrawable、errorDrawable、placeholderDrawable,最后调用target.onLoadFailed。所以如果URL为null,不会直接导致NullPointerException,而是会显示错误占位图(如果有的话)。因此,这里更可能是context为null导致的。所以,重点检查context的来源。如果按照上述方法修改后问题解决,那么可能是之前传入的context在某些情况下变成了null。另外,根据引用[2],我们也要检查AndroidManifest.xml中是否声明了相关的Activity,虽然这个错误和当前的NullPointerException没有直接关系,但是如果有Activity未声明,可能导致Activity的context不可用,但这种情况通常会在启动Activity时崩溃,而不是在Adapter中。根据引用[3],我们也可以考虑使用ViewModel来管理数据,避免因为配置变更(如旋转屏幕)导致Activity重建而引发的context问题。但是,在Adapter中,我们使用当前绑定时的context(即holder.itemView.getContext())已经能够应对Activity重建,因为重建后RecyclerView会重新绑定ViewHolder,此时获取的context是新的Activity的context。因此,解决方案是:在Adapter的onBindViewHolder中,使用holder.itemView.getContext()作为Glide的上下文,而不是在Adapter中保存一个context成员变量。如果问题解决了,那么就不需要进一步操作。如果仍然出现NullPointerException,请检查是否在onBindViewHolder中还有其他代码使用了可能为null的对象。最后,关于引用[4],它提供了Retrofit的依赖,和当前问题无关。请尝试修改Adapter代码,使用上述方法。相关问题:1.在RecyclerView的Adapter中,如何正确使用Context避免内存泄漏?2.Glide加载图片时有哪些常见的异常?如何处理?3.如何避免在RecyclerView中出现NullPointerException?</think>### RecyclerView中SliderAdapter报错NullPointerException的解决方案 您在RecyclerView的SliderAdapter中遇到的`java.lang.NullPointerException: Argument must not be null`错误,通常是由Glide加载图片时关键参数为null引起的。以下是系统分析和解决方案: #### 问题原因分析 1. **Context为空**(最常见原因) Glide初始化需要有效的Context,若Adapter构造函数中未正确传递Context,或Context被提前销毁,会导致: ```java // 错误示例 Glide.with(null) // 此处Context为null .load(imageUrl) .into(holder.imageView); ``` 2. **ImageView未正确初始化** ViewHolder中未成功绑定XML布局的ImageView: ```java // ViewHolder错误示例 public SliderViewHolder(View itemView) { super(itemView); // 忘记绑定imageView = itemView.findViewById(R.id.imageView); } ``` 3. **图片URL/路径为null** 数据源中的图片路径为空时,Glide会抛出NPE: ```java // 数据问题示例 List<String> imageUrls = getData(); // 可能包含null元素 ``` 4. **生命周期冲突** 当Activity/Fragment已销毁但Adapter仍在尝试加载图片时发生[^3]。 --- #### 解决方案 ##### 1. 确保Context正确传递(关键) **Adapter构造函数修正:** ```java public class SliderAdapter extends RecyclerView.Adapter<SliderAdapter.ViewHolder> { private Context context; // 声明Context变量 // 构造函数中接收Context public SliderAdapter(Context context, List<String> imageUrls) { this.context = context.getApplicationContext(); // 使用ApplicationContext避免内存泄漏 this.imageUrls = imageUrls; } } ``` **Activity/Fragment中初始化:** ```java // Activity中正确初始化 SliderAdapter adapter = new SliderAdapter(MainActivity.this, imageUrls); recyclerView.setAdapter(adapter); ``` ##### 2. 验证ImageView绑定 **ViewHolder修正:** ```java public static class ViewHolder extends RecyclerView.ViewHolder { ImageView imageView; // 确保变量名与XML一致 public ViewHolder(View itemView) { super(itemView); imageView = itemView.findViewById(R.id.img_slider); // 确认ID匹配 } } ``` ##### 3. 检查数据源和Glide调用 **onBindViewHolder安全加载:** ```java @Override public void onBindViewHolder(ViewHolder holder, int position) { String imageUrl = imageUrls.get(position); // 添加空值检查 if (imageUrl != null && !imageUrl.isEmpty()) { Glide.with(context) .load(imageUrl) .placeholder(R.drawable.ic_placeholder) // 添加占位符 .error(R.drawable.ic_error) // 添加错误图 .into(holder.imageView); } else { // 处理空URL情况 holder.imageView.setImageResource(R.drawable.ic_fallback); } } ``` ##### 4. 处理生命周期问题 - 使用`Glide.with(context)`时,Glide会自动绑定生命周期[^3] - 对于Fragment中的RecyclerView,使用`getActivity()?`的非空断言: ```kotlin // Kotlin示例 adapter = SliderAdapter(requireActivity(), imageList) ``` --- #### 预防措施 1. **启用Null检查** 在build.gradle中添加: ```gradle android { compileOptions { sourceCompatibility JavaVersion.VERSION_1_8 targetCompatibility JavaVersion.VERSION_1_8 } kotlinOptions { jvmTarget = '1.8' freeCompilerArgs += ["-Xjsr305=strict"] // 启用严格Null检查 } } ``` 2. **使用Objects.requireNonNull** ```java public SliderAdapter(Context context) { this.context = Objects.requireNonNull(context, "Context cannot be null"); } ``` 3. **ViewModel数据管理** 通过ViewModel管理数据,避免配置变更导致Context失效[^3]: ```java ViewModel model = new ViewModelProvider(this).get(SliderViewModel.class); model.getImages().observe(this, images -> { adapter.setData(images); }); ``` --- ### 相关问题 1. **如何正确处理RecyclerView中图片加载的内存泄漏问题?** [涉及Glide生命周期管理及弱引用Context的最佳实践] 2. **Glide加载失败时如何显示自定义错误占位图?** [需讨论`error()`/`fallback()`API的使用及资源管理] 3. **ViewModel在RecyclerView Adapter中如何正确应用?** [结合生命周期感知组件解决数据持久化问题[^3]] 4. **除了Glide,还有哪些Android图片加载库能更好地处理Null安全?** [比较Coil、Picasso等库的Null安全机制] 5. **如何编写单元测试来检测Adapter中的NullPointerException?** [使用Mockito模拟Context/View的测试方案] > 请尝试修正Context传递和空值检查后验证问题是否解决。若仍出现异常,建议检查AndroidManifest.xml中的Activity声明[^2]和RecyclerView的布局层级。 [^1]: Glide空值处理机制 [^2]: Android组件注册要求 [^3]: ViewModel生命周期管理优势
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值