繁体   English   中英

如何使用 Flutter 和 Future Builder 从 Cloud Firestore 正确检索数据?

[英]How to properly retrieve data from Cloud Firestore using Flutter and Future Builder?

我是 Flutter 的新手,尤其是 Cloud Firestore。 我的数据存储为 Map,其中包含我收藏中的另一个 Map。 为了尽可能减少对我的数据库的查询,我会将数据存储在一个文档中,并且只想检索最新的文档。 关于获取数据的代码:

Future<Map<dynamic, dynamic>> fetchData() async {
  final _firestore = FirebaseFirestore.instance;
  late Map<dynamic, dynamic> data;

  try {
    await _firestore
        .collection('collection')
        .orderBy('timestamp', descending: true)
        .limit(1)
        .get()
        .then((value) => data = value.docs.first.data());
  } catch (e) {
    print(e);
  }
  return data;
}

首先,我想知道是否有更好的方法从我收藏的最新文档中获取存储的 map?

其次,我想等待它完成,然后使用 FutureBuilder 将我的 map 的每个元素显示为文本,例如。 在稍后阶段,我正在考虑使用一个在 initState 中调用的提供程序,它可以将检索到的数据发送到各种屏幕。 现在,我只是想显示数据,但似乎我误解了它是如何工作的:

class Home extends StatefulWidget {
  Home({Key? key}) : super(key: key);

  @override
  State<Home> createState() => _HomeState();
}

class _HomeState extends State<Home> {
  late Future<Map<dynamic, dynamic>> fixtures;

  @override
  void initState() {
    fixtures = _getData();
    super.initState();
  }

  Future<Map<dynamic, dynamic>> _getData() async {
    return await fetchData();
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
        body: Container(
      child: FutureBuilder(
          future: fixtures,
          builder: (BuildContext context,
              AsyncSnapshot<Map<dynamic, dynamic>> snapshot) {
            if (snapshot.connectionState == ConnectionState.waiting) {
              return Center(child: Text('loading...'));
            } else {
              if (snapshot.hasError) {
                return Center(child: Text('Error: ${snapshot.error}'));
              } else {
                return Center(child: Text('${snapshot.data!['XY']}'));
              }
            }
          }),
    ));
  }
}

它有点像这样工作,但我觉得它不正确。 我不太明白的是如何从快照中正确访问数据,如何正确存储所有数据,以及是否有更好的方法。 文档总是简单地说 snapshot.data 但实际上并没有使用检索到的数据来执行任何操作。

提前谢谢了。

对我来说看起来不错,如果可能有点令人费解。

我可能会将您的fetchData简化为:

Future<Map<dynamic, dynamic>> fetchData() async {
  var value = await FirebaseFirestore.instance
        .collection('collection')
        .orderBy('timestamp', descending: true)
        .limit(1)
        .get()
  return value.docs.first.data();
}

变化:

  • 删除了catch 由于您并没有真正处理错误,因此最好让它冒泡并仅在顶层捕获错误,然后您可以在其中集中记录它们或将它们发送到 Crashlytics。
  • 结合了awaitthen. 使用另一个。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM