Skip to content

Rapidly fast implementation of SharedPreferences which stores each preference in files separately, performs disk IO via NIO with memory mapped byte buffers and works IPC (between processes).

License

Notifications You must be signed in to change notification settings

sboishtyan/binaryprefs

 
 

Repository files navigation

Build Status API Android Arsenal

Binary Preferences

Rapidly fast and lightweight re-implementation of SharedPreferences which stores each preference in files separately, performs disk operations via NIO with memory mapped byte buffers and works IPC (between processes). Written from scratch.

Api finalization status

Please note that api is not finalized yet and serialization contract or public api may be changed prior 1.0.0.

Advantages

  • Lightweight. Zero dependency
  • Super fast (faster than any other key/value solutions)
  • Small memory footprint while serialize/deserialize data
  • Zero copy in-memory cache
  • Persists only binary data. Not XML or JSON
  • Out of box data encryption support
  • Fully backward compatible with default SharedPreferences interface
  • Store all primitives include double, char, byte and short
  • Store complex data objects backward-compatible (see Persistable class documentation)
  • Fully optimized IPC support (preferences change listeners and in-memory cache works between processes)
  • Handle various exception events

Usage

Minimal working configuration

Preferences preferences = new BinaryPreferencesBuilder(context)
                .build();

Please, use only one instance of preferences by name, it saves you from non-reasoned allocations. You can store one instance of preferences in application class or event better use one instance from IoC like Dagger or some another DI framework.

All parameters are optional and chain-buildable.

Custom preferences name

Builder contains method which defines desirable preferences name:

Preferences preferences = new BinaryPreferencesBuilder(context)
                .name("user_data")
                .build();

Default is "default" name.

Encryption

You can define your own file vice versa encryption or use default:

Preferences preferences = new BinaryPreferencesBuilder(context)
                .encryption(new AesByteEncryptionImpl("16 bytes secret key".getBytes(), "16 bytes initial vector".getBytes()))
                .build();

Default is no-op encryption. Library also provides AesByteEncryptionImpl implementation.

Exception handler

You can listen exceptions which comes during disk IO, serialization, task execution operations:

Preferences preferences = new BinaryPreferencesBuilder(context)
                .exceptionHandler(new ExceptionHandler() {
                    @Override
                    public void handle(Exception e) {
                        //perform metrica report
                    }
                })
                .build();

Default is print handler which performs e.printStacktrace() if exception event are comes.

IPC mode

If your app architecture is process based (services works in separate processes) and you would like to get preferences updates with consistent cache state you can enable this feature:

Preferences preferences = new BinaryPreferencesBuilder(context)
                .supportInterProcess(true)
                .build();

Please, note that one key change delta should be less than 1 (one) megabyte because IPC data transferring is limited by this capacity. Details here: Documentation

Dealing with Persistable

Persistable contract been added for fast and flexible saving and it's restoring complex objects. It's pretty similar like standard java Externalizable contract but without few methods which don't need for. For usage you just need to implement this interface with methods in your data-model.

All Persistable data-objects should be registered by key for understanding de/serialization contract during cache initialization.

How to register Persistable

Preferences preferences = new BinaryPreferencesBuilder(context)
                .registerPersistable(TestUser.KEY, TestUser.class)
                .registerPersistable(TestOrder.KEY, TestOrder.class)
                .build();

Note about deepClone method: you should implement full object hierarchy copying for fast immutable in-memory data fetching.

Sample for explanation: TestUser.java

Logcat preferences dump

You can dump your preferences with adb console command right in logcat:

adb shell am broadcast -a com.ironz.binaryprefs.ACTION_DUMP_PREFERENCE --es "pref_name" "your_pref_name" (optional: --es "pref_key" "your_pref_key")

where:

your_pref_name - is your preferences name which is defined in register method. your_pref_key - is your preference key, this is optional value.

How to register preferences by name:

DumpReceiver.register(name, preferences);

Fully working example of all values dump:

adb shell am broadcast -a com.ironz.binaryprefs.ACTION_DUMP_PREFERENCE --es "pref_name" "user_data"

Example only for user_id key dump:

adb shell am broadcast -a com.ironz.binaryprefs.ACTION_DUMP_PREFERENCE --es "pref_name" "user_data" --es "pref_key" "user_id"

Please note that if you create multiple instances of one preferences (e.g. in Activity#onCreate) you should unregister dump (e.g. in Activity#onDestroy) like this:

DumpReceiver.unregister(name);

Roadmap

  1. Disk I/O encrypt. completed
  2. IPC completed
  3. Externalizable. completed as Persistable
  4. Preferences tooling (key set reading). completed: adb shell am broadcast -a com.ironz.binaryprefs.ACTION_DUMP_PREFERENCE --es "pref_name" "your_pref_name" (optional: --es "pref_key" "your_pref_key")
  5. Custom serializers. completed
  6. Synchronous commits. completed
  7. Store all primitives (like byte, short, char, double). completed
  8. Lock free (avoid locks). completed as LockFactory.
  9. Exact background tasks for each serialization strategies. completed
  10. Reduce events (implement events transaction). completed.
  11. Simplify api (instance creating, exception handles). completed
  12. Finalize serialization and persistence contract
  13. Background initializer
  14. byte[] support
  15. Default preferences migration mechanism
  16. IPC transactions without 1mb limit
  17. File name encrypt
  18. Persistable upgrade/downgrade api
  19. RxJava support
  20. sun.misc.Unsafe serialization mode for api 21+

License

Copyright 2017 Alexander Efremenkov

   Licensed under the Apache License, Version 2.0 (the "License");
   you may not use this file except in compliance with the License.
   You may obtain a copy of the License at

       http://www.apache.org/licenses/LICENSE-2.0

   Unless required by applicable law or agreed to in writing, software
   distributed under the License is distributed on an "AS IS" BASIS,
   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
   See the License for the specific language governing permissions and
   limitations under the License.

About

Rapidly fast implementation of SharedPreferences which stores each preference in files separately, performs disk IO via NIO with memory mapped byte buffers and works IPC (between processes).

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages

  • Java 100.0%