整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:

碎片时间学编程「34]:如何在 JavaScript

碎片时间学编程「34]:如何在 JavaScript 中克隆数组?


avaScript 提供了很多克隆数组的方法,其中大多数在性能和结果方面都非常相似。以下是一些可用选项的简要介绍。

扩展运算符

ES6 引入了扩展运算符 ( ...),它提供了可能是创建数组的浅层克隆的最简单和最常用的方法。

let x=[1, 2, 3, 4];
let y=[...x];

Array.from()

Array.from()有一个非常强大的 API,可以用于许多不同的事情,包括创建一个数组的副本。

let x=[1, 2, 3, 4];
let y=Array.from(x);

Array.prototype.slice()

与扩展运算符类似,Array.prototype.slice()可用于创建数组的浅拷贝。

let x=[1, 2, 3, 4];
let y=x.slice();

Array.prototype.map()

研究一个非正统的选项,Array.prototype.map()可用于将数组的每个元素映射到自身以创建一个新数组。

let x=[1, 2, 3, 4];
let y=x.map(i=> i);

Array.prototype.filter()

类似地,Array.prototype.filter()可用于返回true的每个元素,从而生成一个包含所有原始数组元素的新数组。

let x=[1, 2, 3, 4];
let y=x.filter(()=> true);

对象.assign()

最后,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 来模拟该方法。

最后,附上兼容性

OM to Image

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。

  1. 获取PNG图像base64编码的data URL:
<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);
    });
  1. 获取图像blob:
domtoimage.toBlob(document.getElementById('my-node'))
    .then(function (blob) { 
        console.log('blob', blob)
    });
  1. 获取JPEG图像base64编码的data URL并下载:
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();
});
  1. 获取SVGdata URL,但筛选出所有元素:
function filter (node) {
    return (node.tagName !=='i');
}
 
domtoimage.toSvg(document.getElementById('my-node'), {filter: filter})
    .then(function (dataUrl) {
        /* do something */
});
  1. 以uint8数组的形式获取原始像素数据,每4个数组元素表示一个像素的RGBA数据:
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);
          }
        }
    });

options参数

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节点
  • 计算节点和每个子节点的样式,并将其复制到相应的克隆 创建伪元素,因为它们不是以任何方式克隆的
  • 嵌入web字体 查找所有@font face声明的web字体 解析文件URL,下载相应文件 base64编码的内联作为data:URLs 将所有已处理的CSS放入中,然后将其附加到克隆
  • 嵌入图片 在嵌入图片URL 使用backgroundCSS属性的图片,方法类似于字体
  • 将克隆的节点序列化为XML
  • 将XML包装到标记中,然后包装到SVG中,然后使其成为data URL
  • 或者,要以Uint8Array的形式获取PNG内容或原始像素数据,可以创建一个以SVG为源的图像元素,并将其呈现在已经创建的canvas上,从canvas读取内容

部分源码分析

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;
  • defaultOptions设置默认options选项
  • domtoimage的核心api:
    • toSvg
    • toPng
    • toJpeg
    • toBlob
    • toPixelData
  • 例:toJpeg:将draw函数返回的canvas实例,使用canvas的toDataURL方法生成jpeg图片。toSvg函数将递归地克隆原始DOM节点, 将克隆的节点序列化为XML,将XML包装到标记中,然后包装到SVG中,然后使其转成dataURL。
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