目录

变量

这是一个创建变量并对其进行初始化的示例:

dart
var name = 'Bob';

变量存储引用。名为 name 的变量包含对值为“Bob”的 String 对象的引用。

name 变量的类型被推断为 String ,但您可以通过指定它来更改该类型。如果对象不受限于单一类型,请指定 Object 类型(如果需要,则指定 dynamic )。

dart
Object name = 'Bob';

另一种选择是显式声明将被推断的类型:

dart
String name = 'Bob';

空安全

#

Dart 语言强制执行健全的空安全。

空安全可防止因无意访问设置为 null 的变量而导致的错误。此错误称为空解引用错误。当您访问表达式(其计算结果为 null )的属性或调用其方法时,就会发生空解引用错误。此规则的例外情况是当 null 支持属性或方法时,例如 toString()hashCode 。使用空安全,Dart 编译器会在编译时检测这些潜在错误。

例如,假设您想找到 int 变量 i 的绝对值。如果 inull ,则调用 i.abs() 会导致空解引用错误。在其他语言中,尝试此操作可能会导致运行时错误,但 Dart 的编译器会禁止这些操作。因此,Dart 应用不会导致运行时错误。

空安全引入了三个主要更改:

  1. 当您为变量、参数或其他相关组件指定类型时,您可以控制该类型是否允许 null 。要启用可空性,请在类型声明的末尾添加 ?

    dart
    String? name  // 可空类型。可以是 `null` 或字符串。
    
    String name   // 不可空类型。不能是 `null` ,但可以是字符串。
  2. 您必须在使用变量之前对其进行初始化。可空变量默认为 null ,因此它们默认情况下已初始化。Dart 不会为不可空类型设置初始值。它强制您设置初始值。Dart 不允许您观察未初始化的变量。这可以防止您访问属性或调用方法,其中接收方的类型可以是 null ,但 null 不支持使用的方法或属性。

  3. 您不能访问具有可空类型的表达式的属性或调用其方法。相同的异常适用于 null 支持的属性或方法,例如 hashCodetoString()

健全的空安全将潜在的 运行时错误 转换为 编辑时 分析错误。当非空变量已被:

  • 未初始化为非空值。
  • 分配了 null 值。

此检查允许您在部署应用 之前 纠正这些错误。

默认值

#

具有可空类型的未初始化变量的初始值为 null 。即使是具有数字类型的变量最初也是 null,因为数字(就像 Dart 中的所有其他内容一样)都是对象。

dart
int? lineCount;
assert(lineCount == null);

使用空安全,您必须在使用不可空变量之前对其进行初始化:

dart
int lineCount = 0;

您不必在声明局部变量的地方对其进行初始化,但您确实需要在使用它之前为其赋值。例如,以下代码是有效的,因为 Dart 可以检测到在将 lineCount 传递给 print() 之前它是非空的:

dart
int lineCount;

if (weLikeToCount) {
  lineCount = countLines();
} else {
  lineCount = 0;
}

print(lineCount);

顶级变量和类变量是延迟初始化的;初始化代码在第一次使用变量时运行。

延迟变量

#

late 修饰符有两种用例:

  • 声明一个在声明后初始化的不可空变量。
  • 延迟初始化变量。

通常,Dart 的控制流分析可以检测到在使用不可空变量之前何时将其设置为非空值,但有时分析会失败。两种常见情况是顶级变量和实例变量:Dart 通常无法确定它们是否已设置,因此它不会尝试。

如果您确定在使用变量之前已设置该变量,但 Dart 不同意,则可以通过将变量标记为 late 来纠正此错误:

dart
late String description;

void main() {
  description = 'Feijoada!';
  print(description);
}

当您将变量标记为 late 但在其声明处对其进行初始化时,则在第一次使用该变量时运行初始化程序。这种延迟初始化在以下几种情况下非常方便:

  • 可能不需要该变量,并且初始化该变量的成本很高。
  • 您正在初始化实例变量,并且其初始化程序需要访问 this

在以下示例中,如果从未使用 temperature 变量,则从未调用代价高昂的 readThermometer() 函数:

dart
// 这是程序对 readThermometer() 的唯一调用。
late String temperature = readThermometer(); // 延迟初始化。

final 和 const

#

如果您不打算更改变量,请使用 finalconst ,替换 var 或添加到类型中。 final 变量只能设置一次; const 变量是编译时常量。( const 变量隐式为 final 。)

以下是创建和设置 final 变量的示例:

dart
final name = 'Bob'; // 没有类型注解
final String nickname = 'Bobby';

您不能更改 final 变量的值:

✗ static analysis: failuredart
name = 'Alice'; // 错误:final 变量只能设置一次。

对于希望成为 编译时常量 的变量,请使用 const 。如果 const 变量位于类级别,请将其标记为 static const 。在声明变量的地方,将值设置为编译时常量,例如数字或字符串文字、 const 变量或对常量数字进行算术运算的结果:

dart
const bar = 1000000; // 压力单位 (dynes/cm2)
const double atm = 1.01325 * bar; // 标准大气压

const 关键字不仅仅用于声明常量变量。您还可以使用它来创建常量 ,以及声明创建常量值的构造函数。任何变量都可以具有常量值。

dart
var foo = const [];
final bar = const [];
const baz = []; // 等同于 `const []`

您可以从 const 声明的初始化表达式中省略 const ,如上面的 baz 一样。有关详细信息,请参阅 不要冗余地使用 const

您可以更改非 final、非 const 变量的值,即使它以前具有 const 值:

dart
foo = [1, 2, 3]; // 以前是 const []

您不能更改 const 变量的值:

✗ static analysis: failuredart
baz = [42]; // 错误:常量变量不能赋值。

您可以定义使用 类型检查和强制转换isas )、 集合 if扩展运算符......? )的常量:

dart
const Object i = 3; // i 是具有 int 值的 const Object ...
const list = [i as int]; // 使用类型转换。
const map = {if (i is int) i: 'int'}; // 使用 is 和集合 if。
const set = {if (list is List<int>) ...list}; // ...和扩展运算符。

有关使用 const 创建常量值的更多信息,请参阅 列表映射