目录:
- ?:
- ??
- ?.
- ??=
- =>
- .. (Cascade notation)
- ~/
- Bonus: (Planned, not released) Spread operator: …
1. ?:
三目运算符,在很多编程语言中都有,这里做一个完整的描述。
当变量没有使用 if-else
的情况下有两种可能的值时,三目运算符提供了一种更为简洁的操作。
(1)普通写法:
int heightInCM = 185;
String heightCategory = "";
if(heightInCM > 175){
heightCategory = "Tall";
}else{
heightCategory = "short";
}
(2)使用三目运算符写法:
int heightInCM = 185;
String heightCategory = heightInCM > 175 ? "Tall" : "Short";
三目运算符的结构:
condition ? (statement if true) : (statement if false);
如果需要返回值,只需在前面加 return
。
return condition ? (statement if true) : (statement if false);
而不是像下面这样:
condition ? return (statement if true) : return (statement if false);
2. ??
空检查运算符,检查变量的值是否为空,如果为空则给它赋值。正如下面举的这个例子,如果person.name
为null
,那么 name
会被赋值为 Emily CH
。由此可以看出,空检查运算符极大的简化了我们对 null
的处理。
(1)普通写法:
String name = "";
if(person.name == null){
name = "Emily CH";
}else{
name = person.name;
}
OR
String name = person.name != null ? person.name : "Emily CH";
(2)使用空检查运算符写法:
String name = person.name ?? "Emily CH";
3. ?.
假如有一个类 Point
,有 x
和 y
两个变量。
class Point {
int x;
int y;
Point(this.x, this.y);
}
如果我们没有初始化 Point
就取 x
和 y
的值,那么它不是返回一个 null
, 而是直接报错。
Point point;
print(point.x);
会抛出错误:
Unhandled exception:
NoSuchMethodError: The getter 'x' was called on null.
Receiver: null
Tried calling: x
如果相反,它只是返回一个null
,那么我们可以轻松地处理并解决未初始化的问题。
普通写法:
Point point;
if(point != null){
print(point.x);
}else{
print("No value");
}
这样写也还行,但是它的限制是很明显的,举个栗子,我们解析 JSON
数据时,很多时候都是多层嵌套的。例如:
pages[0].contributors[0].authorDetails.basicInfo.firstName
在这个语句中,任何一个环节都有可能是 null
然后导致报错,例如 authorDetails
类中没有给 basicInfo
这个字段赋值,那么 basicInfo
中取 firstName
的值就会报错。
如果要用上面的 if-else
去一个个环节检查,那将是一场噩梦。
?.
有条件的成员访问运算符应运而生,这个运算符的作用可以简单粗略的说成是如果一个对象不为空,那么获取他的内部变量,否则返回 null
。
回到上面的第一个例子,我们用?.
运算符修改一下:
Point point;
int x = point ?. x ?? 0;
这一段代码中,如果 Point
没有被初始化,这里不会报错而是返回一个 null
,然后 null
会被空检查运算符??
捕捉并赋值 0 给 x
。
类似的,解析 JSON
数据的栗子我们也可以修改一下:
pages[0]?.contributors[0]?.authorDetails?.basicInfo?.firstName ?? "N/A";
现在,有任何一个环节为null
,将会返回 N/A
。
4. ??=
??=
简单的说就是如果左侧为空,则执行赋值。并且当变量为空时,只会分配一个值。这么说似乎有点难理解,看下面这个栗子,你应该就可以理解了。
Point point;
point ??= Point(x:1, y:2); //这个赋值会执行
point ??= Point(x:3, y:4); //这个赋值不会执行
这样可以避免重复的赋值。
5. =>
=>
在 Dart
中称为粗箭头运算符。它有两种应用场景。一是 相当于 {return x;}
的作用,二是作用于单条语句。例如:
// 1. 相当于 `{return x;}` 的作用
String demoFunction(String str){
return str.toLowerCase();
}
String demoFunction(String str) => str.toLowerCase();
// 2. 作用于单条语句
void demoFunction(String str) => print(str);
6. ..
级联运算符更改对象属性(通常是在创建对象时)的一种简单方法,而不是获得对该对象的引用并逐个更改属性。举个粟子:
(1) 普通写法:
Point p = Point();
p.x = 3;
p.y = 6;
(2)使用级联运算符
Point point = Point()
..x = 3
..y = 6;
当需要在Builder模式中设置许多属性时,这非常有用。例如:
final addressBook = (AddressBookBuilder()
..name = 'emily'
..email = 'emily@gmail.com'
..phone = (PhoneNumberBuilder()
..number = '1335668038'
..label = 'home')
.build())
.build();
7. ~/
算术运算符,〜/
运算符除以并返回结果的底数(整数部分)。举个粟子:
int a = 3;
int b = 7;
int c = b ~/a;
print(c); // 2
8. …
…
运算符作为集合的传播运算符,使用此运算符可以轻松解压缩列表。举个粟子:
var demoList = [1,2,3,4,5];
var otherList = [
6,
7,
8,
...demoList,
]
9. 总结
文章是 Android 面向需求开发系列中的译文一篇,更多相关文章请关注。如若有什么问题,也可以通过扫描二维码发消息给我。转载请注明出处,谢谢!