Js中使用fill()进行数组填充的大坑,你遇到过吗?

前言:

最近在做算法题时,遇到需要创建二维数组并进行初始化的情况,刚开始我使用的是 new Array(n).fill(new Array(n).fill('.')) 进行二维数组的初始化,但无论怎样我都通不过测试用例。这让我十分纳闷,无奈之下,只能对照着标准代码一句一句排查,终于让我找到了罪魁祸首:Array(n).fill(new Array(n).fill('.'))。我将Array(n).fill(new Array(n).fill('.'))改成new Array(n).fill(0).map(\_ => new Array(n).fill('.'))之后,用例通过了。

接下来,我们便通过这两句代码来将 js 中的 fill()方法重学一下,避免下次再出现类似的问题。

两句代码之间的区别

相同点 以上两句代码目的都是创建一个 n x n 的二维数组,但是它们的方式和结果是不同的。

不同点

  1. let board = new Array(n).fill(new Array(n).fill('.'));

    这段代码在填充数组时,使用了同一个数组实例(通过 new Array(n).fill('.')创建)。这会创建一个数组,其中每个元素都指向同一个数组 (也就是说,如果你修改其中一个数组,所有数组都会被修改)。这是因为在 JavaScript 中,对象(包括数组)是通过引用传递的。

  2. const board2: string[][] = new Array(n).fill(0).map(\_ => new Array(n).fill('.'));

这段代码通过.map()函数创建了 n 个新的数组实例。这意味着每个子数组都是独立的,修改其中一个并不会影响其他的。这是因为.map()函数会对数组的每个元素执行回调函数,而回调函数中又创建了一个新的数组。

因此,尽管这两段代码看起来都创建了 n x n 的二维数组,但是他们在数组的引用和修改上有很大的不同。

为什么会造成这样的结果呢?

首先 我们先来自己大概实现下 fill 方法:

arduino 复制代码
Array.prototype.fill = function(value, start = 0, end = this.length) {
  // Handle negative values for start
  if (start < 0) {
    start += this.length;
    if (start < 0) start = 0;
  }

  // Handle negative values for end
  if (end < 0) {
    end += this.length;
  }

  // Ensure start is not greater than end
  if (start > end) {
    [start, end] = [end, start];
  }

  // Fill the array
  for (let i = start; i < end; i++) {
    this[i] = value;
  }

  return this;
};

可以看到 fill 方法中是直接将 value 的值给每一项,并且会改变原数组的值。

注意: 这个实现并没有处理一些边缘情况和错误检查,所以它并不是一个完全准确的实现。真实的 Array.prototype.fill() 方法会更加复杂,因为它需要处理各种边缘情况和错误条件。

因此当我们传入一个引用类型时,数组中每一项都是对同一个对象的引用。这意味着如果你修改了一个元素,所有的元素都会受到影响。这个特性在使用 .fill() 方法填充数组时需要特别注意。

fill 方法其他需要注意的地方:

语法:

arr.fill(value ,[start],[end]) 参数:

  1. value:用来填充数组的值。
  2. start (可选):开始填充数组的索引。默认为 0。
  3. end (可选):停止填充数组的索引。默认为 arr.length。
  4. 返回值:被修改的数组,从 start 到 end 被 value 填充。

原地操作

fill() 是一个原地操作,这意味着它会直接修改原始数组,而不是创建一个新的数组。这与 JavaScript 中的许多其他数组方法(如 map() filter())不同,后者会返回新数组并保持原始数组不变

负索引

fill() 方法也接受负索引,它们从数组的末尾开始计算。例如,-1 指的是最后一个元素,-2 指的是倒数第二个元素 ,等等。如果你对 start 或 end 参数使用负索引,fill() 将从数组末尾开始或停止填充

兼容性:

所有现代浏览器中都受支持,但在 Internet Explorer 中不受支持。如果你需要支持旧版浏览器,你可能需要使用polyfill 或者其他方式来实现相同的功能。

总结:

通过这两句代码,让我获益匪浅。虽然花了点时间去排查问题,但结果我还是很开心的。今天的分享就到此为止了,如有错误之处,欢迎大家留言指出,谢谢大家了。

相关推荐
黄尚圈圈17 分钟前
Vue 中引入 ECharts 的详细步骤与示例
前端·vue.js·echarts
浮华似水1 小时前
简洁之道 - React Hook Form
前端
正小安3 小时前
如何在微信小程序中实现分包加载和预下载
前端·微信小程序·小程序
_.Switch5 小时前
Python Web 应用中的 API 网关集成与优化
开发语言·前端·后端·python·架构·log4j
一路向前的月光5 小时前
Vue2中的监听和计算属性的区别
前端·javascript·vue.js
长路 ㅤ   5 小时前
vite学习教程06、vite.config.js配置
前端·vite配置·端口设置·本地开发
长路 ㅤ   5 小时前
vue-live2d看板娘集成方案设计使用教程
前端·javascript·vue.js·live2d
Fan_web5 小时前
jQuery——事件委托
开发语言·前端·javascript·css·jquery
安冬的码畜日常5 小时前
【CSS in Depth 2 精译_044】第七章 响应式设计概述
前端·css·css3·html5·响应式设计·响应式
莹雨潇潇6 小时前
Docker 快速入门(Ubuntu版)
java·前端·docker·容器