实例成员'userEmail'在初始化器中无法访问

时间:2020-09-18 03:26:51

标签: flutter dart instance initializer

以上错误消息正在我的代码中显示。请看下面:

class BottomNaviBar extends StatefulWidget {
  static const String id = 'bottom_navi_bar';

  @override
  _BottomNaviBarState createState() => _BottomNaviBarState();
}

class _BottomNaviBarState extends State<BottomNaviBar> {
  int selectedIndex = 0;
  bool showRecipeNotificationBadge = false;
  bool showProfileNotificationBadge = false;
  final _auth = FirebaseAuth.instance;
  FirebaseUser loggedInUser;
  String userEmail;

  @override
  void initState() {
    super.initState();
    getCurrentUser();
  }

  void getCurrentUser() async {
    try {
      final user = await _auth.currentUser();
      if (user != null) {
        loggedInUser = user;
          userEmail = user.email;
      }
    } catch (e) {
      print(e);
    }
  }


  List<Widget> _widgetOptions = <Widget>[
    RecipeBlog(),
    FavouritesScreen(userEmail: userEmail,), //this is what is causing the error
    ProperHomeScreen(),
    ProfileScreen(),
  ];

  ...

这很奇怪,因为如果我将实际用户的电子邮件地址作为硬编码字符串传递到FavouritesScreen()中,则代码可以正常工作。但这不能正常工作。

有什么想法吗?

于20/10/20更新:

有关更多上下文,这是build的代码:

  @override
  Widget build(BuildContext context) {

    List<Widget> _widgetOptions() {
      return [
        RecipeBlog(),
        FavouritesScreen(userEmail: userEmail,),
        ProperHomeScreen(),
        ProfileScreen(),
      ];
    }

    return Scaffold(
      body: Center(
        child: _widgetOptions.elementAt(selectedIndex),
      ),
      bottomNavigationBar: Theme(
        data: Theme.of(context).copyWith(
          // sets the background color of the `BottomNavigationBar`
          canvasColor: Color(0xFF150A42),
          // sets the active color of the `BottomNavigationBar` if `Brightness` is light
        ),
        child: BottomNavigationBar(
          items: <BottomNavigationBarItem>[
            BottomNavigationBarItem(
              icon: Icon(Icons.search),
              title: Container(),
            ),

            BottomNavigationBarItem(
              icon: Icon(Icons.favorite, color: Colors.redAccent,),
              title: Container(),
            ),
            BottomNavigationBarItem(
              icon: Stack(children: <Widget>[
                Icon(Icons.forum),
                showRecipeNotificationBadge != false
                    ? Positioned(
                  top: 0.0,
                  right: 0.0,
                  child: Icon(Icons.brightness_1,
                      size: 12.0, color: Color(0x00000000)),
                )
                    : Positioned(
                  top: 0.0,
                  right: 0.0,
                  child: Icon(Icons.brightness_1,
                      size: 12.0, color: Colors.redAccent),
                ),
              ]),
              title: Container(),
            ),
            BottomNavigationBarItem(
              icon: Stack(children: <Widget>[
                Icon(Icons.person),
                showProfileNotificationBadge != false
                    ? Positioned(
                  top: 0.0,
                  right: 0.0,
                  child: Icon(Icons.brightness_1,
                      size: 12.0, color: Color(0x00000000)),
                )
                    : Positioned(
                  top: 0.0,
                  right: 0.0,
                  child: Icon(Icons.brightness_1,
                      size: 12.0, color: Colors.redAccent),
                ),
              ]),
              title: Container(),
            ),
          ],
          currentIndex: selectedIndex,
          selectedItemColor: Color(0xFF150A42),
          backgroundColor: Colors.white,
          unselectedItemColor: Colors.black38,
          onTap: _onItemTapped,
          type: BottomNavigationBarType.fixed,
        ),
      ),
      floatingActionButton: selectedIndex == 0 ? null : FloatingActionButton(
        backgroundColor: Color(0xff02aab0),
        onPressed: () {Navigator.push(
          context,
          MaterialPageRoute(builder: (context) => BottomNaviBar()),
        );},
        child: Icon(Icons.search),
      ),
    );
  }
}

1 个答案:

答案 0 :(得分:3)

该错误指出您不能在初始化程序中使用实例成员。这意味着您不能使用另一个实例变量来初始化实例变量。

一个简单的例子:

class Test {
  String initial = "this thing";
  String other = initial;//This leads to the same error that you have
}

这是因为初始化程序在构造函数之前执行。这意味着在初始化时无法访问this,而在访问实例成员时会隐式调用。在对值进行硬编码时,this不再需要初始化变量,因此它可以工作。

在前面的示例中,调用initial实际上是在进行this.initial,但是this不可用,因此将不起作用。

一种解决方法是将_widgetOptions的初始化更改为initState


但是,首先建议不要尝试执行问题中显示的内容。除可能非常特殊的情况外,Widget不应存储在变量中。这是因为,当您更新要传递给窗口小部件的参数时,窗口小部件对象本身将不会更新,因此在重建时不会看到任何可视更改。

请改为使_widgetOptions为您在build中调用的函数,该函数返回List中的Widget个,以便每次创建新Widget对象时,函数称为:

List<Widget> _widgetOptions() {
  return [
    RecipeBlog(),
    FavouritesScreen(userEmail: userEmail,),
    ProperHomeScreen(),
    ProfileScreen(),
  ];
}

此方法还从根本上消除了您收到的当前错误,无需进行任何额外的工作,因为在构造函数创建this之前无法调用该方法。