Dart 中成员变量的初始化方式有区别吗?

问题描述 投票:0回答:1

在 Dart 中,立即赋值与在 Java 中的构造函数中赋值有区别吗?

class Example {
    int x = 3;
}

class Example {
    int x;
    Example() {
        x = 3;
    }
}

我问这个问题是因为当我使用 Flutter 并尝试将使用 setState 的 Function 分配给变量时,前一种方法不可能,但后一种方法可以。

flutter dart constructor
1个回答
31
投票

在你的小事上,这并不重要。

一般来说,可以通过以下几种方式初始化实例变量:

内联(字段初始值设定项)

class Example1 {
  T x = value;
}

优点:

  • 直接、简洁。
  • 成员将在all构造函数中初始化。
  • 可用于初始化
    final
    或不可为 null 的成员。
  • 成员在调用基类构造函数之前初始化,这在基类构造函数调用被派生类重写的成员函数时很重要。

缺点:

初始化列表

class Example2 {
  T x;

  Example2() : x = value;
}

优点:

  • 可用于初始化
    final
    或不可为 null 的成员。
  • 成员在调用基类构造函数之前初始化,这在基类构造函数调用被派生类重写的成员函数时很重要。
  • 可以利用构造参数。
  • 初始化变量always指的是成员变量,而不是构造函数参数。

缺点:

  • 如果类有多个构造函数,则需要重复初始化,或者构造函数应重定向到公共构造函数。
  • 不能依赖于
    this
    ,因为初始化发生在
    this
    变得有效之前(即,不能依赖于其他实例成员)。
  • 只能初始化封闭类的成员。因为初始化列表是在调用基类构造函数之前执行的,所以它们无法设置基类成员。

构造函数体

class Example3 {
  T x;

  Example3() {
    x = value;
  } 
}

优点:

  • 可以利用构造参数。
  • 可用于执行更复杂的初始化,例如无法通过单个表达式初始化成员的情况。
  • 可以使用
    this
    (即可以使用其他实例成员)。
  • 可用于设置基类成员。

缺点:

  • 不能用于初始化非
    late
    final
    或不可为 null 的成员。
  • 如果类有多个构造函数,则需要重复初始化或者需要重构初始化代码(例如但不限于重定向到公共构造函数)。
  • 成员在调用基类构造函数之后初始化。
  • 如果构造函数有一个隐藏成员变量的参数,则很容易意外引用该参数而不是成员。 (有关详细信息,请参阅
  • https://github.com/dart-lang/linter/issues/2552。)
我可能忘记了一些要点,但我认为这应该涵盖主要内容。

首先发生直接内联初始化,然后是初始化列表,然后是构造函数体。另请参阅

在参数列表和初始化器列表中分配值之间的差异,这解释了为什么 this

 仅对对象初始化的后期阶段有效。

举个例子,成员初始化的位置很重要:

class Base { Base() { doSomething(); } void doSomething() {} } class DerivedEarly extends Base { int? x; DerivedEarly() : x = 42; @override void doSomething() => print(x); } class DerivedLate extends Base { int? x; DerivedLate() { x = 42; } @override void doSomething() => print(x); } void main() { DerivedEarly(); // Prints: 42 DerivedLate(); // Prints: null }
    
© www.soinside.com 2019 - 2024. All rights reserved.