65

I am new to developing in android. In my android app I'm using HashMap, but I'm getting a warning:

**"Use new SparseArray<String>(...) instead for better performance"**

What does this mean, and how can I use SparseArray<String> instead?

Naveen Kumar
  • 3,738
  • 4
  • 29
  • 50

4 Answers4

117

Use new SparseArray<String>(...) instead for better performance

You are getting this warning because of reason described here.

SparseArrays map integers to Objects. Unlike a normal array of Objects, there can be gaps in the indices. It is intended to be more efficient than using a HashMap to map Integers to Objects.

Now

how i use SparseArray ?

You can do it by below ways:

  1. HashMap way:

    Map<Integer, Bitmap> _bitmapCache = new HashMap<Integer, Bitmap>();
    private void fillBitmapCache() {
         _bitmapCache.put(R.drawable.icon, BitmapFactory.decodeResource(getResources(), R.drawable.icon));
         _bitmapCache.put(R.drawable.abstrakt, BitmapFactory.decodeResource(getResources(), R.drawable.abstrakt));
         _bitmapCache.put(R.drawable.wallpaper, BitmapFactory.decodeResource(getResources(), R.drawable.wallpaper));
         _bitmapCache.put(R.drawable.scissors, BitmapFactory.decodeResource(getResources(), 
     }
    
    Bitmap bm = _bitmapCache.get(R.drawable.icon);
    
  2. SparseArray way:

    SparseArray<Bitmap> _bitmapCache = new SparseArray<Bitmap>();
    private void fillBitmapCache() {
         _bitmapCache.put(R.drawable.icon, BitmapFactory.decodeResource(getResources(), R.drawable.icon));
         _bitmapCache.put(R.drawable.abstrakt, BitmapFactory.decodeResource(getResources(), R.drawable.abstrakt));
         _bitmapCache.put(R.drawable.wallpaper, BitmapFactory.decodeResource(getResources(), R.drawable.wallpaper));
         _bitmapCache.put(R.drawable.scissors, BitmapFactory.decodeResource(getResources(), 
     }
    
    Bitmap bm = _bitmapCache.get(R.drawable.icon);
    

Hope it Will Help.

Sufian
  • 6,405
  • 16
  • 66
  • 120
Bhavesh Patadiya
  • 25,740
  • 15
  • 81
  • 107
  • Make certain to see the linked documentation which goes on to detail that the "implementation is not intended to be appropriate for data structures that may contain large numbers of items." A SparseArray is generally slower than a comparable HashMap. – Kevin Mark May 01 '16 at 03:54
  • I'd read the docs, but still had some questions and read a bit about Java maps as well. Somethings I found on HashMap: It provides constant-time performance for get/put, IF the hash function disperses "properly". Map permits a null key (table does not). On sparse arrays, docs say better performance with small sets of data. What size? (quote) "For containers holding up to hundreds of items, the performance difference is less than 50%" -- I take that to mean sparse begin losing out to map(/table?) when working with hundreds of key:values. In my case - I wanted sparse. – Magic Marbles Oct 12 '18 at 23:43
42

SparseArray is used when you are using an Integer as a key.

When using the SparseArray, the key will stay as a primitive variable at all times unlike when you use the HashMap where it is required to have a Object as a key which will cause the int to become an Integer object just for a short time while getting the object in the map.

By using the SparseArray you will save the Garbage Collector some work.

So use just like a Map<Integer,String>.

Nicklas Gnejs Eriksson
  • 3,395
  • 2
  • 21
  • 19
  • 5
    +1, avoids the autoboxing (and associated objects+GC) that happens with HashMap. @cyril-mottier wrote about it here: https://speakerdeck.com/cyrilmottier/optimizing-android-ui-pro-tips-for-creating-smooth-and-responsive-apps – orip Jun 26 '13 at 13:29
  • Technically you are **not** "using an `Integer` as a key". You are using an `int`. – Richard Le Mesurier Sep 09 '15 at 06:10
5

It's a hint that there is a better data structure for your code.

That hint is from Lint. You usually get it when you have a HashMap of integers to something else.

Its biggest advantage is to treat the integer key as a primitive. In other words, it won't covert to an Integer (the Java object) to use it as a key.

This is a big deal when using large maps. HashMap will result in the creation of many, many Integer objects in that case.

See a few more pieces of information here.

Sufian
  • 6,405
  • 16
  • 66
  • 120
Christian Garbin
  • 2,512
  • 1
  • 23
  • 31
  • https://developer.android.com/reference/android/util/SparseArray.html says: " The implementation is not intended to be appropriate for data structures that may contain large numbers of items. It is generally slower than a traditional HashMap, ..." - that means: if you are interested in speed and want to store many values you have to use HashMap. – The incredible Jan Jul 19 '17 at 12:21
0

SparseArray is better memory efficient Data Structure than HashMap for integer key value pairs. And it is specific to Android and I believe it can't be used for Java SE.

Sometimes, Linter warning shows that SparseArray must be used as in the above. This dumb warning might be right or wrong. You need to decide to use SparseArray rather HashMap if you find it suitable or you need to suppress it with @SuppressLint annotation.

To decide whether to implement, it is worthwhile doing a little research and evaluate your situation. You can find all some useful links here

https://gunhansancar.com/sparsearray-vs-hashmap/

https://developer.android.com/reference/android/util/SparseIntArray

Farruh Habibullaev
  • 2,342
  • 1
  • 26
  • 33