Android ImageView 長圖部分機(jī)型加載無法顯示問題的解決方法

相信很多朋友和我一樣汹买,用ImageView控件加載長圖的時(shí)候會遇到這樣的一個(gè)問題荡含,同一張長圖在有些機(jī)型可以正常顯示了罪,但是在部分機(jī)型確顯示不了镶殷,是不是很郁悶蜜托,然后就各種百度 Google╮(╯▽╰)╭

接下來抄囚,和大家分享一下,我遇到這個(gè)問題及我的解決之路橄务;

和大家一樣遇到這個(gè)問題幔托,百度Google,但是百度出來的各種解決方案并不是我想要的蜂挪,或是我需要的ε=(′ο`*)))唉重挑;

所以還是從源頭找起,終于在Android studio 的logcat 的打印中發(fā)現(xiàn)了這么一句異常

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

大概的意思就是Bitmap太大了棠涮,導(dǎo)致無法渲染成texture

在網(wǎng)上搜索了一番谬哀,終于找到無法加載的具體原因了,那就是

(敲重點(diǎn)(*^▽^*))當(dāng)APP開啟硬件加速的時(shí)候故爵,GPU對于openglRender 渲染有一個(gè)限制值玻粪,超過了這個(gè)限制值隅津,就無法渲染,不同的手機(jī)會有不同的限制值劲室;

找到問題的關(guān)鍵所在了伦仍,這也就解釋了為什么同一張圖片在有些手機(jī)上可以顯示,在部分機(jī)型無法顯示很洋。簡單的說充蓝,就是這張圖片的width 或height 剛好超過了openglRender?的限制值

網(wǎng)上也是提供了各種解決方案,一個(gè)簡單粗暴的方法是關(guān)閉硬件加速喉磁,

在 APP 層:

<application? android:hardwareAccelerated="false"?>

或是

在view層設(shè)置:setLayerType(View.LAYER_TYPE_SOFTWARE, null);

這樣的確解決了圖片加載問題谓苟,但你會在app運(yùn)行的時(shí)候,發(fā)現(xiàn)app變得十分卡頓协怒。果斷拋棄了這個(gè)方法

(第二種是我同事遇到類似問題的解決方案涝焙,但是我自己試過,不知道是我配置有問題還是怎樣孕暇,反正不起作用仑撞,反而之前可以正常顯示的手機(jī)不能顯示了,果斷拋棄)

仔細(xì)一想妖滔,這個(gè)問題的關(guān)鍵就在于openglRender?的限制值隧哮,如果我知道openglRender?的限制值,然后當(dāng)圖片超過這個(gè)限制的話座舍,對圖片進(jìn)行壓縮不就解決了嗎沮翔?

順著這個(gè)思路,終于在網(wǎng)上找到了一個(gè)獲取openglRender?的限制值的方案

private static int getOpenglRenderLimitValue() {

int[] maxSize =new int[1];

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

return maxSize[0];

}

有個(gè)這個(gè)限制值就可以對圖片進(jìn)行處理了曲秉,然后沾沾自喜采蚀,以為問題解決了,然后工程一跑岸浑,崩潰了 ̄へ ̄搏存,logcat 查看崩潰日志瑰步,找到崩潰的原因矢洲,發(fā)現(xiàn)是因?yàn)間etOpenglRenderLimitValue返回的值為0導(dǎo)致的。仔細(xì)找了下缩焦,發(fā)現(xiàn)logcat有這樣的一行錯(cuò)誤:

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

后面查了一下读虏,發(fā)現(xiàn)原來是GLES10.glGetIntegerv returns 0 in Lollipop?,意思就是GLES10.glGetIntegerv在Android5.0 及以上的話,返回的值為0袁滥;查了一下原因盖桥,發(fā)現(xiàn)原來在進(jìn)行OpenGL方法的調(diào)用時(shí),需要手動創(chuàng)建OpenGL的Context题翻。而這個(gè)工作在Android?5.0之前是由framework來完成的揩徊。這里就是因?yàn)闆]有創(chuàng)建這個(gè)Context導(dǎo)致調(diào)用結(jié)果為0。

知道原因后就好解決了,在 stackoverflow 上找到了對應(yīng)問題的解決方案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?的限制值熄赡,我們就可以根據(jù)自己的需要進(jìn)行處理,我這邊的處理是齿税,當(dāng)圖片的高度超過限制值的話彼硫,對bitmap進(jìn)行縮小,保證圖片的尺寸不會超過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();

// 計(jì)算縮放比例

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

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

// 取得想要縮放的matrix參數(shù)

? Matrix matrix =new Matrix();

matrix.postScale(scaleWidth, scaleHeight);

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

matrix,true);

}

好了拧篮,以上就是我解決imageview部分機(jī)型無法加載長圖的整個(gè)過程,代碼已貼上牵舱,個(gè)人覺得還蠻完美的串绩,當(dāng)然,如有更好的解決方案歡迎留言芜壁,讓我也學(xué)習(xí)學(xué)習(xí)赏参,感謝Thanks?(?ω?)?。

PS:上述的這種解決方案不適合有查看大圖需求的場景沿盅。有這種場景需求的可以通過通過Android提供的BitmapRegionDecoder類來處理大圖加載把篓。它的原理是每次只根據(jù)需要加載圖片的一部分,然后根據(jù)當(dāng)前用戶的操作去截取圖片不同部分進(jìn)行更新腰涧。具體的用法可以參考官方文檔韧掩。可以參考鴻洋大神的Android 高清加載巨圖方案 拒絕壓縮圖片窖铡,順便推薦一個(gè)工具類SubsamplingScaleImageView

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末疗锐,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子费彼,更是在濱河造成了極大的恐慌滑臊,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,542評論 6 504
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件箍铲,死亡現(xiàn)場離奇詭異雇卷,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)颠猴,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,822評論 3 394
  • 文/潘曉璐 我一進(jìn)店門关划,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人翘瓮,你說我怎么就攤上這事贮折。” “怎么了资盅?”我有些...
    開封第一講書人閱讀 163,912評論 0 354
  • 文/不壞的土叔 我叫張陵调榄,是天一觀的道長踊赠。 經(jīng)常有香客問我,道長每庆,這世上最難降的妖魔是什么臼疫? 我笑而不...
    開封第一講書人閱讀 58,449評論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮扣孟,結(jié)果婚禮上烫堤,老公的妹妹穿的比我還像新娘。我一直安慰自己凤价,他們只是感情好鸽斟,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,500評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著利诺,像睡著了一般富蓄。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上慢逾,一...
    開封第一講書人閱讀 51,370評論 1 302
  • 那天立倍,我揣著相機(jī)與錄音,去河邊找鬼侣滩。 笑死口注,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的君珠。 我是一名探鬼主播寝志,決...
    沈念sama閱讀 40,193評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼策添!你這毒婦竟也來了材部?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,074評論 0 276
  • 序言:老撾萬榮一對情侶失蹤唯竹,失蹤者是張志新(化名)和其女友劉穎乐导,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體浸颓,經(jīng)...
    沈念sama閱讀 45,505評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡物臂,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,722評論 3 335
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了猾愿。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片鹦聪。...
    茶點(diǎn)故事閱讀 39,841評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡账阻,死狀恐怖蒂秘,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情淘太,我是刑警寧澤姻僧,帶...
    沈念sama閱讀 35,569評論 5 345
  • 正文 年R本政府宣布规丽,位于F島的核電站,受9級特大地震影響撇贺,放射性物質(zhì)發(fā)生泄漏赌莺。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,168評論 3 328
  • 文/蒙蒙 一松嘶、第九天 我趴在偏房一處隱蔽的房頂上張望艘狭。 院中可真熱鬧,春花似錦翠订、人聲如沸巢音。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,783評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽官撼。三九已至,卻和暖如春似谁,著一層夾襖步出監(jiān)牢的瞬間傲绣,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,918評論 1 269
  • 我被黑心中介騙來泰國打工巩踏, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留秃诵,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,962評論 2 370
  • 正文 我出身青樓塞琼,卻偏偏與公主長得像顷链,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子屈梁,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,781評論 2 354

推薦閱讀更多精彩內(nèi)容