【问题标题】:Can you call Firestore.instance multiple times in Flutter?你可以在 Flutter 中多次调用 Firestore.instance 吗?
【发布时间】:2020-08-26 05:13:38
【问题描述】:

使用 Flutter 应用和 Cloud Firestore 插件。似乎您可以随时调用 Firestore.instance ,但是从插件文档中查看示例应用程序,它们的设置与自述文件不同。他们在主文件中初始化它,然后像这样引用主文件:

final FirebaseApp app = await FirebaseApp.configure(
    name: 'test',
    options: const FirebaseOptions(
      googleAppID: '1:79601577497:ios:5f2bcc6ba8cecddd',
      gcmSenderID: '79601577497',
      apiKey: 'AIzaSyArgmRGfB5kiQT6CunAOmKRVKEsxKmy6YI-G72PVU',
      projectID: 'flutter-firestore',
    ),
  );
  final Firestore firestore = Firestore(app: app);

  runApp(MaterialApp(
      title: 'Firestore Example', home: MyHomePage(firestore: firestore)));
}

这是必需的吗?我在流中遇到一个奇怪的错误,如果我创建一个流并发送任何更新流的数据,然后退出应用程序并重新启动它。当我再次尝试将数据推送到该流时,应用程序崩溃。我正在开发一个聊天功能,所以我的聊天页面会像这样流式传输所有消息:

@override
  Widget build(BuildContext context) {
    DMProfile dmProfile = Provider.of<DMProfile>(context);
    PlayerProfile playerProfile = Provider.of<PlayerProfile>(context);
    return StreamBuilder<List<Message>>(
      stream: Firestore.instance
          .collection('chats')
          .document(widget.chatId)
          .collection('messages')
          .orderBy('timestamp', descending: true)
          .limit(20)
          .snapshots()
          .map((list) =>
              list.documents.map((doc) => Message.fromFirestore(doc)).toList()),
      builder: (context, snapshot) {
        if (!snapshot.hasData) {
          return LoadingPage();
        }
        List<Message> messageStream = snapshot.data;
        if (messages.length == 0) {
          messages = messageStream;
          hasMore = !(messageStream.length < 20);
        } else if (messages.first.timestamp != null &&
            messageStream.first.timestamp != null) {
          if (messageStream.first.timestamp
                  .compareTo(messages.first.timestamp) !=
              0) {
            print('Not equal');
            messages.insert(0, messageStream.first);
          }
        }
        return Expanded(
          child: ListView.builder(
            reverse: true,
            padding: EdgeInsets.symmetric(horizontal: 10.0, vertical: 20.0),
            controller: listScrollController,
            itemCount: messages.length,
            itemBuilder: (context, index) {
              final messageText = messages[index].message;
              final messageSender = messages[index].senderId;

              final currentUser =
                  widget.isPlayer ? playerProfile.userId : dmProfile.userId;

              return MessageBubble(
                imageUrl: messageSender == playerProfile.userId
                    ? playerProfile.imageUrl
                    : dmProfile.imageUrl,
                text: messageText,
                isMe: currentUser == messageSender,
              );
            },
          ),
        );
      },
    );
  }

我这样发送消息:

sendMessage({
    String sender,
    String receiver,
    String message,
    bool isPlayer,
    String dmName,
    String playerName,
    String dmImage,
    String playerImage,
  }) async {
    String sender1 = sender.compareTo(receiver) < 0 ? sender : receiver;
    String sender2 = sender.compareTo(receiver) > 0 ? sender : receiver;
    String chatName = '${sender1}_$sender2';
    await Firestore.instance
        .collection(CHAT_COLLECTION)
        .document(chatName)
        .setData({
      'dm': isPlayer ? receiver : sender,
      'dm_image': dmImage,
      'dm_name': dmName,
      'player': isPlayer ? sender : receiver,
      'player_image': playerImage,
      'player_name': playerName,
      'dm_hidden': false,
      'player_hidden': false,
      'timestamp': FieldValue.serverTimestamp(),
      'dm_seen': isPlayer ? false : true,
      'player_seen': isPlayer ? true : false,
      'last_message': {
        'message': message,
        'sender_name': isPlayer ? playerName : dmName,
        'sender_id': sender,
        'receiver_id': receiver,
      },
    }, merge: true);
    await Firestore.instance
        .collection(CHAT_COLLECTION)
        .document(chatName)
        .collection(MESSAGES_COLLECTION)
        .add({
      'sender_id': sender,
      'receiver_id': receiver,
      'sender_name': isPlayer ? playerName : dmName,
      'message': message,
      'timestamp': FieldValue.serverTimestamp(),
    });
  }

但如果我执行这些步骤:

  1. 启动应用程序并转到聊天屏幕(已启动流)
  2. 在 Android 上使用返回按钮退出应用
  3. 重新打开应用程序并转到聊天屏幕(再次启动流)
  4. 发送消息 我收到以下错误并且应用程序崩溃:

    E/AndroidRuntime(21665):致命异常:主要 E/AndroidRuntime(21665):进程:com.crimsonowl.scryer,PID:21665 E/AndroidRuntime(21665): java.lang.NullPointerException: 尝试在空对象引用上调用虚拟方法 'void io.flutter.plugin.common.MethodChannel.invokeMethod(java.lang.String, java.lang.Object)' E/AndroidRuntime(21665):在 io.flutter.plugins.firebase.cloudfirestore.CloudFirestorePlugin$EventObserver.onEvent(CloudFirestorePlugin.java:451) E/AndroidRuntime(21665):在 io.flutter.plugins.firebase.cloudfirestore.CloudFirestorePlugin$EventObserver.onEvent(CloudFirestorePlugin.java:433) E/AndroidRuntime(21665):在 com.google.firebase.firestore.Query.lambda$addSnapshotListenerInternal$2(com.google.firebase:firebase-firestore@@21.3.0:1025) E/AndroidRuntime(21665):在 com.google.firebase.firestore.Query$$Lambda$3.onEvent(未知来源:6) E/AndroidRuntime(21665):在 com.google.firebase.firestore.core.AsyncEventListener.lambda$onEvent$0(com.google.firebase:firebase-firestore@@21.3.0:42) E/AndroidRuntime(21665):在 com.google.firebase.firestore.core.AsyncEventListener$$Lambda$1.run(未知来源:6) E/AndroidRuntime(21665): 在 android.os.Handler.handleCallback(Handler.java:883) E/AndroidRuntime(21665):在 android.os.Handler.dispatchMessage(Handler.java:100) E/AndroidRuntime(21665):在 com.google.android.gms.internal.tasks.zzb.dispatchMessage(com.google.android.gms:play-services-tasks@@17.0.2:6) E/AndroidRuntime(21665): 在 android.os.Looper.loop(Looper.java:214) E/AndroidRuntime(21665): 在 android.app.ActivityThread.main(ActivityThread.java:7356) E/AndroidRuntime(21665): 在 java.lang.reflect.Method.invoke(Native Method) E/AndroidRuntime(21665):在 com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:492) E/AndroidRuntime(21665): 在 com.android.internal.os.ZygoteInit.main(ZygoteInit.java:930) I/Process (21665):发送信号。 PID:21665 SIG:9

我已经尝试了所有方法并查看了尽可能多的文档,但似乎无法找到解决方案。

【问题讨论】:

  • 这是一个已知问题,正在跟踪here

标签: flutter google-cloud-firestore


【解决方案1】:

在查看 cloud_firestore 的 github 页面并阅读其他一些已发布的问题后,这是新版本 cloud_firestore 的错误。现在降级为0.13.4+2,它解决了这个问题。需要密切关注未来的版本和错误修复,看看这个问题何时得到解决。 Flutter 官方包得到了很好的支持和频繁的更新

【讨论】:

    【解决方案2】:

    要真正回答标题,是的,您可以使用 Firestore.instance 多次。它实际上不是一个函数,因此您实际上并没有“调用”它。您正在从一个类中获取一个静态变量。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2022-12-10
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2018-07-16
      • 1970-01-01
      相关资源
      最近更新 更多