Dart ——??运算符
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本身(仅当name为null时)。
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
}
总结
-
??是空值合并运算符,核心逻辑:「左侧非空返左侧,左侧为空返右侧」,用于给可空变量兜底。 - 关键特性:只判断
null、右侧惰性求值、不修改原变量。 - 常用拓展:
??=用于给变量本身设置默认值(仅为空时生效);?. + ??用于安全访问对象属性并兜底。 - 实际场景:接口数据兜底、变量默认值初始化、避免空指针异常,是 Dart 开发中的必备语法。