在 ECMAScript 2021(ES12)中,JavaScript 引入了新的逻辑赋值操作符 &&=??=。这些操作符将逻辑运算符与赋值运算符相结合,提供了更加简洁、直观的赋值方式。

虽然已经进入标准比较久了,但是我在实际开发中见到的还比较少,今天我们一起来学习下。

逻辑与赋值操作符 &&=

&&= 的工作原理

&&= 操作符是逻辑与运算符(&&)和赋值运算符(=)的结合。它的作用是:仅当左侧变量为真值(truthy)时,才将右侧的值赋给左侧变量。传统的写法需要使用 if 语句或逻辑运算符,&&= 则提供了更为简洁的方式。

传统写法对比:

// 使用 if 语句
if (x) {
  x = y;
}

// 使用逻辑与运算符
x = x && y;

// 使用 `&&=` 操作符(ES2021)
x &&= y;

真值和假值的概念

在 JavaScript 中,以下值被认为是假值(falsy)

  • false
  • 0
  • '' (空字符串)
  • null
  • undefined
  • NaN

其他所有值都被视为真值(truthy)

示例解析

示例 1:用户登录状态

let isLoggedIn = true;
isLoggedIn &&= getUserData(); // 如果已经登录,获取用户数据

function getUserData() {
  return { name: 'ConardLi', age: 17 };
}

// 结果:isLoggedIn 变为 { name: 'ConardLi', age: 17 }

示例 2:库存更新

let stock = 17;

function sell(quantity) {
  stock >= quantity &&= stock - quantity;
}

sell(5); // stock 为 5
sell(17); // stock 保持为 5,因为 5 >= 17 为 false

在上述示例中,sell 函数仅在库存足够时才减少库存,防止出现负库存的情况。

应用场景

(1)条件更新属性

let config = {
  debugMode: true,
  logLevel: null
};

// 仅当 debugMode 为真时,设置 logLevel
config.debugMode &&= (config.logLevel = 'verbose');

// 结果:config.logLevel 为 'verbose'

(2)链式判断

let user = {
  isActive: true,
  hasMembership: true
};

// 仅当用户活跃且有会员资格时,给予折扣
user.isActive &&= user.hasMembership &&= applyDiscount();

function applyDiscount() {
  return '已应用折扣';
}

// 结果:user.hasMembership 变为 '已应用折扣'

空值合并赋值操作符 ??=

??= 的工作原理

??= 操作符结合了空值合并运算符(??)和赋值运算符(=)。它的作用是:仅当左侧变量为 nullundefined 时,才将右侧的值赋给左侧变量。这有助于在变量未被初始化时提供默认值。

传统写法对比:

// 使用 if 语句
if (options.timeout === null || options.timeout === undefined) {
  options.timeout = 3000;
}

// 使用空值合并运算符
options.timeout = options.timeout ?? 3000;

// 使用 `??=` 操作符(ES2021)
options.timeout ??= 3000;

与其他赋值方式的比较

问题:

  • 逻辑或 || 会将 0''false 等合法值视为需要赋默认值的情况,可能导致意外的结果。
let delay = 0;
delay = delay || 1000; // delay 被误赋值为 1000

解决方案:

  • ??= 仅在变量为 nullundefined 时才赋值,避免了上述问题。
let delay = 0;
delay ??= 1000; // delay 保持为 0

使用三元运算符:

user.name = (user.name !== null && user.name !== undefined) ? user.name : 'Anonymous';

问题: 代码较为冗长,可读性不高。

使用 ??=

user.name ??= 'Anonymous';

优势: 简洁明了,仅在变量为 nullundefined 时才赋值,不影响其他假值。

应用场景

??= 操作符非常适合为可能未定义的变量设置默认值,且不会干扰有效的假值。

示例:

let score = 0;
score ??= 100;    // 保持为 0

let tag = '';
tag ??= 'default'; // 保持为 ''

let active = false;
active ??= true;   // 保持为 false

配置默认参数:

function initializeSettings(settings) {
  settings.theme ??= 'light';
  settings.notificationsEnabled ??= true;
  settings.autoSaveInterval ??= 300;

  return settings;
}

最后

JavaScript 的 &&=??= 操作符为我们提供了更为简洁和精确的赋值方式:

  • &&= 操作符: 仅在左侧变量为真值时才进行赋值,适合用于需要在保持假值的同时,根据真值条件更新变量的场景。
  • ??= 操作符: 仅在左侧变量为 nullundefined 时才进行赋值,适合用于为未定义或空值变量设置默认值的场景。

两个新出的 JavaScript 运算符的更多相关文章

  1. javascript运算符语法概述

    × 目录 [1]个数 [2]优先级 [3]结合性[4]类型[5]规则表 前面的话 javascript中的运算符大多由标点符号表示,少数由关键字表示,它们的语法言简意赅,它们的数量却着实不少.运算符始 ...

  2. JavaScript 运算符

    JavaScript 运算符 JavaScript 运算符用于赋值,比较值,执行算术运算等. JavaScript 算术运算符 算术运算符用于执行两个变量或值的运算. 赋值 y = 5, 以下表格将向 ...

  3. JavaScript运算符有哪些

    JavaScript中的运算符有很多,主要分为算术运算符,等同全同运算符,比较运算符,字符串运算符,逻辑运算符,赋值运算符等.这些运算符都有一些属于自己的运算规则,下面就为大家介绍一下JavaScri ...

  4. JavaScript运算符:递增递减运算符前置和后置的区别

    从两段代码说起 var num1 = 2; var num2 = 20; var num3 = --num1 + num2; var num4 = num1 + num2; console.log(n ...

  5. java基础59 JavaScript运算符与控制流程语句(网页知识)

    1.JavaScript运算符 1.1.加减乘除法 加法:+(加法,连接符,正数)          true是1,false是0    减法:-    乘法:*    除法:/ 1.2.比较运算符 ...

  6. JavaScript运算符:递增和递减(++i,--i 和 i++,i-- 的区别)

    递增和递减操作符直接借鉴自C,而且各有两个版本:前置型 (递增 ++i ,递减 --i )和 后置型 (递增 i++ ,递减 i-- ).书本上对两者的定义是:前置型应该位于要操作的变量之前,而后置型 ...

  7. javaScript运算符学习笔记

    1.赋值运算符 javaScript运算符可以分为简单赋值和复合赋值运算.简单赋值运算是将赋值运算符(=)右边的表达式的值保存到赋值运算符左边的变量中,复合赋值运算则是混合了其他操作(算术运算操作,位 ...

  8. JavaScript 运算符是什么?

    ㈠JavaScript 运算符 ⑴运算符 = 用于赋值. ⑵运算符 + 用于加值. ⑶示例:    向变量赋值,并把它们相加: ; // 向 x 赋值 5 ; // 向 y 赋值 2 var z = ...

  9. JavaScript运算符与流程控制

    JavaScript运算符与流程控制 运算符 赋值运算符 使用=进行变量或常量的赋值. <script> ​ let username = "YunYa"; ​ < ...

  10. JavaScript运算符

    JavaScript运算符 1.算数运算符 设定a = 5. 运算符 描述 例子 结果 + 加 b=a+2 b=7 - 减 b=a-2 b=3 * 乘 b=a*2 b=10 / 除 b=a/2 b=2 ...

随机推荐

  1. Tesla 开发者 API 指南:BLE 密钥 – 身份验证和车辆命令

    注意:本工具只能运行于 mac 或者 linux, win下不支持. 1. 克隆项目到本地 https://github.com/teslamotors/vehicle-command.git 2. ...

  2. 【DVWA】搭建安全测试环境

    一.搭建DVWA环境 用docker输入命令: docker run -d --name zc_dvwa -p 8081:80 vulnerables/web-dvwa 安装完成: 输入地址:http ...

  3. C#|.net core 基础 - 如何判断连续子序列

    前两天同事遇到了一个小需求,想判断一个集合是不是在另一个集合中存在,并且要求顺序一致,然后一起讨论了下应该怎么做,有没有什么比较好的方式?下面分享一下我们想到的方法,如果你也有不同的想法也可以分享给我 ...

  4. git 回退之前某次提交

    git 版本回退,用于误提交或者版本回退 一.回滚到之前的某次版本,且该版本后的提交都不要(用于版本回退) 1).首先查看版本号,用 git log 命令查看要回退的版本对应的commit Id co ...

  5. SQL Server – Concurrency 并发控制

    前言 以前写过相关的, 但这篇主要讲一下概念. 帮助理解 Entity Framework with MySQL 学习笔记一(乐观并发) Asp.net core 学习笔记 ( ef core tra ...

  6. Figma 学习笔记 – Component

    参考 Guide to Components in Figma Figma Tutorial: Components - The Basics (Youtube) 定义与用途 Figma 的 Comp ...

  7. java_day1_认识计算机,java环境,Java关键字、标识符、注释

    一.认识计算机 1.组成: 硬件:cpu,内存,显卡,... 软件: 系统软件:WPS, wegame, steam, IDEA,..... 应用软件:WPS, wegame, steam, IDEA ...

  8. ++i与i++在效率上的细微差别

    在一些特定的使用中, i++ 可能将原值用中间量存起来以待使用,下面看相关程序的汇编代码(使用 gcc ). i++ 源程序: #include <stdio.h> int main(){ ...

  9. 2How To Use Python On A Web Page With Jinja2 - Fla 21:30

    # localhost:5000/user/John @app.route("/user/<name>") def user(name): return render_ ...

  10. 【赵渝强老师】Redis的RDB持久化

    Redis 提供了多种不同级别的持久化方式: RDB 持久化可以在指定的时间间隔内生成数据集的时间点快照(point-in-time snapshot). AOF (Append-only file) ...