亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Android中超大圖片無法顯示的問題解決

 更新時間:2017年01月18日 11:58:19   作者:viclee108  
最近在工作中發(fā)現(xiàn)一個問題,在做圖片瀏覽的時候發(fā)現(xiàn)超大圖圖片無法顯示,無奈只能上網(wǎng)找解決方法,后來通過測試找到了解決的方法,下面這篇文章就主要介紹了Android中超大圖無法顯示的問題解決方法,需要的朋友可以參考借鑒。

發(fā)現(xiàn)問題

最近在做圖片瀏覽功能時遇到了一個很蛋疼的問題,在開啟硬件加速情況下,超大圖無法正常顯示(圖的長寬有一個大于9000),而且程序不會crash,只是圖片加載不出來,View顯示為黑色。通過查看日志,發(fā)現(xiàn)系統(tǒng)打印出了下面的內(nèi)容:

W OpenGLRenderer( 4014): Bitmap too large to be uploaded into a texture (600x9518, max=8192x8192) 

從日志內(nèi)容可以看出,這是由OpenGL打印出來的日志,是由于圖片的尺寸太大導(dǎo)致的。而且我們可以發(fā)現(xiàn),由于這個問題系統(tǒng)日志是以Warning級別打印出來的,并沒有拋出異常,程序并不會報錯,只是圖片顯示不出來,很難發(fā)現(xiàn)問題。當(dāng)我們把頁面的硬件加速關(guān)掉后,圖片就可以顯示出來了。

問題分析

從日志最后的內(nèi)容可以看出,OpenGL對圖片尺寸的限制是8192,這個尺寸是怎么得到的呢?是否所有的設(shè)備都是這個值呢?

要解釋這個問題,需要先來看一下GLES10中的一個常量GL_MAX_TEXTURE_SIZE,從字面上看,它表示Texture的最大值。 查看文檔:

https://www.khronos.org/opengles/sdk/1.1/docs/man/glGet.xml

這里給出的解釋是: The value gives a rough estimate of the largest texture that the GL can handle。也就是OpenGL可以處理的最大尺寸的粗略估計值。既然是粗略估計值,那肯定是有一個準(zhǔn)確值的。

網(wǎng)上搜索了一下,基本上都是說通過執(zhí)行下面的代碼,可以得到這個準(zhǔn)確值。

int[] maxTextureSize = new int[1]; 
GLES10.glGetIntegerv(GL10.GL_MAX_TEXTURE_SIZE, maxTextureSize, 0); 

然而,這段代碼在我的設(shè)備上運行的結(jié)果始終是0。這是由于在Android 5.0之后,在進行OpenGL方法的調(diào)用時,需要手動創(chuàng)建OpenGL的Context。而這個工作在5.0之前是由framework來完成的。我們這里就是因為沒有創(chuàng)建這個Context導(dǎo)致調(diào)用結(jié)果為0。

那么有效的代碼就是下面這樣子的:

private void getGLESTextureLimitBelowLollipop() { 
 int[] maxSize = new int[1]; 
 GLES10.glGetIntegerv(GLES10.GL_MAX_TEXTURE_SIZE, maxSize, 0); 
 Toast.makeText(this," " + maxSize[0],Toast.LENGTH_LONG).show(); 
} 
 
private void getGLESTextureLimitEqualAboveLollipop() { 
 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); 
 
 Toast.makeText(this," " + maxSize[0],Toast.LENGTH_LONG).show(); 
} 
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) { 
 getGLESTextureLimitEqualAboveLollipop(); 
} else { 
 getGLESTextureLimitBelowLollipop(); 
} 

在我的5.0的手機上,執(zhí)行上面的代碼,得到這個最大值maxSize是16384,也就是說,當(dāng)圖片的長和寬有一個超過這個值得時候,在開啟硬件加速的情況下,圖片就顯示不出來了。在不同的手機上運行上述代碼,運行結(jié)果不盡相同,說明這個值是設(shè)備相關(guān)的。

問題解決

既然知道了問題所在,下面就是想辦法來解決這個問題了。

首先,前面既然提到了是硬件加速導(dǎo)致的這個問題,最簡單的方法當(dāng)然是關(guān)閉硬件加速,可以在Activity級別,也可以在View級別關(guān)閉硬件加速。這種是屬于簡單粗暴型的。
Activity級別:

android:hardwareAccelerated="false"

View級別:

view.setLayerType(View.LAYER_TYPE_SOFTWARE, null);

另外一種方法是通過縮小Bitmap的方式,保證圖片的尺寸不會超過OpenGL的限制,但是,對于查看高清圖的情況,不允許對圖片進行縮放,這個方法是無效的。

最后也是最合理的方式,就是通過Android提供的BitmapRegionDecoder類來處理大圖加載。它的原理是每次只根據(jù)需要加載圖片的一部分,然后根據(jù)當(dāng)前用戶的操作去截取圖片不同部分進行更新。具體的用法可以參考官方文檔。

總結(jié)

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對各位Android開發(fā)者們能有一定的幫助,如果有疑問大家可以留言交流。

相關(guān)文章

最新評論