Flutter之shared_preferences的使用、源碼分析(二)

shared_preferences源碼分析

實(shí)例化對(duì)象源碼分析
接下來(lái)我們來(lái)對(duì)shared_preferences進(jìn)行分析,我們?cè)谑褂玫臅r(shí)候需要通過(guò)getInstance實(shí)例化一個(gè)對(duì)象,接下來(lái)我們看下這里面它都做了什么操作。

靜態(tài)變量分析
我們先來(lái)看下它定義了三個(gè)靜態(tài)變量:

_prefix: 設(shè)置持久化數(shù)據(jù)和讀取持久化數(shù)據(jù)時(shí)統(tǒng)一設(shè)置前綴(flutter.)
_completer: 持久化數(shù)據(jù)異步通知,就是當(dāng)shared_preferences實(shí)例化完成后通過(guò)completer.future來(lái)返回結(jié)果
_manualDartRegistrationNeeded: 是否需要手動(dòng)注冊(cè),因?yàn)樯婕暗?code>Linux、WindowsMac Os的持久化數(shù)據(jù)時(shí),是需要手動(dòng)進(jìn)行注冊(cè)的,默認(rèn)為true

static const String _prefix = 'flutter.';
static Completer<SharedPreferences>? _completer;
static bool _manualDartRegistrationNeeded = true;

getInstance()源碼分析

當(dāng)我們獲取實(shí)例化對(duì)象時(shí)先判斷_completer是否為空,如果不為空則直接返回它的的future結(jié)果,否則它會(huì)實(shí)例化一個(gè)SharedPreferencesCompleter對(duì)象,然后通過(guò)_getSharedPreferencesMap來(lái)獲取持久化的map對(duì)象,獲取到map對(duì)象后,通過(guò)completer.complete(SharedPreferences._(preferencesMap))Map結(jié)果返回出去,代碼如下:

static Future<SharedPreferences> getInstance() async {
  if (_completer == null) {
    final completer = Completer<SharedPreferences>();
    try {
      final Map<String, Object> preferencesMap =
        await _getSharedPreferencesMap();
      completer.complete(SharedPreferences._(preferencesMap));
    } on Exception catch (e) {
      // If there's an error, explicitly return the future with an error.
      // then set the completer to null so we can retry.
      completer.completeError(e);
      final Future<SharedPreferences> sharedPrefsFuture = completer.future;
      _completer = null;
      return sharedPrefsFuture;
    }
    _completer = completer;
  }
  return _completer!.future;
}

_getSharedPreferencesMap()源碼分析

在我們調(diào)用getInstance()方法里,會(huì)調(diào)用_getSharedPreferencesMap()來(lái)獲取持久化的Map數(shù)據(jù),我們接下來(lái)看看它是如何獲取的,首先它通過(guò)_store.getAll()就可以直接獲取到本地的所有持久化數(shù)據(jù),當(dāng)我們調(diào)用_store時(shí),它會(huì)判斷是否需要手動(dòng)注冊(cè)

不需要手動(dòng)注冊(cè)時(shí):
iOS、Android等平臺(tái)中使用不需要手動(dòng)注冊(cè),所以它直接就返回的對(duì)應(yīng)的實(shí)例對(duì)象

需要手動(dòng)注冊(cè)時(shí):
先判斷是否是web,如果是就返回localStorage,否則判斷是Linux還是Windows,然后根據(jù)平臺(tái)的不同返回其對(duì)應(yīng)的實(shí)例。

static Future<Map<String, Object>> _getSharedPreferencesMap() async {
  final Map<String, Object> fromSystem = await _store.getAll();
  assert(fromSystem != null);
  // Strip the flutter. prefix from the returned preferences.
  final Map<String, Object> preferencesMap = <String, Object>{};
  for (String key in fromSystem.keys) {
    assert(key.startsWith(_prefix));
    preferencesMap[key.substring(_prefix.length)] = fromSystem[key]!;
  }
  return preferencesMap;
}

static SharedPreferencesStorePlatform get _store {
  // TODO(egarciad): Remove once auto registration lands on Flutter stable.
  // https://github.com/flutter/flutter/issues/81421.
  if (_manualDartRegistrationNeeded) {
    // Only do the initial registration if it hasn't already been overridden
    // with a non-default instance.
    if (!kIsWeb &&
        SharedPreferencesStorePlatform.instance
        is MethodChannelSharedPreferencesStore) {
      if (Platform.isLinux) {
        SharedPreferencesStorePlatform.instance = SharedPreferencesLinux();
      } else if (Platform.isWindows) {
        SharedPreferencesStorePlatform.instance = SharedPreferencesWindows();
      }
    }
    _manualDartRegistrationNeeded = false;
  }

  return SharedPreferencesStorePlatform.instance;
}

_setValue()源碼分析

不管我們是存儲(chǔ)什么內(nèi)容的數(shù)據(jù),最終都會(huì)調(diào)用_setValue()來(lái)進(jìn)行存儲(chǔ),

首先它會(huì)檢查存入的value是否為空,如果為空就拋出異常,否則就用_prefix + key來(lái)作為存入的key值。

判斷存入的值是不是List<String>,如果是先把value通過(guò)toList()方法轉(zhuǎn)換,然后在存入,否則直接存入,這步存入操作只是存入緩存中,當(dāng)應(yīng)用程序退出時(shí)將消失
最后通過(guò)_store來(lái)異步寫(xiě)入到磁盤(pán)中

Future<bool> _setValue(String valueType, String key, Object value) {
  ArgumentError.checkNotNull(value, 'value');
  final String prefixedKey = '$_prefix$key';
  if (value is List<String>) {
    // Make a copy of the list so that later mutations won't propagate
    _preferenceCache[key] = value.toList();
  } else {
    _preferenceCache[key] = value;
  }
  return _store.setValue(valueType, prefixedKey, value);
}

本篇主要講了shared_preferences的源碼,下篇講講項(xiàng)目中使用shared_preferences的封裝部分。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容