• Loading Large Bitmaps Efficiently in Android
  • ş imşek, Android Developer @trendyol.com

Loading large images into Memory is always a pain, as we often see OOM (Out Of Memory) bugs in app crash reports. As you all know, Android has limited memory. We must bear that in mind.

Stackoverflow has a lot of problems with loading large images, and when your app hits OOM, you can choose to copy and paste the answers directly to solve this problem. So, you can skip this article, but I want to cover some of the basics of loading large graphs and how they actually work.

I just want to explain the logic behind the image decoding. I suggest you use Picasso or Glide to load images. There is no need to reinvent the wheel.

Load the image into memory

That’s easy. You just need to use BitmapFactory to decode your images.

Bitmap bitmap = BitmapFactory.decodeResource(getResources(), R.mipmap.hqimage);
imageView.setImageBitmap(bitmap);
Copy the code

Everything looks fine. But I’ll tell you one thing, let’s look at how much memory this decoded image actually takes up.

The bitmap.getBytecount () method returns the size of the bitmap. The size of this image in memory is 12262248 bytes, equivalent to 12.3 MB. Yes, you might be confused. Because the actual size of the image on disk is about 3.5MB, the getByteCount() method returns a much larger value. Here’s why:

Images stored on disk are compressed (in JPG, PNG, or similar formats). Once an image is loaded into memory, it is no longer compressed and takes up the memory space required for all the pixels of as many images as possible.

Steps to load the large image

  • Get the width and height of the image
  • Calculate the zoom ratio based on the width and height of the image
  • Load the image into memory according to the zoom ratio.

BitmapFactory.Options

BitmapFactory can provide us with metadata for images. We can use this class for the first step.

BitmapFactory.Options options = new BitmapFactory.Options();
options.inJustDecodeBounds = true;
BitmapFactory.decodeResource(getResources(), R.mipmap.hqimage, options);
Copy the code

We will BitmapFactory. Examples of the Options to BitmapFactory. DecodeSource () method. What is the meaning of the options. InJustDecodeBounds = true? This code means that we do not want to load images into memory. We just want to get information about the image (width, height, etc.) and use that information to calculate the scaling.

We run this code and print the information for the image:

options.outHeight : 1126
options.outWidth : 2000
options.bitmap : null
Copy the code

It outputs only the height and width of the image.

Reducing Image Size (In Memory)

Now we need to compute inSampleSize. What is inSampleSize? InSampleSize is a property of the BitmapFactory.Options class that sets the image’s zoom ratio.

If we have a 1000×1000 image and set inSampleSize to 2 before decoding, we will have a 500×500 image after decoding. If we have a 200×400 image and set inSampleSize to 5 before decoding, we will have a 40×80 image after decoding.

BitmapFactory.Options options = new BitmapFactory.Options();
options.inJustDecodeBounds = true;
options.inSampleSize = 3; 
BitmapFactory.decodeResource(getResources(), R.mipmap.hqimage, options);
Copy the code

Can we just do that? No, because we don’t know what the image size is. If it’s a small image, and we make it smaller, then our users will see pixels instead of images. Some images need to be zoomed in by 5 times, others by 2 times. We can’t set the scaling ratio to a constant, so we have to calculate its value based on the size of the image.

How you calculate the value of inSampleSize is up to you. I mean, you can write inSampleSize calculations as you want. In the Official Android documentation, the result is a power of two.

Options. InSampleSize = calculateInSampleSize (options, 500500); options.inJustDecodeBounds =false;
Bitmap smallBitmap = BitmapFactory.decodeResource(getResources(), R.mipmap.hqimage, options);
Copy the code

Here we set the value to inJustDecodeBounds to false and get a Bitmap object. The bitmap.getBytecount () method now returns an image size of 3.1 MB. This is how big it is in memory. As I said before, images are compressed when stored on disk. When we load them into memory they take up more space. Using this method, we reduced the size of memory it takes from 12.3 MB to 3.1 MB, a 75% reduction.

Reducing Image Size (In Disk)

We can also compress images on disk using Bitmap’s compress method.

Let’s look at the compressed size of the image without changing its quality. 100 means the same quality as the original.

ByteArrayOutputStream bos = new ByteArrayOutputStream();
bitmap.compress(Bitmap.CompressFormat.JPEG, 100, bos);
byte[] bitmapdata = bos.toByteArray();
Copy the code

The calculated size of the image on disk is 1.6MB.

We changed the quality parameter in the COMPRESS method to 50 and calculated the image size again.

bitmap.compress(Bitmap.CompressFormat.JPEG, 50, bos);
Copy the code

The calculated size of the image on disk is 24.4 KB.

Note: When changing the quality parameter in the COMPRESS method, the compression format should be. JPEG. Changes are invalid when set to PNG format.

Here’s a comparison: