_MapStream <FirebaseUser,User>抛出异常,StreamProvider <User>监听了该异常,但未提供`catchError`

问题描述

找到解决方案。在下面检查答案

任何想法为什么会发生这种情况,导致该错误的原因以及如何解决

我正在YouTube上观看TheNetNin​​ja的教程,我发现人们有类似的问题,但不完全相同。

仅当使用电子邮件并通过登录时,突然关闭应用程序(使用Android Studio中的“停止”按钮)而不注销然后重新启动时,会引发异常。对于Google登录,它似乎工作正常。另外,如果我在Firebase控制台中手动创建一个新用户并首次登录,则不会发生这种情况。

例外:

════════ Exception caught by provider ══════════════════════════════════════════════════════════════
The following assertion was thrown:
An exception was throw by _MapStream<FirebaseUser,User> listened by

StreamProvider<User>,but no `catchError` was provided.

Exception:
RangeError (index): Invalid value: Only valid value is 0: 1

════════════════════════════════════════════════════════════════════════════════════════════════════

我在pubspec.yaml中使用firebase_auth: ^0.16.0,并且认为它可能与该版本有关。

登录后,我得到此输出

I/BiChannelGoogleApi(17821): [FirebaseAuth: ] getGoogleApiForMethod() returned Gms: com.google.firebase.auth.api.internal.zzaq@cd1fa83
D/FirebaseAuth(17821): Notifying id token listeners about user ( 1EMwk8483hQWnSQwJeQYm9AcpiD2 ).

该帐户似乎已通过验证并登录,但是随后屏幕没有变化,因此我认为它在验证过程中卡在了某处。

奇怪的是,如果我正常登录并例外,然后重新启动应用程序,它就可以正常工作。如果我继续重新启动它,它就可以正常工作,但是如果我突然停止它,我会再次遇到异常。

这是我的验证码

import 'package:firebase_auth/firebase_auth.dart';
import 'package:Flutter/material.dart';
import 'package:Flutter/services.dart';
import 'package:google_sign_in/google_sign_in.dart';
import 'package:common_ui_module/Utils.dart';
import 'package:user_module/Role.dart';
import 'package:user_module/User.dart';
import 'package:localization_module/AppLocalizations.dart';

class AuthService {
  final FirebaseAuth _auth = FirebaseAuth.instance;

    return user != null
        ? User(uid: user.uid)
        : null;
  }

  Stream<User> get user {
    // map FireBaseUsers to User objects
    return _auth.onAuthStateChanged.map(_userFromFireBaseUser);
  }


// sign in with email and pass
  Future<User> signInWithEmailAndPassword(String email,String pass) async {
    try {
      AuthResult result =
          await _auth.signInWithEmailAndPassword(email: email,password: pass);
      FirebaseUser user = result.user;
      assert(await user.getIdToken() != null);
      print(
          "Signed in: ${user.email},pass: $pass,phone: ${user.phoneNumber}");

      /// return the mapped user
      return _userFromFireBaseUser(user);
    } catch (e) {
      print(e.toString());
      return null;
    }
  }


// sign out
  Future signOut() async {
    try {
      return await _auth.signOut();
    } catch (e) {
      print(e.toString());
      return null;
    }
  }
}

这是在“登录”和主屏幕之间切换的包装器

class Wrapper extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    final User user = Provider.of<User>(context);

    if (user != null) {
      return RestaurantOrdeRSScreen();
    } else {
      return LoginScreen();
    }
  }
}

我已经将主应用程序包装在MultiProvider中:

Widget build(BuildContext context) {
    return MultiProvider(
      providers: [
        /// PROVIDER FOR AUTHENTICATION
        StreamProvider<User>.value(
          value: AuthService().user,),//...
       // some other providers
      ],child: MaterialApp(
        home: SplashScreen(),);
  }

解决方法

终于找到了解决方案!

对我来说,当我尝试操作displayName时就是这样的分裂。这就是为什么我使索引超出范围错误。

List<String> name = ["",""];
    if (user != null && user.displayName != null) {
      name = user.displayName.split(" ");
    }

    return user != null
        ? User(
            id: 0,token: user.uid,firstName: name[0],lastName: name[1],...

因此,如果遇到这种错误,请在尝试将FirebaseUser转换为自己的自定义User对象时,仔细检查User属性操作。