avaScript 提供了很多克隆數組的方法,其中大多數在性能和結果方面都非常相似。以下是一些可用選項的簡要介紹。
ES6 引入了擴展運算符 ( ...),它提供了可能是創建數組的淺層克隆的最簡單和最常用的方法。
let x=[1, 2, 3, 4];
let y=[...x];
Array.from()有一個非常強大的 API,可以用于許多不同的事情,包括創建一個數組的副本。
let x=[1, 2, 3, 4];
let y=Array.from(x);
與擴展運算符類似,Array.prototype.slice()可用于創建數組的淺拷貝。
let x=[1, 2, 3, 4];
let y=x.slice();
研究一個非正統的選項,Array.prototype.map()可用于將數組的每個元素映射到自身以創建一個新數組。
let x=[1, 2, 3, 4];
let y=x.map(i=> i);
類似地,Array.prototype.filter()可用于返回true的每個元素,從而生成一個包含所有原始數組元素的新數組。
let x=[1, 2, 3, 4];
let y=x.filter(()=> true);
最后,Object.assign()方法可以以與創建對象克隆數組完全相同的方式使用。
let x=[1, 2, 3, 4];
let y=Object.assign([], x);
更多內容請訪問 https://www.icoderoad.com
內容首發于工粽號:程序員大澈,每日分享一段優質代碼片段,歡迎關注和投稿!
大家好,我是大澈!
本文約 900+ 字,整篇閱讀約需 1 分鐘。
今天分享一段優質 JS 代碼片段,輕松實現了對象間的深度克隆。
老規矩,先閱讀代碼片段并思考,再看代碼解析再思考,最后評論區留下你的見解!
const a={ x: 1, y: { y1: 'a' }, z: new Set([1, 2]) };
const b=structuredClone(a);
// a !==b, a.y !==b.y, a.z !==b.z
分享原因
這段代碼展示了 structuredClone 方法在深度復制對象方面的應用,這對于處理復雜的數據結構非常有用。
它比 JSON.parse(JSON.stringify()) 方法更安全,能避免一些類型轉換問題,例如:保留日期對象的原有類型,而不是轉換為字符串。
這個方法相對簡便易用,只需將需要拷貝的對象作為參數傳入即可,項目中推薦去使用!
代碼解析
1. const a={ ... };
這里定義了一個名為 a 的對象,它包含一個基本類型的屬性 x,一個嵌套對象 y ,以及一個 Set 類型的屬性 z 。
2. structuredClone(a);
structuredClone() 是 JavaScript 中用于深拷貝對象的方法。
它使用結構化克隆算法創建給定值的深層拷貝。
語法是 structuredClone(value, { transfer }) :
value:被克隆的對象,可以是任何結構化克隆支持的類型。
transfer(可選參數):是一個可轉移對象的數組,里面的值并沒有被克隆,而是被轉移到被拷貝對象上。
支持多種類型:可以復制大多數內置值,如數組、普通對象、日期(Date)、正則表達式(RegExp)、集合(Set)、映射(Map)等,還包括一些 Web API 類型,如 AudioData、Blob、CryptoKey 等。
但它的一些限制也不可忽略:
不支持拷貝函數,嘗試復制函數會導致 DataCloneError 異常。
不支持 DOM 節點,復制 DOM 節點也會引發異常。
不復制屬性描述符、setter 和 getter 等元數據屬性,getter 屬性的值會被復制,但 getter 函數本身不會被復制。
不遍歷或復制對象原型,克隆后的對象不再屬于原始構造函數的實例,不過其屬性會被復制。
在 Web Workers 中的支持不完全,具體可查看 MDN 瀏覽器兼容性表。在不支持的平臺上,可以使用 polyfill 來模擬該方法。
最后,附上兼容性:
dom-to-image是一個js庫,可以將任意dom節點轉換為矢量(SVG)或光柵(PNG或JPEG)圖像。
npm install dom-to-image -S
/* in ES 6 */
import domtoimage from 'dom-to-image';
/* in ES 5 */
var domtoimage=require('dom-to-image');
所有高階函數都接受DOM節點和渲染選項options ,并返回promises。
<div id="my-node"></div>
var node=document.getElementById('my-node');
// options 可不傳
var options={}
domtoimage.toPng(node, options)
.then(function (dataUrl) {
var img=new Image();
img.src=dataUrl;
document.body.appendChild(img);
})
.catch(function (error) {
console.error('oops, something went wrong!', error);
});
domtoimage.toBlob(document.getElementById('my-node'))
.then(function (blob) {
console.log('blob', blob)
});
domtoimage.toJpeg(document.getElementById('my-node'), { quality: 0.95 })
.then(function (dataUrl) {
var link=document.createElement('a');
link.download='my-image-name.jpeg';
link.href=dataUrl;
link.click();
});
function filter (node) {
return (node.tagName !=='i');
}
domtoimage.toSvg(document.getElementById('my-node'), {filter: filter})
.then(function (dataUrl) {
/* do something */
});
var node=document.getElementById('my-node');
domtoimage.toPixelData(node)
.then(function (pixels) {
for (var y=0; y < node.scrollHeight; ++y) {
for (var x=0; x < node.scrollWidth; ++x) {
pixelAtXYOffset=(4 * y * node.scrollHeight) + (4 * x);
/* pixelAtXY is a Uint8Array[4] containing RGBA values of the pixel at (x, y) in the range 0..255 */
pixelAtXY=pixels.slice(pixelAtXYOffset, pixelAtXYOffset + 4);
}
}
});
Name | 類型 | Default | Description |
filter | Function | —— | 以DOM節點為參數的函數。如果傳遞的節點應包含在輸出中,則應返回true(排除節點意味著也排除其子節點) |
bgcolor | String | —— | 背景色的字符串值,任何有效的CSS顏色值。 |
height | Number | —— | 渲染前應用于節點的高度(以像素為單位)。 |
width | Number | —— | 渲染前應用于節點的寬度(以像素為單位)。 |
style | Object | —— | object對象,其屬性在渲染之前要復制到節點的樣式中。 |
quality | Number | 1.0 | 介于0和1之間的數字,表示JPEG圖像的圖像質量(例如0.92=>92%)。默認值為1.0(100%) |
cacheBust | Boolean | false | 設置為true可將當前時間作為查詢字符串附加到URL請求以啟用清除緩存。 |
imagePlaceholder | Boolean | undefined | 獲取圖片失敗時使用圖片的數據URL作為占位符。默認為未定義,并將在失敗的圖像上引發錯誤。 |
dom-to-image使用SVG的一個特性,它允許在標記中包含任意HTML內容。
dom-to-image.js
// Default impl options
var defaultOptions={
// Default is to fail on error, no placeholder
imagePlaceholder: undefined,
// Default cache bust is false, it will use the cache
cacheBust: false
};
var domtoimage={
toSvg: toSvg,
toPng: toPng,
toJpeg: toJpeg,
toBlob: toBlob,
toPixelData: toPixelData,
impl: {
fontFaces: fontFaces,
images: images,
util: util,
inliner: inliner,
options: {}
}
};
if (typeof module !=='undefined')
module.exports=domtoimage;
else
global.domtoimage=domtoimage;
function toJpeg(node, options) {
options=options || {};
return draw(node, options)
.then(function (canvas) {
return canvas.toDataURL('image/jpeg', options.quality || 1.0);
});
}
復制代碼
function draw(domNode, options) {
return toSvg(domNode, options)
.then(util.makeImage)
.then(util.delay(100))
.then(function (image) {
var canvas=newCanvas(domNode);
canvas.getContext('2d').drawImage(image, 0, 0);
return canvas;
});
function newCanvas(domNode) {
var canvas=document.createElement('canvas');
canvas.width=options.width || util.width(domNode);
canvas.height=options.height || util.height(domNode);
if (options.bgcolor) {
var ctx=canvas.getContext('2d');
ctx.fillStyle=options.bgcolor;
ctx.fillRect(0, 0, canvas.width, canvas.height);
}
return canvas;
}
}
function toSvg(node, options) {
options=options || {};
copyOptions(options);
return Promise.resolve(node)
.then(function (node) {
return cloneNode(node, options.filter, true);
})
.then(embedFonts)
.then(inlineImages)
.then(applyOptions)
.then(function (clone) {
return makeSvgDataUri(clone,
options.width || util.width(node),
options.height || util.height(node)
);
});
function applyOptions(clone) {
if (options.bgcolor) clone.style.backgroundColor=options.bgcolor;
if (options.width) clone.style.width=options.width + 'px';
if (options.height) clone.style.height=options.height + 'px';
if (options.style)
Object.keys(options.style).forEach(function (property) {
clone.style[property]=options.style[property];
});
return clone;
}
}
作者:知其
https://juejin.cn/post/6988045156473634852
*請認真填寫需求信息,我們會在24小時內與您取得聯系。