普通视图

发现新文章,点击刷新页面。
今天 — 2026年2月3日首页

Dart ——??运算符

作者 Haha_bj
2026年2月3日 11:31

你想了解 Dart 中的 ?? 运算符,它是 Dart 里处理 ** 可选值(可空类型)** 的常用运算符,核心作用是「空值合并」,我们从基础用法、核心特点、拓展用法到实际场景,一步步讲清楚,让你快速掌握。

一、先明确 ?? 运算符的核心定义

?? 被称为 「空值合并运算符(Null Coalescing Operator)」,作用是:判断运算符左侧的表达式是否为 null

  • 如果左侧不为 null,直接返回左侧的值
  • 如果左侧为 null,返回右侧的备选值

简单理解:「选左边的,左边是空的就选右边的」,用于给可空变量设置默认值,避免空值引发错误。

二、基础用法(最常用场景)

1. 基本语法格式

// 语法:左侧表达式 ?? 右侧备选值
variable ?? defaultValue

2. 示例(基础场景)

void main() {
  // 场景 1:左侧不为 null,返回左侧值
  String? name1 = "张三";
  String result1 = name1 ?? "未知用户";
  print("result1: $result1"); // 输出:result1: 张三

  // 场景 2:左侧为 null,返回右侧默认值
  String? name2 = null;
  String result2 = name2 ?? "未知用户";
  print("result2: $result2"); // 输出:result2: 未知用户

  // 场景 3:结合数值类型使用
  int? age1 = 20;
  int ageResult1 = age1 ?? 18;
  print("ageResult1: $ageResult1"); // 输出:ageResult1: 20

  int? age2 = null;
  int ageResult2 = age2 ?? 18;
  print("ageResult2: $ageResult2"); // 输出:ageResult2: 18
}

3. 关键注意点

  • ?? 只判断「左侧是否为 null」,不判断空字符串、0 等「空值但非 null」的情况

    void main() {
      // 左侧是空字符串(不是 null),返回左侧空字符串,而非右侧默认值
      String? emptyStr = "";
      String result = emptyStr ?? "默认字符串";
      print("result: '$result'"); // 输出:result: ''
    }
    
  • 右侧的备选值只有在左侧为 null 时才会被执行(惰性求值),提升性能:

    void main() {
      String? name = "李四";
      // 因为左侧不为 null,右侧的 print 不会执行,也不会调用 getDefaultName()
      String result = name ?? (print("右侧执行了") + getDefaultName());
      print("result: $result");
    }
    
    String getDefaultName() => "未知用户";
    

    运行结果:仅输出 result: 李四,右侧的打印和函数调用都未执行。

三、拓展用法:和其他运算符结合

1. ??=:空值赋值运算符(给变量本身设置默认值)

??=?? 的衍生运算符,作用是:如果变量本身为 null,就给它赋值右侧的值;如果变量不为 null,则不做任何修改(相当于「给变量设置默认值,仅在变量为空时生效」)。

void main() {
  // 场景 1:变量为 null,赋值右侧值
  String? name1 = null;
  name1 ??= "王五";
  print("name1: $name1"); // 输出:name1: 王五

  // 场景 2:变量不为 null,不修改
  String? name2 = "赵六";
  name2 ??= "未知用户";
  print("name2: $name2"); // 输出:name2: 赵六
}

对比 ????=

  • name ?? "未知":返回一个值,不修改 name 本身
  • name ??= "未知":直接修改 name 本身(仅当 namenull 时)。

2. ?. + ??:安全访问 + 空值兜底(高频组合)

?. 是「安全访问运算符」,作用是:如果对象不为 null,就访问对象的属性 / 方法;如果对象为 null,则返回 null(避免空指针异常)。

两者结合,可实现「安全访问属性,若对象为空或属性为空,就返回默认值」,是业务开发中的高频组合。

// 定义一个用户类
class User {
  String? name;
  User(this.name);
}

void main() {
  // 场景 1:对象不为 null,属性也不为 null
  User? user1 = User("钱七");
  String result1 = user1?.name ?? "未知用户";
  print("result1: $result1"); // 输出:result1: 钱七

  // 场景 2:对象不为 null,但属性为 null
  User? user2 = User(null);
  String result2 = user2?.name ?? "未知用户";
  print("result2: $result2"); // 输出:result2: 未知用户

  // 场景 3:对象本身为 null
  User? user3 = null;
  String result3 = user3?.name ?? "未知用户";
  print("result3: $result3"); // 输出:result3: 未知用户
}

四、实际业务场景示例

1. 接口返回数据兜底(避免空值展示)

// 模拟接口返回的用户数据(可能为 null)
Map<String, dynamic>? apiResponse = {
  "username": "小明",
  "age": null,
  "address": ""
};

void main() {
  // 提取用户名,为空则显示「游客」
  String username = apiResponse?["username"] ?? "游客";
  // 提取年龄,为空则默认 18
  int age = apiResponse?["age"] ?? 18;
  // 提取地址,为空字符串(非 null)则显示「未填写地址」(需额外判断)
  String address = apiResponse?["address"] ?? "未填写地址";
  address = address.isEmpty ? "未填写地址" : address;

  print("用户名:$username,年龄:$age,地址:$address");
  // 输出:用户名:小明,年龄:18,地址:未填写地址
}

2. 初始化可空变量的默认值

// 定义可空的配置变量
String? appTitle;
int? appVersionCode;

void initAppConfig() {
  // 给配置设置默认值(仅当变量为 null 时生效)
  appTitle ??= "我的Dart应用";
  appVersionCode ??= 1;
}

void main() {
  initAppConfig();
  print("应用标题:$appTitle,版本号:$appVersionCode");
  // 输出:应用标题:我的Dart应用,版本号:1
}

总结

  1. ??空值合并运算符,核心逻辑:「左侧非空返左侧,左侧为空返右侧」,用于给可空变量兜底。
  2. 关键特性:只判断 null、右侧惰性求值、不修改原变量。
  3. 常用拓展:??= 用于给变量本身设置默认值(仅为空时生效);?. + ?? 用于安全访问对象属性并兜底。
  4. 实际场景:接口数据兜底、变量默认值初始化、避免空指针异常,是 Dart 开发中的必备语法。
昨天以前首页
❌
❌