Android advanced thorough understanding of the LruCache cache mechanism principle

Android advanced thorough understanding of the LruCache cache mechanism principle

[[421640]]

Preface

Android has three levels of cache, the main ones are memory cache and hard disk cache. The implementation of these two cache mechanisms applies the LruCache algorithm. Today we will thoroughly understand the cache mechanism in Android from usage to source code analysis;

1. Introduction to LruCache Concept

1. What is LruCache?

LruCache is a cache class provided by Android 3.1, so you can directly use LruCache to implement memory cache in Android. DisLruCache is not yet part of the Android SDK, but the official Android documentation recommends using this algorithm to implement hard disk cache;

LruCache is a generic class. Its main algorithm principle is to store the most recently used objects in LinkedHashMap with strong references (the object reference method we usually use). When the cache is full, the least recently used object is removed from the memory, and get and put methods are provided to complete the cache acquisition and addition operations;

2. Use of LruCache

The use of LruCache is very simple. Let's take image cache as an example.

  1. int maxMemory = ( int ) (Runtime.getRuntime().totalMemory()/1024);
  2. int cacheSize = maxMemory/8;
  3. mMemoryCache = new LruCache<String,Bitmap>(cacheSize){
  4. @Override
  5. protected int sizeOf(String key , Bitmap value) {
  6. return value.getRowBytes()*value.getHeight()/1024;
  7. }
  8. };

① Set the size of the LruCache cache, which is generally 1/8 of the available capacity of the current process;

②Rewrite the sizeOf method to calculate the size of each picture to be cached;

Note: The total cache capacity and the size of each cache object must be in the same unit;

2. Implementation principle of LruCache

The core idea of ​​LruCache is easy to understand. It is to maintain a cache object list, where the object list is arranged in the order of access. That is, the object that has not been accessed will be placed at the end of the queue and will be eliminated. The object that has been accessed recently will be placed at the head of the queue and will be eliminated last.

1. Constructor

  1. public LruCache( int maxSize) {
  2. if (maxSize <= 0) {
  3. throw new IllegalArgumentException( "maxSize <= 0" );
  4. }
  5. this.maxSize = maxSize;
  6. this.map = new LinkedHashMap<K, V>(0, 0.75f, true );
  7. }

When creating an LruCache object, it does the following internally:

  • Record the maximum capacity maxSize;
  • Create a LinkedHashMap with an initial capacity of 0 to store objects;
  • Therefore, when creating an LruCache object, we need to specify a maximum capacity value maxSize, which can be a specific value or memory space, and the corresponding method is sizeOf:
  1. protected int sizeOf(K key , V value) {
  2. return 1;
  3. }

If we want to control the cache by quantity, we can see that the default sizeOf method returns 1, which means that each time a piece of data is added, the space it occupies is 1, until the number of added contents reaches maxSize. At this time, if we add more data, the system will select the least frequently used values ​​and remove them;

If the cache limit is memory space, you need to rewrite size to return the size of the space occupied by the object:

  1. int maxMemory = ( int ) (Runtime.getRuntime().maxMemory())/1024; //Get available memory and convert it to KB
  2. int maxCache = maxMemory/10; //The cache size is 1/10 of the available memory
  3. LruCache mMemoryCache = new LruCache<String,Bitmap>(maxCache){
  4. @Override//Override the sizeOf method to return the memory occupied by each piece of data
  5. protected int sizeOf(String key , Bitmap value) {
  6. return value.getRowBytes()*value.getHeight() / 1024;
  7. // return value.getByteCount() / 1024;
  8. }
  9. };

2. Read content

Reading content corresponds to the get method, the source code is as follows:

  1. public final V get(K key ) {
  2. if ( key == null ) { //If key is null , throw an exception directly
  3. throw new NullPointerException( "key == null" );
  4. }
  5. V mapValue;
  6. synchronized (this) {
  7. mapValue = map.get( key ); //Read the value corresponding to key
  8. if (mapValue != null ) {
  9. hitCount++;
  10. return mapValue; //If the content is obtained, return
  11. }
  12. missCount++; //If no content is read, the missCount count increases and execution continues
  13. }
  14. /* Trying to create a value may take some time. When the creation is completed, the map may have changed
  15. If the key used for creation already exists, causing a conflict, the value created by create is discarded.
  16. * Attempt to   create a value. This may take a long time , and the map
  17. * may be different when   create () returns . If a conflicting value was
  18. * added to the map while create () was working, we leave that value in  
  19. * the map and release the created value.
  20. */
  21. //If the value is not obtained, call the create method to try to create an object
  22. // The create method returns null by default . You can override this method to decide how to add a value when no value is obtained.
  23. V createdValue = create ( key );
  24. if (createdValue == null ) {
  25. return   null ;
  26. }
  27. //The following part is the modification actions on the map after creation, including storing objects, adjusting size , etc.
  28. synchronized (this) {
  29. createCount++;
  30. mapValue = map.put( key , createdValue);
  31. if (mapValue != null ) {
  32. // There was a conflict so undo that last put
  33. map.put( key , mapValue);
  34. } else {
  35. size += safeSizeOf( key , createdValue);
  36. }
  37. }
  38. if (mapValue != null ) {
  39. entryRemoved( false , key , createdValue, mapValue);
  40. return mapValue;
  41. } else {
  42. trimToSize(maxSize);
  43. return createdValue;
  44. }
  45. }

The get method tries to read the content according to the passed key. If it cannot be read, you can choose whether to create an object. If you choose to create an object, you need to override the create method to return the object to be created. It is also very simple to use;

3. Storage content

The put method is used to store content:

  1. public final V put(K key , V value) {
  2. if ( key == null || value == null ) {
  3. throw new NullPointerException( "key == null || value == null" );
  4. }
  5. V previous;
  6. synchronized (this) {
  7. putCount++;
  8. size += safeSizeOf( key , value);
  9. previous = map.put( key , value);
  10. if (previous != null ) {
  11. size -= safeSizeOf( key , previous);
  12. }
  13. }
  14. if (previous != null ) {
  15. entryRemoved( false , key , previous, value);
  16. }
  17. trimToSize(maxSize);
  18. return previous;
  19. }

When storing content, you can see how the cache size is controlled: after storing the content, size += safeSizeOf(key, value); will be executed. The default implementation of safeSizeOf is the sizeOf method. Each time an object is stored, the size will be increased by the corresponding value. If the stored key already has data, the size will remain unchanged;

It also provides an entryRemoved method, which is called when the data is removed (remove is called, the new value is overwritten, and the data that exceeds the cache is deleted). The default implementation is empty.

At the end of put, trimToSize is called, which is a method to control the cache size. This method will be called every time new data is stored. When the current size exceeds the maximum value of the cache, the least recently used data will be deleted through this method;

In addition to normal storage and reading, LruCache also provides a method to read all cached objects at once:

  1. public synchronized final Map<K, V> snapshot() {
  2. return new LinkedHashMap<K, V>(map);
  3. }

4. trimToSize() method

  1. public void trimToSize( int maxSize) {
  2. //Dead loop
  3. while ( true ) {
  4. K key ;
  5. V value;
  6. synchronized (this) {
  7. //If the map is empty and the cache size is not equal to 0 or the cache size is less than 0, throw an exception
  8. if ( size < 0 || (map.isEmpty() && size != 0)) {
  9. throw new IllegalStateException(getClass().getName()
  10. + ".sizeOf() is reporting inconsistent results!" );
  11. }
  12. //If the cache size is smaller than the maximum cache, or the map is empty, there is no need to delete the cache object and exit the loop
  13. if ( size <= maxSize || map.isEmpty()) {
  14. break;
  15. }
  16. //The iterator gets the first object, which is the element at the end of the queue and the element least recently accessed
  17. Map.Entry<K, V> toEvict = map.entrySet().iterator(). next ();
  18. key = toEvict.getKey();
  19. value = toEvict.getValue();
  20. //Delete the object and update the cache size
  21. map.remove( key );
  22. size -= safeSizeOf( key , value);
  23. evictionCount++;
  24. }
  25. entryRemoved( true , key , value, null );
  26. }
  27. }

Summarize

LruCache maintains a collection LinkedHashMap, which is sorted in access order. When the put() method is called, elements are added to the collection, and trimToSize() is called to determine whether the cache is full. If it is full, the LinkedHashMap iterator is used to delete the tail element, that is, the element least recently accessed. When the get() method is called to access the cache object, the LinkedHashMap get() method is called to obtain the corresponding collection element, and the element is updated to the head of the queue;

<<:  WeChat launches a new feature that can remotely lock the screen

>>:  The first wave! 12 UI design tips that designers must pay attention to

Recommend

How to use the points system to stimulate user retention

A reasonable points system generally consists of ...

Android performance optimization: rendering

Google recently released an online course on Andr...

Solid info! A brief discussion on 4 ideas for community user growth

Well, let’s start from the small circle again. Re...

How does Tongcheng operate blind boxes that are popular all over the Internet?

Tongcheng Travel spent only 9 yuan to buy the rea...

Dear advertiser, what misunderstanding do you have about advertising creativity?

A few days ago, a colleague sent me an App screen...

WeChat is slowly undermining the dominance of Android and iOS

[[130868]] The Economist recently wrote that the ...

HTML 5 melee: Chrome, Safari, Firefox, IE and Opera each show their strengths

Chrome and Opera are better at supporting the lat...

How to master information flow video promotion?

Currently, the trend of online video advertising ...

How to achieve user experience beyond expectations?

Each product iteration requires that the new user...