Android Jetpack DataStore

導(dǎo)語(yǔ)

Jetpack簡(jiǎn)介及其它組件文章
DataStore就是SharedPreferences(簡(jiǎn)稱(chēng)SP)的替代品旋廷,Google為什么要用DataStore來(lái)替代SP呢趋观,因?yàn)镾P存在著很多問(wèn)題懂扼,我之前在Android SharedPreferences轉(zhuǎn)為MMKV中有詳細(xì)說(shuō)明了SP的不足,但是當(dāng)時(shí)的有些觀點(diǎn)還有些淺薄恨课,所以使用了MMKV來(lái)替代SP险胰,現(xiàn)在我更推薦大家使用DataStore替代SP志衣,下面會(huì)詳細(xì)講出屯援。

主要內(nèi)容

  • DataStore的基本概念
  • DataStore猛们、SP、MMKV對(duì)比
  • DataStore的基本使用
  • DataStore的封裝

DataStore的基本概念

Jetpack DataStore 是一種數(shù)據(jù)存儲(chǔ)解決方案狞洋,允許您使用協(xié)議緩沖區(qū)存儲(chǔ)鍵值對(duì)或類(lèi)型化對(duì)象弯淘。DataStore 使用 Kotlin 協(xié)程和 Flow 以異步、一致的事務(wù)方式存儲(chǔ)數(shù)據(jù)徘铝。
所以如果想要使用DataStore耳胎,就必須使用Kotlin,因?yàn)镈ataStore用到了flow惕它,flow用到了協(xié)程怕午,協(xié)程是Kotlin的特性。
但是淹魄,如果將DataStore封裝起來(lái)郁惜,那么直接使用Java調(diào)用的話,也是可以正常使用的甲锡,所以大家也不用擔(dān)心兆蕉。
DataStore 提供兩種不同的實(shí)現(xiàn):Preferences DataStore 和 Proto DataStore。

  • Preferences DataStore 使用鍵存儲(chǔ)和訪問(wèn)數(shù)據(jù)缤沦。此實(shí)現(xiàn)不需要預(yù)定義的架構(gòu)虎韵,也不確保類(lèi)型安全。
  • Proto DataStore 將數(shù)據(jù)作為自定義數(shù)據(jù)類(lèi)型的實(shí)例進(jìn)行存儲(chǔ)缸废。此實(shí)現(xiàn)要求您使用協(xié)議緩沖區(qū)來(lái)定義架構(gòu)包蓝,但可以確保類(lèi)型安全。
    這里我們重點(diǎn)講Preferences DataStore企量,這足以夠大多數(shù)用戶使用测萎。

DataStore、SP届巩、MMKV對(duì)比

大量文章指出MMKV的性能是多么多么高硅瞧,但其實(shí)MMKV在存大數(shù)據(jù)的時(shí)候,因?yàn)镸MKV是自行管理一塊內(nèi)存恕汇,性能是反而更低的腕唧。
我們先來(lái)看一下三種方案進(jìn)行1000次Int值的連續(xù)寫(xiě)入耗時(shí):


連續(xù)寫(xiě)入1000次Int值耗時(shí)

這么一看MMKV快到離譜,但是SP是可以異步寫(xiě)入的瘾英,而DataStore是基于協(xié)程的枣接,我們關(guān)心的卡頓是主流程的流暢,所以我們只需要考慮主線程的耗時(shí)方咆,所以會(huì)變成這樣:


連續(xù)寫(xiě)入1000次Int值耗時(shí)(異步)

可以看到耗時(shí)大大縮減月腋,但是還是比MMKV慢很多蟀架,這是Int數(shù)據(jù)類(lèi)型的寫(xiě)入瓣赂,Int數(shù)據(jù)是很小的榆骚,我們換成長(zhǎng)字符串,再來(lái)看一下效果:
連續(xù)寫(xiě)入1000次長(zhǎng)字符串耗時(shí)(異步)

這時(shí)我們會(huì)發(fā)現(xiàn)煌集,MMKV反而成為那個(gè)最慢的妓肢,DataStore遙遙領(lǐng)先。

所以我們要注意的是主線程的耗時(shí)苫纤,而這個(gè)數(shù)據(jù)其實(shí)也并不重要碉钠,因?yàn)樵陧?xiàng)目中也很少有1000次大數(shù)據(jù)的寫(xiě)入,而1000次大數(shù)據(jù)耗時(shí)也不過(guò)不到1秒卷拘,所以選合適的更重要喊废。

DataStore的基本使用

創(chuàng)建 Preferences DataStore
    private val Context.dataStore: DataStore<Preferences> by preferencesDataStore(
        name = "settings"
    )

name參數(shù)是 Preferences DataStore 的名稱(chēng)。類(lèi)似SP中的name栗弟。

將內(nèi)容寫(xiě)入 Preferences DataStore
    val EXAMPLE_COUNTER = intPreferencesKey("example_counter")
    suspend fun putIntData() {
        context.dataStore.edit { settings ->
            val currentCounterValue = settings[EXAMPLE_COUNTER] ?: 0
            settings[EXAMPLE_COUNTER] = currentCounterValue + 1
        }
    }

我們通過(guò)intPreferencesKey來(lái)定義我們的Key的name以及存儲(chǔ)的數(shù)據(jù)類(lèi)型污筷,如果是存放String類(lèi)型就是stringPreferencesKey。
通過(guò)settings[EXAMPLE_COUNTER]是可以取到數(shù)據(jù)的乍赫,通過(guò)賦值可以用來(lái)存數(shù)據(jù)瓣蛀。

從 Preferences DataStore 讀取內(nèi)容
    val EXAMPLE_COUNTER = intPreferencesKey("example_counter")
    val exampleCounterFlow: Flow<Int> = context.dataStore.data
        .map { preferences ->
            // No type safety.
            preferences[EXAMPLE_COUNTER] ?: 0
        }

通過(guò)這種方式取到的是一個(gè)被Flow,通過(guò)exampleCounterFlow.first()就可以拿到真實(shí)的數(shù)據(jù)雷厂。

在同步代碼中使用 DataStore

DataStore 的主要優(yōu)勢(shì)之一是異步 API惋增,但可能不一定始終能將周?chē)拇a更改為異步代碼。
Kotlin 協(xié)程提供 runBlocking() 協(xié)程構(gòu)建器改鲫,以幫助消除同步與異步代碼之間的差異诈皿。您可以使用 runBlocking() 從 DataStore 同步讀取數(shù)據(jù)。

val exampleData = runBlocking { context.dataStore.data.first() }

對(duì)界面線程執(zhí)行同步 I/O 操作可能會(huì)導(dǎo)致 ANR 或界面卡頓钩杰。您可以通過(guò)從 DataStore 異步預(yù)加載數(shù)據(jù)來(lái)減少這些問(wèn)題:

override fun onCreate(savedInstanceState: Bundle?) {
    lifecycleScope.launch {
        context.dataStore.data.first()
        // You should also handle IOExceptions here.
    }
}

這樣纫塌,DataStore 可以異步讀取數(shù)據(jù)并將其緩存在內(nèi)存中。以后使用 runBlocking() 進(jìn)行同步讀取的速度可能會(huì)更快讲弄,或者如果初始讀取已經(jīng)完成措左,可能也可以完全避免磁盤(pán) I/O 操作。

DataStore的封裝

封裝

既然我們都使用Kotlin了避除,那就使用Kotlin的特性擴(kuò)展函數(shù)來(lái)實(shí)現(xiàn)吧怎披。

import androidx.datastore.core.DataStore
import androidx.datastore.preferences.core.*
import kotlinx.coroutines.flow.first
import kotlinx.coroutines.flow.map
import kotlinx.coroutines.runBlocking

/**
 * Created by 郭士超 on 2022/11/6 15:48
 * Describe:DataStoreExt.kt
 */

/**
 * 存放數(shù)據(jù)
 */
fun <T> DataStore<Preferences>.putData(key: String, value: T) {
    runBlocking {
        when(value) {
            is String -> {
                putString(key, value)
            }
            is Int -> {
                putInt(key, value)
            }
            is Long -> {
                putLong(key, value)
            }
            is Float -> {
                putFloat(key, value)
            }
            is Double -> {
                putDouble(key, value)
            }
            is Boolean -> {
                putBoolean(key, value)
            }
        }
    }
}

/**
 * 取出數(shù)據(jù)
 */
fun <T> DataStore<Preferences>.getData(key: String, defaultValue: T): T {
     val data = when(defaultValue) {
        is String -> {
            getString(key, defaultValue)
        }
        is Int -> {
            getInt(key, defaultValue)
        }
        is Long -> {
            getLong(key, defaultValue)
        }
        is Float -> {
            getFloat(key, defaultValue)
        }
        is Double -> {
            getDouble(key, defaultValue)
        }
        is Boolean -> {
            getBoolean(key, defaultValue)
        }
        else -> {
            throw IllegalArgumentException("This type cannot be saved to the Data Store")
        }
    }
    return data as T
}


/**
 * 清空數(shù)據(jù)
 */
fun DataStore<Preferences>.clear() = runBlocking { edit { it.clear() } }


/**
 * 存放String數(shù)據(jù)
 */
private suspend fun DataStore<Preferences>.putString(key: String, value: String) {
    edit {
        it[stringPreferencesKey(key)] = value
    }
}

/**
 * 存放Int數(shù)據(jù)
 */
private suspend fun DataStore<Preferences>.putInt(key: String, value: Int) {
    edit {
        it[intPreferencesKey(key)] = value
    }
}

/**
 * 存放Long數(shù)據(jù)
 */
private suspend fun DataStore<Preferences>.putLong(key: String, value: Long) {
    edit {
        it[longPreferencesKey(key)] = value
    }
}

/**
 * 存放Float數(shù)據(jù)
 */
private suspend fun DataStore<Preferences>.putFloat(key: String, value: Float) {
    edit {
        it[floatPreferencesKey(key)] = value
    }
}

/**
 * 存放Double數(shù)據(jù)
 */
private suspend fun DataStore<Preferences>.putDouble(key: String, value: Double) {
    edit {
        it[doublePreferencesKey(key)] = value
    }
}

/**
 * 存放Boolean數(shù)據(jù)
 */
private suspend fun DataStore<Preferences>.putBoolean(key: String, value: Boolean) {
    edit {
        it[booleanPreferencesKey(key)] = value
    }
}


/**
 * 取出String數(shù)據(jù)
 */
private fun DataStore<Preferences>.getString(key: String, default: String? = null): String = runBlocking {
    return@runBlocking data.map {
        it[stringPreferencesKey(key)] ?: default
    }.first()!!
}

/**
 * 取出Int數(shù)據(jù)
 */
private fun DataStore<Preferences>.getInt(key: String, default: Int = 0): Int = runBlocking {
    return@runBlocking data.map {
        it[intPreferencesKey(key)] ?: default
    }.first()
}

/**
 * 取出Long數(shù)據(jù)
 */
private fun DataStore<Preferences>.getLong(key: String, default: Long = 0): Long = runBlocking {
    return@runBlocking data.map {
        it[longPreferencesKey(key)] ?: default
    }.first()
}

/**
 * 取出Float數(shù)據(jù)
 */
private fun DataStore<Preferences>.getFloat(key: String, default: Float = 0.0f): Float = runBlocking {
    return@runBlocking data.map {
        it[floatPreferencesKey(key)] ?: default
    }.first()
}

/**
 * 取出Double數(shù)據(jù)
 */
private fun DataStore<Preferences>.getDouble(key: String, default: Double = 0.00): Double = runBlocking {
    return@runBlocking data.map {
        it[doublePreferencesKey(key)] ?: default
    }.first()
}

/**
 * 取出Boolean數(shù)據(jù)
 */
private fun DataStore<Preferences>.getBoolean(key: String, default: Boolean = false): Boolean = runBlocking {
    return@runBlocking data.map {
        it[booleanPreferencesKey(key)] ?: default
    }.first()
}
使用

封裝完成我們應(yīng)該如何使用呢?

//第一步先創(chuàng)建一個(gè)自己的Application
class MyApplication: Application() {

    companion object {
        lateinit var instance : MyApplication
    }

    override fun onCreate() {
        super.onCreate()
        instance = this
    }

}
import androidx.datastore.core.DataStore
import androidx.datastore.preferences.core.Preferences
import androidx.datastore.preferences.preferencesDataStore

/**
 * Created by 郭士超 on 2022/11/6 16:38
 * Describe:AppDataStore.kt
 */
object AppDataStore {

    // 創(chuàng)建DataStore
    private val MyApplication.appDataStore: DataStore<Preferences> by preferencesDataStore(
        name = "App"
    )

    // DataStore變量
    private val dataStore = MyApplication.instance.appDataStore

    private fun <T> putData(key: String, value: T) {
        dataStore.putData(key, value)
    }

    private fun <T> gutData(key: String, value: T): T {
        return dataStore.getData(key, value)
    }

    fun clear() {
        dataStore.clear()
    }

    private const val NUMBER = "number"
    fun putNumber(number: Int) {
        dataStore.putData(NUMBER, number)
    }
    fun getNumber(): Int {
        return dataStore.getData(NUMBER, 0)
    }

}

封裝好之后瓶摆,我們?cè)谑褂玫臅r(shí)候凉逛,不同的模塊使用不同的XxxDataStore,這樣可以解耦合群井,我們將put和get方法放到XxxDataStore中統(tǒng)一管理状飞,方便我們快速定位這個(gè)方法都在哪里調(diào)用,從而更快定位到問(wèn)題或者加快開(kāi)發(fā)速度。

更多內(nèi)容戳這里(整理好的各種文集)

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末诬辈,一起剝皮案震驚了整個(gè)濱河市酵使,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌焙糟,老刑警劉巖口渔,帶你破解...
    沈念sama閱讀 219,490評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異穿撮,居然都是意外死亡缺脉,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,581評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén)悦穿,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)攻礼,“玉大人,你說(shuō)我怎么就攤上這事栗柒∶鼗祝” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 165,830評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵傍衡,是天一觀的道長(zhǎng)深员。 經(jīng)常有香客問(wèn)我,道長(zhǎng)蛙埂,這世上最難降的妖魔是什么倦畅? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,957評(píng)論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮绣的,結(jié)果婚禮上叠赐,老公的妹妹穿的比我還像新娘。我一直安慰自己屡江,他們只是感情好芭概,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,974評(píng)論 6 393
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著惩嘉,像睡著了一般罢洲。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上文黎,一...
    開(kāi)封第一講書(shū)人閱讀 51,754評(píng)論 1 307
  • 那天惹苗,我揣著相機(jī)與錄音,去河邊找鬼耸峭。 笑死桩蓉,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的劳闹。 我是一名探鬼主播院究,決...
    沈念sama閱讀 40,464評(píng)論 3 420
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼洽瞬,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了业汰?” 一聲冷哼從身側(cè)響起片任,我...
    開(kāi)封第一講書(shū)人閱讀 39,357評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎蔬胯,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體位他,經(jīng)...
    沈念sama閱讀 45,847評(píng)論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡氛濒,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,995評(píng)論 3 338
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了鹅髓。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片舞竿。...
    茶點(diǎn)故事閱讀 40,137評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖窿冯,靈堂內(nèi)的尸體忽然破棺而出骗奖,到底是詐尸還是另有隱情,我是刑警寧澤醒串,帶...
    沈念sama閱讀 35,819評(píng)論 5 346
  • 正文 年R本政府宣布执桌,位于F島的核電站,受9級(jí)特大地震影響芜赌,放射性物質(zhì)發(fā)生泄漏仰挣。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,482評(píng)論 3 331
  • 文/蒙蒙 一缠沈、第九天 我趴在偏房一處隱蔽的房頂上張望膘壶。 院中可真熱鬧,春花似錦洲愤、人聲如沸颓芭。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,023評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)亡问。三九已至,卻和暖如春肛宋,著一層夾襖步出監(jiān)牢的瞬間玛界,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,149評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工悼吱, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留慎框,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,409評(píng)論 3 373
  • 正文 我出身青樓后添,卻偏偏與公主長(zhǎng)得像笨枯,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,086評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容