flutter 项目中 数据的管理是个很重要的环节, 今天这篇博客主要就是讲讲 flutter 中数据的存储和状态管理的几种方式,以及优缺点.
下面说说各种存储方式的优缺点
shared_preferences 其实就是个 key-value 的存储方式, 可以做一些开关配置的存储,比如是否首次登陆这种.
存储简单,直接通过 put
和 get
key 值来存储和获取.
// 如果_sp已存在,直接返回,为null时创建
static Future get sp async {
if (_sp == null) {
_sp = await SharedPreferences.getInstance();
}
return _sp!;
}
static Future save(String key, String value) async {
Logger.i("save key = $key , value = $value");
return (await sp).setString(key, value);
}
static dynamic get(String key) async {
return (await sp).get(key);
}
一般都是一些配置的存储, 如果是对象或者数组的存储,就会很麻烦.
数据库存储, 一般都是存储对象和数组.
能够存储复杂的数据和数组
操作麻烦, 要先创建表,管理数据库的版本信息, 以及更新库表操作, 如果不熟练 很容易出错.
dbHelp
var databasesPath = await getDatabasesPath();
String path = join(databasesPath, '$dbName.db');
//根据数据库文件路径和数据库版本号创建数据库表
db = await openDatabase(path, version: dbVersion, onConfigure: onConfigure,
onCreate: (Database db, int version) async {
var batch = db.batch();
GlobalConfigDao.createGlobalConfigV1(batch);
await batch.commit();
Timer(Duration(seconds: 1), () {
GlobalConfigDao.insertGlobalConfig(
GlobalConfigData(fontFamily: 0, themeColor: 1));
});
}, onUpgrade: (db, oldVersion, newVersion) async {
// var batch = db.batch();
// switch (oldVersion) {
// case 1:
// WelcomeConfigDao.createWelcomeConfigV2(batch);
// break;
// }
// await batch.commit();
});
GlobalConfigDao
static void createGlobalConfigV1(Batch batch) {
batch.execute('DROP TABLE IF EXISTS globalConfig');
batch.execute('''CREATE TABLE globalConfig (
id INTEGER PRIMARY KEY AUTOINCREMENT,
themeColor INTEGER,
fontFamily INTEGER)''');
}
redux_persist
其实说到底也类似于 shared_preferences
, 他也是通过 key 来存储的, 只不过我们可以通过对象的 toJson 方法把对象转成 json,然后 redux_persist 把当前的 json 转成 string 存储到 key 里面去.
存储对象 readConfig
part 'readConfig.g.dart';
@JsonSerializable(explicitToJson: true)
class ReadConfig {
int textFontIndex;
int bgColorIndex;
ReadConfig({this.bgColorIndex: 2, this.textFontIndex: 2});
factory ReadConfig.fromJson(Map json) =>
_$ReadConfigFromJson(json);
Map toJson() => _$ReadConfigToJson(this);
// 命名构造函数
ReadConfig.empty();
}
readConfigRedux
class UpdateReaderConfigAction {
ReadConfig readConfig;
UpdateReaderConfigAction(this.readConfig);
}
ReadConfig _update(ReadConfig readConfig, action) {
readConfig = action.readConfig;
return readConfig;
}
final ReadConfigReducer = combineReducers([
TypedReducer(_update),
]);
state
class DzState {
ReadConfig readConfig;
DzState(
{
this.readConfig,});
static DzState fromJson(dynamic json) => json != null
? DzState(
readConfig: ReadConfig.fromJson(json["readConfig"] ??
ReadConfig(bgColorIndex: 2, textFontIndex: 2).toJson()))
: DzState.empty();
dynamic toJson() => {
"readConfig": readConfig,
};
static DzState empty() {
return DzState(
readConfig: ReadConfig(bgColorIndex: 2, textFontIndex: 2),
);
}
}
DzState appReducer(DzState state, action) {
return new DzState(
readConfig: ReadConfigReducer(state.readConfig, action));
}
在 main.dart 中调用
final persistor = Persistor(
storage: FlutterStorage(),
serializer: JsonSerializer(DzState.fromJson),
);
// Load initial state
final initialState = await persistor.load();
final store = Store(
appReducer,
initialState: initialState ?? DzState.empty(),
middleware: [persistor.createMiddleware()],
);
既可以存简单的 key-value 的值, 也可以存储对象和数组.
创建一个对象比较复杂, 很容易初始化出错, 而且没法给对象中的单独变量修改,要修改的时候 必须把当前对象获取到,然后修改变量的值, 然后把整个对象给修改掉.
ReadConfig config = DzStore.getStore().state.readConfig;
config.textFontIndex = value.toInt();
DzStore.getStore().dispatch(UpdateReaderConfigAction(config));
数据的状态管理 一般用这三个.下面分别说说三个的优缺点.
只要 redux 里面的数值改变, 引用的地方都会刷新,比如启动多个页面 其中一个页面头像和昵称修改,那么回退回去后,页面的头像和昵称也会发生修改.数据及时同步
只要用 StoreBuilder
的地方, 都会重新刷新,很容易造成没必要的性能浪费.
只要 provider 里面的数据发生改变, 引用当前的 provider 的页面才会重新刷新
如果当前 provider 有多个状态, 那么只要修改其中一个状态, 那页面也是都会刷新.也容易造成性能的浪费.
bloc 可以说是集成了 redux 和 provider 的所有优点, 并且优化了他们的缺点.
bloc 定义事件, 每个事件一个 state, 只要处理返回的对象 state 就行了,就减少了性能的浪费, 减少了没必要的刷新.
后面就在实战项目中来体现.