解决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 的二维数组
,但是它们的方式和结果是不同的。
不同点
let board = new Array(n).fill(new Array(n).fill('.'));
这段代码在填充数组时,使用了同一个
数组实例(通过 new Array(n).fill('.')创建)
。这会创建一个数组,其中每个元素都指向同一个数组(也就是说,如果你修改其中一个数组,所有数组都会被修改)。这是因为在 JavaScript 中,对象(包括数组)是通过引用
传递的。const board2: string[][] = new Array(n).fill(0).map(\_ => new Array(n).fill('.'));
这段代码通过.map()函数创建了 n 个新的数组实例。这意味着每个子数组都是独立的,修改其中一个并不会影响其他的。这是因为.map()函数会对数组的每个元素执行回调函数,而回调函数中又创建了一个新的数组。
因此,尽管这两段代码看起来都创建了 n x n 的二维数组,但是他们在数组的引用和修改上有很大的不同。
为什么会造成这样的结果呢?
首先 我们先来自己大概实现下 fill
方法:
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])
参数:
- value:用来填充数组的值。
- start (可选):开始填充数组的索引。默认为 0。
- end (可选):停止填充数组的索引。默认为 arr.length。
- 返回值:被修改的数组,从 start 到 end 被 value 填充。
原地操作
fill()
是一个原地操作,这意味着它会直接修改原始数组
,而不是创建一个新的数组。这与 JavaScript 中的许多其他数组方法(如 map()
或 filter()
)不同,后者会返回新数组并保持原始数组不变。
负索引
fill()
方法也接受负索引
,它们从数组的末尾开始计算。例如,-1 指的是最后一个元素,-2 指的是倒数第二个元素,等等。如果你对 start 或 end 参数使用负索引,fill() 将从数组末尾
开始或停止填充。
兼容性:
所有现代浏览器中都受支持,但在 Internet Explorer 中不受支持。
如果你需要支持旧版浏览器,你可能需要使用polyfill
或者其他方式来实现相同的功能。
总结:
通过这两句代码,让我获益匪浅。虽然花了点时间去排查问题,但结果我还是很开心的。今天的分享就到此为止了,如有错误之处,欢迎大家留言指出,谢谢大家了。
到此这篇关于解决JS使用fill()进行数组填充遇到的问题的文章就介绍到这了,更多相关JS fill()数组填充内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!