相关内容JavaScript中是否存在“空合并”操作符?JavaScript现在有一个??我看到使用更频繁的运算符。以前大多数JavaScript代码使用||。
let userAge = null
// These values will be the same.
let age1 = userAge || 21
let age2 = userAge ?? 21
在什么情况下会??||表现不同?
相关内容JavaScript中是否存在“空合并”操作符?JavaScript现在有一个??我看到使用更频繁的运算符。以前大多数JavaScript代码使用||。
let userAge = null
// These values will be the same.
let age1 = userAge || 21
let age2 = userAge ?? 21
在什么情况下会??||表现不同?
当前回答
函数f(输入){ Const val = input || 1; 返回41 + val; } 函数g(输入){ val = input ??1; 返回41 + val; } Console.log ("using ||:", f(0)); console.log(“使用? ?:“g (0));
null(ish)合并运算符只适用于null和undefined。所以,当你不想要这些值,但你将接受其他假值时,使用null(ish)合并运算符:
console.log(空? ?“nullish”); console.log(未定义? ?“nullish”); console.log ("" ??“nullish”); console.log (0 ??“nullish”); console.log(假的? ?“nullish”);
逻辑上的“或”将跳过任何错误的值,并给出另一个值。 逻辑或将跳过任何错误的值,并给出另一个参数。这是有效的,现在是惯用的,但它并不总是你想要的:
console.log(null ||“假”); 控制台.log(未定义||“假”); console.log(“” || ”假“); 控制台.log(0 ||“假”); console.log(false ||“假”);
这里有一些关于如何决定你需要哪一个的规则。最简单的测试:
你只是想防止空(和未定义-它通常是一样的事情)?然后使用??如果您不确定,那么默认使用空合并操作符可能是个好主意。 你知道你也不想要0或""吗?然后使用||。
第二个问题实际上很棘手。您如何知道需要丢弃虚假值呢?好吧,第一个代码片段展示了这样做会发生什么:f(0)将丢弃0,从而产生不同的结果。这是一种常见的错误来源。由于构造a = b || c通常会引入一个回退值(在本例中是c),它可能会在您无意时意外地返回到它。
function updateAge(years) {
var yearsToAdd = years || 1;
return this.age + yearsToAdd
}
如果您想为调用updateAge()(没有参数)提供一个回退,那么这个方法是可行的,但是如果您调用updateAge(0)(没有更新)则会失败。类似地,你可以有:
function sayMyName(greeting) {
var prefix = greeting || "Hello, my name is ";
return prefix + this.firstName;
}
同样,这适用于sayMyName()(没有参数),但不适用于sayMyName("")(显式地没有问候)。
概括地说,如果您提供的回退值与假值不同,那么您可能会遇到问题。然而,如果你的回退是错误的值- num || 0或str || "",那么这并不重要。
很少(或者应该是)您可能期望一个混合类型(数字、字符串或对象等)并为其提供一个回退:input ||回退是有效的,但将拒绝空字符串和零。它通常是错误的,除非你明确地不想要这些。
简单地说,你应该使用空合并运算符??在任何时候。判断它是否是一个潜在的错误将会减少认知负担。也没有太多理由去避免它。它比布尔OR更新,所以它不能在旧的环境中工作,这是真的,但是现在你应该为那些环境编译你的代码。如果您不能或不愿意这样做,那么您没有选择,应该使用布尔or。
其他回答
如MDN所述:
与逻辑OR(||)运算符相反,如果左操作数是一个非空或未定义的假值,则返回该操作数。换句话说,如果您使用||为另一个变量foo提供一些默认值,如果您认为一些错误的值可用,则可能会遇到意想不到的行为(例如。或0)。更多示例见下文。
还有在你链接的问题的答案中:
无论第一个操作数的类型是什么,如果将其强制转换为布尔型结果为false,则赋值操作将使用第二个操作数。注意以下所有情况: 警报(布尔(null));/ /错误 警报(布尔(定义));/ /错误 警报(布尔(0));/ /错误 alert(布尔(" "));/ /错误 警报(布尔(“false”));// true—gotcha!:)
函数f(输入){ Const val = input || 1; 返回41 + val; } 函数g(输入){ val = input ??1; 返回41 + val; } Console.log ("using ||:", f(0)); console.log(“使用? ?:“g (0));
null(ish)合并运算符只适用于null和undefined。所以,当你不想要这些值,但你将接受其他假值时,使用null(ish)合并运算符:
console.log(空? ?“nullish”); console.log(未定义? ?“nullish”); console.log ("" ??“nullish”); console.log (0 ??“nullish”); console.log(假的? ?“nullish”);
逻辑上的“或”将跳过任何错误的值,并给出另一个值。 逻辑或将跳过任何错误的值,并给出另一个参数。这是有效的,现在是惯用的,但它并不总是你想要的:
console.log(null ||“假”); 控制台.log(未定义||“假”); console.log(“” || ”假“); 控制台.log(0 ||“假”); console.log(false ||“假”);
这里有一些关于如何决定你需要哪一个的规则。最简单的测试:
你只是想防止空(和未定义-它通常是一样的事情)?然后使用??如果您不确定,那么默认使用空合并操作符可能是个好主意。 你知道你也不想要0或""吗?然后使用||。
第二个问题实际上很棘手。您如何知道需要丢弃虚假值呢?好吧,第一个代码片段展示了这样做会发生什么:f(0)将丢弃0,从而产生不同的结果。这是一种常见的错误来源。由于构造a = b || c通常会引入一个回退值(在本例中是c),它可能会在您无意时意外地返回到它。
function updateAge(years) {
var yearsToAdd = years || 1;
return this.age + yearsToAdd
}
如果您想为调用updateAge()(没有参数)提供一个回退,那么这个方法是可行的,但是如果您调用updateAge(0)(没有更新)则会失败。类似地,你可以有:
function sayMyName(greeting) {
var prefix = greeting || "Hello, my name is ";
return prefix + this.firstName;
}
同样,这适用于sayMyName()(没有参数),但不适用于sayMyName("")(显式地没有问候)。
概括地说,如果您提供的回退值与假值不同,那么您可能会遇到问题。然而,如果你的回退是错误的值- num || 0或str || "",那么这并不重要。
很少(或者应该是)您可能期望一个混合类型(数字、字符串或对象等)并为其提供一个回退:input ||回退是有效的,但将拒绝空字符串和零。它通常是错误的,除非你明确地不想要这些。
简单地说,你应该使用空合并运算符??在任何时候。判断它是否是一个潜在的错误将会减少认知负担。也没有太多理由去避免它。它比布尔OR更新,所以它不能在旧的环境中工作,这是真的,但是现在你应该为那些环境编译你的代码。如果您不能或不愿意这样做,那么您没有选择,应该使用布尔or。
简而言之,当您关心从一个可能为空的未定义源分配一个变量时,并且希望为该变量提供一个默认值,请使用空合并运算符??
如果您想避免被伪真理[1]的JavaScript定义搞得一团糟,那么就不要使用||。
空聚结算子??
JavaScript中falsy的定义
angular的一个真实例子
/* parse the url pattern
/search?keyword=mykeyword&page=3
/search?keyword=mykeyword
*/
ngOnInit(): void {
const keyword = this.route.snapshot.queryParamMap.get('keyword') ?? 'default keyword';
const page = this.route.snapshot.queryParamMap.get('page') ?? '0';
this.queryResult$ = this.service.getQueryResult(keyword, +page);
this.keyword = keyword;
}
[1]: 每种编程语言都有自己对伪真理的定义。判断一个值是否为真值的基本方法是,如果显式类型转换函数bool(value)的结果为真,则该值为真值。
对于那些可能跨语言工作,PHP和c#也有??参考PHP, c#。
当||被用作布尔条件时,该值为false。例如:
let userAge = false
// These values will be the same.
let age1 = userAge || 21 // 21
let age2 = userAge ?? 21 // false
逻辑上的OR仍然会为任何不为true的值给出下一个值。在这种情况下,它给出的值是21。在哪里? ?只处理null和undefined。
基于MDN文档:
与逻辑OR(||)运算符相反,如果左操作数是一个非空或未定义的假值,则返回该操作数。
概念的例子:
当使用||为一个非undefined或null的假值时: || 'name';// ==>返回'name 但是当使用??上述情况: 假的? ?“名字”;// ==>返回false
真实的例子: 假设,我们有一个电话变量,在我们的表单中不是强制性的,空字符串(")对我们是有效的,我们想要doSomething()如果电话变量为空或未定义,现在猜猜看:
当使用||为一个非undefined或null的假值时: Const phone = ";//假设它从某个动作变成了空字符串 手机|| doSomething();// ==>该死,doSomething是运行 //但我们想要运行它,如果它是' undefined '或' null '空字符串是有效的 但是当使用??上述情况: Const phone = ";//与上述假设相同 电话? ?doSomething ();// ==>是的,没错 //空字符串对我们是有效的,doSomething不运行
注意:实际上这是一个例子,在一个真实的项目中,你可以更好地理解这个可爱的操作用法。
注意:对于undefined或null,它们都是相互作用的。