中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

android大圖片加載OOM處理問題怎么解決

發布時間:2022-10-18 16:31:09 來源:億速云 閱讀:129 作者:iii 欄目:編程語言

這篇文章主要介紹“android大圖片加載OOM處理問題怎么解決”的相關知識,小編通過實際案例向大家展示操作過程,操作方法簡單快捷,實用性強,希望這篇“android大圖片加載OOM處理問題怎么解決”文章能幫助大家解決問題。

我們在編寫Android程序的時候經常要用到許多圖片,不同圖片總是會有不同的形狀、不同的大小,但在大多數情況下,這些圖片都會大于我們程序所需要的大小。比如說系統圖片庫里展示的圖片大都是用手機攝像頭拍出來的,這些圖片的分辨率會比我們手機屏幕的分辨率高得多。大家應該知道,我們編寫的應用程序都是有一定內存限制的,程序占用了過高的內存就容易出現OOM(OutOfMemory)異常。

先看一個栗子:主要功能為:ListView 的滑動列表展示,每個條目是一張圖片

xml主頁面布局

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    <ListView
        android:id="@+id/listView"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
</RelativeLayout>

ListView 的item 布局

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">

    <ImageView
        android:id="@+id/imageView"
        android:layout_width="match_parent"
        android:layout_height="100dp"
</LinearLayout>

布局界面非常簡單,不用多說什么,java代碼

public class MainActivity extends AppCompatActivity

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        ListView listView = (ListView) findViewById(R.id.listView);
        listView.setAdapter(new MyAdapter());
    }
    class MyAdapter extends BaseAdapter {
        @Override
        public int getCount() {
            return 50;
        }

        @Override
        public Object getItem(int position) {
            return null;
        }

        @Override
        public long getItemId(int position) {
            return position;
        }

        @Override
        public View getView(int position, View convertView, ViewGroup parent) {

            View view = View.inflate(MainActivity.this, R.layout.list_item, null);
            ImageView imageView = (ImageView) view.findViewById(R.id.imageView);

            imageView.setImageResource(R.drawable.girl);//完全顯示圖片大小,ListView 滑動界面異常卡頓

            return

由于我們測試的圖片非常大,屬于超清的那種,我們用默認的imageView.setImageResource(R.drawable.girl) 方法,是完全顯示此高清圖片,此時

在模擬器上運行之后,雖然能夠顯示,但是界面灰常卡頓有木有,可以說根本劃不動的,口說無憑,上圖

唉!不好意思沒圖了,本來已經錄好圖了,奈何超過2M沒法上傳,總之就是很卡很卡啦

看看,不騙人,真的很卡呀!!!

然后,接下來我們就要解決這個問題了,怎么解決呢,我們通過思考就會發現,其實造成這種結果單位主要原因就是要顯示的圖片太大了,我們完全不需要展示這麼大的圖片,沒錯我們可以把圖片壓縮一下,比如壓縮成 100*100 的啦.

我們主要使用的就是BitmapFactory.decodeResource()方法,該方法有多個重載,我們使用
decodeResource (Resources res,
int id,
BitmapFactory.Options opts)(該方法返回一個Bitmap對象.)
和 BitmapFactory.Options() 這個類該類可以對要加載的圖片進行配置

下面我給出代碼詳細注釋:

首先創建一個方法 decodeSampleBitmapFromResource() 主要來把大圖片解析為適當的圖片

public static Bitmap decodeSampleBitmapFromResource(Resources res, int resId, int reqWidth, int reqHeight) {
        // 第一次解析將inJustDecodeBounds設置為true,來獲取圖片大小
        BitmapFactory.Options options = new BitmapFactory.Options();
        //將這個參數的inJustDecodeBounds屬性設置為true就可以讓解析方法禁止為bitmap分配內存
        options.inJustDecodeBounds = true;

        /*解析源圖片,返回一個 bitmap 對象,當 options.inJustDecodeBounds = true;
        禁止為bitmap分配內存,返回值也不再是一個Bitmap對象,而是null。雖然Bitmap是null了,
         但是BitmapFactory.Options的outWidth、outHeight和outMimeType屬性都會被賦值。
         這個技巧讓我們可以在加載圖片之前就獲取到圖片的長寬值和MIME類型,從而根據情況對圖片進行壓縮*/
        BitmapFactory.decodeResource(res, resId, options);

        // 這個方法用來計算inSampleSize值
        options.inSampleSize = calculateInSampleSize(options, reqWidth, reqHeight);
        // 使用獲取到的inSampleSize值再次解析圖片
        options.inJustDecodeBounds = false;

        /*計算完inSampleSize 的合適大小后,需要把 options.inJustDecodeBounds = false;
        然后把再 BitmapFactory.decodeResource(res,resId,options)
        此時  options.inJustDecodeBounds = false; ,BitmapFactory.decodeResource() 方法返回一個bitmap對象給 imageView.setImageBitmap()方法
        從而顯示一個合適大小的圖片
        */
        return

calculateInSampleSize(options, reqWidth, reqHeight); 的實現

/*計算出合適的inSampleSize值:*/
    public static int calculateInSampleSize(BitmapFactory.Options options, int reqWidth, int reqHeight) {
        //源圖片的高度和寬度
        final int height = options.outHeight;//得到要加載的圖片高度
        final int width = options.outWidth;
        int inSampleSize = 1;
        if (height > reqHeight || width > reqWidth) {
            //計算出實際寬高和目標寬高的比例
            final int heightRatio = Math.round((float) height / (float) reqHeight);
            final int widthRatio = Math.round((float) width / (float) reqWidth);

            // 選擇寬和高中最小的比率作為inSampleSize的值,這樣可以保證最終圖片的寬和高
            // 一定都會大于等于目標的寬和高。
            inSampleSize = heightRatio < widthRatio ? heightRatio : widthRatio;

        }
        return

上面是對用到的兩個自定義的方法的解釋,把他倆用到 主Java 代碼中如下:

最后給出java完整程序

public class MainActivity extends AppCompatActivity

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        ListView listView = (ListView) findViewById(R.id.listView);
        listView.setAdapter(new MyAdapter());
    }

    class MyAdapter extends BaseAdapter {
        @Override
        public int getCount() {
            return 50;
        }

        @Override
        public Object getItem(int position) {
            return null;
        }

        @Override
        public long getItemId(int position) {
            return position;
        }

        @Override
        public View getView(int position, View convertView, ViewGroup parent) {

            View view = View.inflate(MainActivity.this, R.layout.list_item, null);
            ImageView imageView = (ImageView) view.findViewById(R.id.imageView);
           // imageView.setImageResource(R.drawable.girl);//完全顯示圖片大小,ListView 滑動界面異常卡頓

            //對非常大的圖片進行壓縮顯示(壓縮成100*100顯示),以適合的寬高顯示大小
            imageView.setImageBitmap(decodeSampleBitmapFromResource(getResources(), R.drawable.girl, 100, 100));
            return view;
        }

    }

    /*計算出合適的inSampleSize值:*/
    public static int calculateInSampleSize(BitmapFactory.Options options, int reqWidth, int reqHeight) {
        //源圖片的高度和寬度
        final int height = options.outHeight;//得到要加載的圖片高度
        final int width = options.outWidth;
        int inSampleSize = 1;
        if (height > reqHeight || width > reqWidth) {
            //計算出實際寬高和目標寬高的比例
            final int heightRatio = Math.round((float) height / (float) reqHeight);
            final int widthRatio = Math.round((float) width / (float) reqWidth);

            // 選擇寬和高中最小的比率作為inSampleSize的值,這樣可以保證最終圖片的寬和高
            // 一定都會大于等于目標的寬和高。
            inSampleSize = heightRatio < widthRatio ? heightRatio : widthRatio;

        }
        return inSampleSize;

    }

    public static Bitmap decodeSampleBitmapFromResource(Resources res, int resId, int reqWidth, int reqHeight) {
        // 第一次解析將inJustDecodeBounds設置為true,來獲取圖片大小
        BitmapFactory.Options options = new BitmapFactory.Options();
        //將這個參數的inJustDecodeBounds屬性設置為true就可以讓解析方法禁止為bitmap分配內存
        options.inJustDecodeBounds = true;

        //解析源圖片,返回一個 bitmap 對象,當 options.inJustDecodeBounds = true;
        /*禁止為bitmap分配內存,返回值也不再是一個Bitmap對象,而是null。雖然Bitmap是null了,
         但是BitmapFactory.Options的outWidth、outHeight和outMimeType屬性都會被賦值。
         這個技巧讓我們可以在加載圖片之前就獲取到圖片的長寬值和MIME類型,從而根據情況對圖片進行壓縮*/
        BitmapFactory.decodeResource(res, resId, options);

        // 調用上面定義的方法計算inSampleSize值
        options.inSampleSize = calculateInSampleSize(options, reqWidth, reqHeight);
        // 使用獲取到的inSampleSize值再次解析圖片
        options.inJustDecodeBounds = false;

        /*計算完inSampleSize 的合適大小后,需要把 options.inJustDecodeBounds = false;
        然后把再 BitmapFactory.decodeResource(res,resId,options)
        此時  options.inJustDecodeBounds = false; ,BitmapFactory.decodeResource() 方法返回一個bitmap對象給 imageView.setImageBitmap()方法
        從而顯示一個合適大小的圖片
        */
        return

我們把 imageView.setImageResource(R.drawable.girl);//完全顯示圖片大小,ListView 滑動界面異常卡頓 這行代碼,改為這樣寫

 //對非常大的圖片進行壓縮顯示(壓縮成100*100顯示),以適合的寬高顯示大小
imageView.setImageBitmap(decodeSampleBitmapFromResource(getResources(), R.drawable.girl, 100, 100));

在我快速的來回滑動當中,然后就

android大圖片加載OOM處理問題怎么解決

并且報了下面的異常信息

android大圖片加載OOM處理問題怎么解決

很明顯的OOM了有木有,那我們又該用啥辦法解決呢,仔細思考我們有發現了問題的關鍵所在,就是ListView 條目的圖片快速加載,快速釋放,在某一時刻,圖片加載過多,導致了 OOM 的問題.我們自然就可以想到,能不能別讓滾出屏幕的圖片離開呢,是不是可以把他們緩存起來呢,

所以,如果朋友們熟悉LruCache 這個類的話,又會想到解決方案了,不熟悉的也沒關系,我還是會給出詳細注釋的:

首先看 我們自定義的 initLruCache(); 方法,在 onCreate() 方法中調用

/**
     * 初始化緩存
     */
    private void initLruCache() {

        // 獲取到可用內存的最大值,使用內存超出這個值會引起OutOfMemory異常。
        // LruCache通過構造函數傳入緩存值,以KB為單位。
        int maxMemory = (int) (Runtime.getRuntime().maxMemory() / 1024);
        // 使用最大可用內存值的1/8作為緩存的大小。
        int cacheSize = maxMemory / 8;

        mMemoryCache = new LruCache<String, Bitmap>(cacheSize) {
            @Override
            protected int sizeOf(String key, Bitmap bitmap) {
                // 重寫此方法來衡量每張圖片的大小,默認返回圖片數量。
                return bitmap.getByteCount() / 1024;
            }
        };
    }

然后再ListView 的適配器中的 getView() 方法中給ImageView 設置圖片時,吧圖片從緩存中取出,放入就行了

//得到資源id的唯一標示
            Bitmap bitmap = mMemoryCache.get(imageKey);//從lruCache中取出該緩存
            //如果緩存中有該資源
            if (bitmap != null) {
                imageView.setImageBitmap(bitmap);
            } else {
                //對非常大的圖片進行壓縮顯示(壓縮成100*100顯示),以適合的寬高顯示大小
                Bitmap bitmap2 = decodeSampleBitmapFromResource(getResources(), R.drawable.girl, 100, 100);
                imageView.setImageBitmap(bitmap2);
                mMemoryCache.put(imageKey, bitmap2);//添加到緩存中

完整java 代碼如下

public class MainActivity extends AppCompatActivity
    private LruCache<String, Bitmap> mMemoryCache;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        initLruCache();

        ListView listView = (ListView) findViewById(R.id.listView);
        listView.setAdapter(new MyAdapter());
    }

    /**
     * 初始化緩存
     */
    private void initLruCache() {

        // 獲取到可用內存的最大值,使用內存超出這個值會引起OutOfMemory異常。
        // LruCache通過構造函數傳入緩存值,以KB為單位。
        int maxMemory = (int) (Runtime.getRuntime().maxMemory() / 1024);
        // 使用最大可用內存值的1/8作為緩存的大小。
        int cacheSize = maxMemory / 8;

        mMemoryCache = new LruCache<String, Bitmap>(cacheSize) {
            @Override
            protected int sizeOf(String key, Bitmap bitmap) {
                // 重寫此方法來衡量每張圖片的大小,默認返回圖片數量。
                return bitmap.getByteCount() / 1024;
            }
        };
    }


    class MyAdapter extends BaseAdapter {
        @Override
        public int getCount() {
            return 50;
        }

        @Override
        public Object getItem(int position) {
            return null;
        }

        @Override
        public long getItemId(int position) {
            return position;
        }

        @Override
        public View getView(int position, View convertView, ViewGroup parent) {

            View view = View.inflate(MainActivity.this, R.layout.list_item, null);
            ImageView imageView = (ImageView) view.findViewById(R.id.imageView);
//            imageView.setImageResource(R.drawable.girl);//完全顯示圖片大小,ListView 滑動界面異常卡頓

            String imageKey = String.valueOf(R.drawable.girl);//得到資源id的唯一標示
            Bitmap bitmap = mMemoryCache.get(imageKey);//從lruCache中取出該緩存
            //如果緩存中有該資源
            if (bitmap != null) {
                imageView.setImageBitmap(bitmap);
            } else {
                //對非常大的圖片進行壓縮顯示(壓縮成100*100顯示),以適合的寬高顯示大小
                Bitmap bitmap2 = decodeSampleBitmapFromResource(getResources(), R.drawable.girl, 100, 100);
                imageView.setImageBitmap(bitmap2);
                mMemoryCache.put(imageKey, bitmap2);//添加到緩存中
            }
            return view;
        }

    }


    /*計算出合適的inSampleSize值:*/
    public static int calculateInSampleSize(BitmapFactory.Options options, int reqWidth, int reqHeight) {
        //源圖片的高度和寬度
        final int height = options.outHeight;//得到要加載的圖片高度
        final int width = options.outWidth;
        int inSampleSize = 1;
        if (height > reqHeight || width > reqWidth) {
            //計算出實際寬高和目標寬高的比例
            final int heightRatio = Math.round((float) height / (float) reqHeight);
            final int widthRatio = Math.round((float) width / (float) reqWidth);

            // 選擇寬和高中最小的比率作為inSampleSize的值,這樣可以保證最終圖片的寬和高
            // 一定都會大于等于目標的寬和高。
            inSampleSize = heightRatio < widthRatio ? heightRatio : widthRatio;

        }
        return inSampleSize;
    }

    public static Bitmap decodeSampleBitmapFromResource(Resources res, int resId, int reqWidth, int reqHeight) {
        // 第一次解析將inJustDecodeBounds設置為true,來獲取圖片大小
        BitmapFactory.Options options = new BitmapFactory.Options();
        //將這個參數的inJustDecodeBounds屬性設置為true就可以讓解析方法禁止為bitmap分配內存
        options.inJustDecodeBounds = true;

        //解析源圖片,返回一個 bitmap 對象,當 options.inJustDecodeBounds = true;
        /*禁止為bitmap分配內存,返回值也不再是一個Bitmap對象,而是null。雖然Bitmap是null了,
         但是BitmapFactory.Options的outWidth、outHeight和outMimeType屬性都會被賦值。
         這個技巧讓我們可以在加載圖片之前就獲取到圖片的長寬值和MIME類型,從而根據情況對圖片進行壓縮*/
        BitmapFactory.decodeResource(res, resId, options);

        // 調用上面定義的方法計算inSampleSize值
        options.inSampleSize = calculateInSampleSize(options, reqWidth, reqHeight);
        // 使用獲取到的inSampleSize值再次解析圖片
        options.inJustDecodeBounds = false;

        /*計算完inSampleSize 的合適大小后,需要把 options.inJustDecodeBounds = false;
        然后把再 BitmapFactory.decodeResource(res,resId,options)
        此時  options.inJustDecodeBounds = false; ,BitmapFactory.decodeResource() 方法返回一個bitmap對象給 imageView.setImageBitmap()方法
        從而顯示一個合適大小的圖片
        */
        return

關于“android大圖片加載OOM處理問題怎么解決”的內容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業相關的知識,可以關注億速云行業資訊頻道,小編每天都會為大家更新不同的知識點。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

天峻县| 同心县| 连城县| 修文县| 鄯善县| 房产| 陆丰市| 岫岩| 甘孜| 承德市| 迁西县| 伊春市| 瓦房店市| 二连浩特市| 文安县| 教育| 象州县| 淮滨县| 上蔡县| 新昌县| 呈贡县| 涟水县| 永清县| 锦屏县| 肥乡县| 鄂州市| 海南省| 嘉义市| 马边| 宿松县| 长子县| 通化县| 常德市| 安仁县| 呼玛县| 资溪县| 宜兴市| 辉南县| 大关县| 马尔康县| 海阳市|