Android ImageView 长图部分机型加载无法显示问题的解决方法

相信很多朋友和我一样,用ImageView控件加载长图的时候会遇到这样的一个问题,同一张长图在有些机型可以正常显示,但是在部分机型确显示不了,是不是很郁闷,然后就各种百度 Google╮(╯▽╰)╭

接下来,和大家分享一下,我遇到这个问题及我的解决之路;

和大家一样遇到这个问题,百度Google,但是百度出来的各种解决方案并不是我想要的,或是我需要的ε=(´ο`*)))唉;

所以还是从源头找起,终于在Android studio 的logcat 的打印中发现了这么一句异常

W/OpenGLRenderer:Bitmaptoolargetobeuploaded intoatexture(1080x4196, max=4096x4096)

大概的意思就是Bitmap太大了,导致无法渲染成texture

在网上搜索了一番,终于找到无法加载的具体原因了,那就是

(敲重点(*^▽^*))当APP开启硬件加速的时候,GPU对于openglRender 渲染有一个限制值,超过了这个限制值,就无法渲染,不同的手机会有不同的限制值;

找到问题的关键所在了,这也就解释了为什么同一张图片在有些手机上可以显示,在部分机型无法显示。简单的说,就是这张图片的width 或height 刚好超过了openglRender 的限制值

网上也是提供了各种解决方案,一个简单粗暴的方法是关闭硬件加速,

在 APP 层:

<application  android:hardwareAccelerated="false" >

或是

在view层设置:setLayerType(View.LAYER_TYPE_SOFTWARE, null);

这样的确解决了图片加载问题,但你会在app运行的时候,发现app变得十分卡顿。果断抛弃了这个方法

(第二种是我同事遇到类似问题的解决方案,但是我自己试过,不知道是我配置有问题还是怎样,反正不起作用,反而之前可以正常显示的手机不能显示了,果断抛弃)

仔细一想,这个问题的关键就在于openglRender 的限制值,如果我知道openglRender 的限制值,然后当图片超过这个限制的话,对图片进行压缩不就解决了吗?

顺着这个思路,终于在网上找到了一个获取openglRender 的限制值的方案

private static int getOpenglRenderLimitValue() {

int[] maxSize =new int[1];

GLES10.glGetIntegerv(GLES10.GL_MAX_TEXTURE_SIZE, maxSize,0);

return maxSize[0];

}

有个这个限制值就可以对图片进行处理了,然后沾沾自喜,以为问题解决了,然后工程一跑,崩溃了 ̄へ ̄,logcat 查看崩溃日志,找到崩溃的原因,发现是因为getOpenglRenderLimitValue返回的值为0导致的。仔细找了下,发现logcat有这样的一行错误:

E/libEGL﹕ call to OpenGL ES API with no current context (logged once per thread)

后面查了一下,发现原来是GLES10.glGetIntegerv returns 0 in Lollipop ,意思就是GLES10.glGetIntegerv在Android5.0 及以上的话,返回的值为0;查了一下原因,发现原来在进行OpenGL方法的调用时,需要手动创建OpenGL的Context。而这个工作在Android 5.0之前是由framework来完成的。这里就是因为没有创建这个Context导致调用结果为0。

知道原因后就好解决了,在 stackoverflow 上找到了对应问题的解决方案GLES10.glGetIntegerv returns 0 in Lollipop only

完整的代码如下:

public static int getOpenglRenderLimitValue() {

int maxsize ;

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {

maxsize =getOpenglRenderLimitEqualAboveLollipop();

}else {

maxsize =getOpenglRenderLimitBelowLollipop();

}

return maxsize ==0 ? 4096 : maxsize;

}

private static int getOpenglRenderLimitBelowLollipop() {

int[] maxSize =new int[1];

GLES10.glGetIntegerv(GLES10.GL_MAX_TEXTURE_SIZE, maxSize,0);

return maxSize[0];

}

private static int getOpenglRenderLimitEqualAboveLollipop() {

EGL10 egl = (EGL10) EGLContext.getEGL();

EGLDisplay dpy = egl.eglGetDisplay(EGL10.EGL_DEFAULT_DISPLAY);

int[] vers =new int[2];

egl.eglInitialize(dpy, vers);

int[] configAttr = {

EGL10.EGL_COLOR_BUFFER_TYPE, EGL10.EGL_RGB_BUFFER,

EGL10.EGL_LEVEL,0,

EGL10.EGL_SURFACE_TYPE, EGL10.EGL_PBUFFER_BIT,

EGL10.EGL_NONE

  };

EGLConfig[] configs =new EGLConfig[1];

int[] numConfig =new int[1];

egl.eglChooseConfig(dpy, configAttr, configs,1, numConfig);

if (numConfig[0] ==0) {// TROUBLE! No config found.

  }

EGLConfig config = configs[0];

int[] surfAttr = {

EGL10.EGL_WIDTH,64,

EGL10.EGL_HEIGHT,64,

EGL10.EGL_NONE

  };

EGLSurface surf = egl.eglCreatePbufferSurface(dpy, config, surfAttr);

final int EGL_CONTEXT_CLIENT_VERSION =0x3098;// missing in EGL10

  int[] ctxAttrib = {

EGL_CONTEXT_CLIENT_VERSION,1,

EGL10.EGL_NONE

  };

EGLContext ctx = egl.eglCreateContext(dpy, config, EGL10.EGL_NO_CONTEXT, ctxAttrib);

egl.eglMakeCurrent(dpy, surf, surf, ctx);

int[] maxSize =new int[1];

GLES10.glGetIntegerv(GLES10.GL_MAX_TEXTURE_SIZE, maxSize,0);

egl.eglMakeCurrent(dpy, EGL10.EGL_NO_SURFACE, EGL10.EGL_NO_SURFACE,

EGL10.EGL_NO_CONTEXT);

egl.eglDestroySurface(dpy, surf);

egl.eglDestroyContext(dpy, ctx);

egl.eglTerminate(dpy);

return maxSize[0];

}

好了,知道openglRender 的限制值,我们就可以根据自己的需要进行处理,我这边的处理是,当图片的高度超过限制值的话,对bitmap进行缩小,保证图片的尺寸不会超过OpenGL的限制,附上代码:

public static void loadImage(Context context, String url,final ImageView image,

final int width,final int height) {

if (height > getOpenglRenderLimitValue()) {

width = width * getOpenglRenderLimitValue() /height;

height = getOpenglRenderLimitValue();

Glide.with(context)

.load(url)

.asBitmap()

.placeholder(R.color.whitesmoke)

.error(R.color.whitesmoke)

.override(width, height)

.into(new SimpleTarget() {

@Override

          public void onResourceReady(Bitmap bitmap,

GlideAnimation glideAnimation) {

image.setImageBitmap(decodeSampledBitmap(bitmap,width,height));

}

});

}else {

Glide.with(context)

.load(url)

.asBitmap()

.placeholder(R.color.whitesmoke)

.error(R.color.whitesmoke)

.override(width, height)

.into(image);

}

}

public static Bitmap decodeSampledBitmap(Bitmap bitmap,

int reqWidth,int reqHeight) {

int width = bitmap.getWidth();

int height = bitmap.getHeight();

// 计算缩放比例

  float scaleWidth = ((float) reqWidth) / width;

float scaleHeight = ((float) reqHeight) / height;

// 取得想要缩放的matrix参数

  Matrix matrix =new Matrix();

matrix.postScale(scaleWidth, scaleHeight);

return Bitmap.createBitmap(bitmap,0,0, width, height,

matrix,true);

}

好了,以上就是我解决imageview部分机型无法加载长图的整个过程,代码已贴上,个人觉得还蛮完美的,当然,如有更好的解决方案欢迎留言,让我也学习学习,感谢Thanks♪(・ω・)ノ。

PS:上述的这种解决方案不适合有查看大图需求的场景。有这种场景需求的可以通过通过Android提供的BitmapRegionDecoder类来处理大图加载。它的原理是每次只根据需要加载图片的一部分,然后根据当前用户的操作去截取图片不同部分进行更新。具体的用法可以参考官方文档。可以参考鸿洋大神的Android 高清加载巨图方案 拒绝压缩图片,顺便推荐一个工具类SubsamplingScaleImageView

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,039评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,223评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,916评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,009评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,030评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,011评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,934评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,754评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,202评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,433评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,590评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,321评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,917评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,568评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,738评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,583评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,482评论 2 352

推荐阅读更多精彩内容