關(guān)于Flutter 使用GetStorage本地緩存的使用和需要注意的問題

使用

使用相對來說比較簡單
存儲:

GetStorage().write('your_key', your_value);

讀?。?/p>

GetStorage().read('your_key');
問題

存儲:
存儲的時候 我們持久化的值 若是 系統(tǒng)的基礎(chǔ)數(shù)據(jù)類型即 bool、String、int或是List、Map這些集合類型,且集合中也都是系統(tǒng)提供的數(shù)據(jù)類型,完全沒有問題,
但是有時候 我們灰直接存儲我們自定義的模型數(shù)據(jù),比如

HomeTabItemModel(
    this.title,
    this.image,
  );
GetStorage().write('home_tab', itemModel);

這種也是可以直接存儲的 但是 需要注意 itemModel 需要實現(xiàn)toJson() 方法,否則存儲失敗 或是 下次獲取不到
原因是持久化時會調(diào)用json.encode()方法,該方法會調(diào)用非系統(tǒng)類的toJson()方法
這里需要注意 write()方法 寫入分為兩部分 一部分是內(nèi)存中的數(shù)據(jù)(寫入自定義模型,內(nèi)存中就是我們自定義的模型);另一部分是磁盤中的數(shù)據(jù),該數(shù)據(jù)是json序列化后的數(shù)據(jù) 與 自定義模型的toJson()對應(yīng),所以此時 我們磁盤和內(nèi)存中的數(shù)據(jù)類型是不同的

///存儲
Future<void> write(String key, dynamic value) async {
    // 內(nèi)存
    writeInMemory(key, value);
    // final _encoded = json.encode(value);
    // await _concrete.write(key, json.decode(_encoded));
    // 磁盤
    return  _tryFlush();
  }
void writeInMemory(String key, dynamic value) {
    _concrete.write(key, value);
  }

內(nèi)存

void write(String key, dynamic value) {
    subject
      ..value[key] = value
      ..changeValue(key, value);
    //return _writeToStorage(subject.value);
  }

磁盤

Future<void> _writeToStorage(Map<String, dynamic> data) async {
    localStorage.update(fileName, (val) => json.encode(subject.value),
        ifAbsent: () => json.encode(subject.value));
  }
/// If [toEncodable] is omitted, it defaults to a function that returns the
  /// result of calling `.toJson()` on the unencodable object.
  String encode(Object? value, {Object? toEncodable(dynamic object)?}) {
    toEncodable ??= _toEncodable;
    if (toEncodable == null) return encoder.convert(value);
    return JsonEncoder(toEncodable).convert(value);
  }

讀?。?/p>

讀取分類兩種情況
1、系統(tǒng)類型的讀取,這種不管什么時間讀取 讀取到的都是一樣的都是我們存儲的數(shù)據(jù)類型

GetStorage().read<bool>('your_key');
GetStorage().read<String>('your_key');
...

2、自定義類型存儲后的讀取
這種形式的讀取分為兩種情況
1》存儲后 待下次冷啟動才會調(diào)用read方法進行讀取,這種讀取到的是已經(jīng)經(jīng)過json序列化后的Map或是List數(shù)據(jù),原因在存儲的地方已經(jīng)說明

Iterable? json = GetStorage().read('your_key');
    if (json != null) {
      List<ItemModel> list = List<ItemModel>.from(json.map((e) => ItemModel.fromJson(e)));
      if (list.isNotEmpty) {
        tabs(list);
      }
    }

2》存儲后 還會在某個時刻讀取,這種情況讀取到的直接是我們自定的類型,此時讀取到的是內(nèi)存中經(jīng)過存儲更新后的數(shù)據(jù),這種情況就會導(dǎo)致讀取到的數(shù)據(jù)類型與磁盤中不同,最好讀取方法這么寫

void readFromCache() {
    Map<String, dynamic> map = Map.from(GetStorage().read("your_key") ?? {});
    List<ItemModel> models = [];
    for (var item in map.entries) {
      ItemModel itemModel;
      if (item.value is Map) {
        itemModel = ItemModel.fromJson(item.value);
      } else {
        itemModel = item.value;
      }
      models.add(itemModel);
    }
  }
總結(jié)

根據(jù)上述問題 建議在存儲時盡量不使用我們自定義的數(shù)據(jù)類型直接存儲,存儲前要對數(shù)據(jù)進行序列化,存儲系統(tǒng)提供的數(shù)據(jù)類型,這樣可保證 內(nèi)存和 磁盤中的數(shù)據(jù)一致,獲取時不容易出錯

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

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

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