Js通过deepMerge实现默认配置和自定义配置的合并

Js通过deepMerge实现默认配置和自定义配置的合并

在做自定义配置文件时,为了简化操作,通常会采用默认配置,再在此基础上合并(Merge)用户自定义配置,且用户自定义配置会覆盖对应的默认配置,经测试发现,使用 deepMerge 库可以轻松灵活实现。<font color="red">提示:请确保您的计算机安装了 NodeJS 运行环境。</font>

1 实现目标

通过 defaultConfig 对象默认配置值,可以确保在没有进行用户配置的情况下,系统按照默认配置运行,使应用能够“开箱即用”,提升用户体验;当用户需要满足各种不同需要时,可以通过 userConfig 对象进行用户功能配置,满足灵活应用的需要。

2 关于 deepMerge

2.1 deepMerge 库相关介绍

图1 deepmerge

deepMerge 库可以通过 npm 安装。

安装包的链接为:https://www.npmjs.com/package/deepmerge

github 仓库地址为:https://github.com/TehShrike/deepmerge

2.2 deepMerge 的引入

可以在 Shell 里运行下面的命令安装:

npm install deepmerge

安装完成后可通过下面的 js 代码引入:

const merge = require('deepmerge')

2.3 deepMerge 基本用法

merge(x, y, [options])

深度合并两个对象 x 和 y ,返回一个新的合并后的对象,元素来自 x 和 y , x 和 y 都不会发生改变。
注意:上面的 options 是可选项,默认情况下(即没有输入 options 时),两个对象的数组会连接起来。

  • 下面的代码测试 deepMerge 库默认配置下的效果:
const merge = require('deepmerge')

// 下面定义数组覆盖选项,当要实现数组覆盖的时候,可以在 merge 时使用该选项
const overwriteMerge = (destinationArray, sourceArray, options) => sourceArray

let x = {
    a: {
        a1: 'A1',
        a2: 'A2'
    },
    b: {
        b1: ['B1']
    }
}

let y = {
    a: {
        a1: 'A1-new',
        a3: 'created'
    },
    b: {
        b1: ['B2']
    }
}

let z1 = merge(x, y)
console.log('下面采用默认选项的 merge 结果:')
console.log(z1)

let z2 = merge(x, y, { arrayMerge: overwriteMerge })
console.log('下面采用数组覆盖选项的 merge 结果:')
console.log(z2)
  • 得到对象 z 的结果打印如下:
//下面采用默认选项的 merge 结果:
{
    a: {
        a1: 'A1-new',
        a2: 'A2',
        a3: 'created'
    },
    b: {
        b1: ['B1', 'B2']
    }
}
//下面采用数组覆盖选项的 merge 结果:
{
    a: {
        a1: 'A1-new',
        a2: 'A2',
        a3: 'created'
    },
    b: {
        b1: ['B2']
    }
}
  • 结果解析
    1)值类型会被覆盖
    值类型为:字符串(string)、数值(number)、布尔值(boolean)、null、undefined,当 x,y 对象中有相同的key 对应的数据类型为值类型时,值会被覆盖,如:
    x.a.a1 = 'A1'
    y.a.a1 = 'A1-new'
    merge的结果就是 'A1-new' 会覆盖 'A1',z.a.a1 = 'A1-new'
    2)“左侧”内容会保留
    如某键在 x (左侧)中存在,在 y(右侧) 中不存在,则 x 的内容会被保留,如:x.a.a2 = 'A2', y.a.a2不存在,所以:z.a.a2 = 'A2'( x.a.a2 被保留在 z.a.a2 中)
    3)默认配置下引用类型:对象(Object)、数组(Array)默认会合并
    如对象属性:x.a.a3 不存在,y.a.a3 = 'created',y.a.a3的内容就被合并到 z 中,z.a.a3 = 'created'
    如数组元素:x.b.b1 = [ 'B1' ],y.b.b1 = [ 'B2' ],此两项内容被合并到 z 中,z.b.b1 = [ 'B1', 'B2' ]
    4)配置数组属性覆盖
    当配置数组属性覆盖时,在 merge 后的对象 z 里,z.b.b1 的值取 y.b.b1 的值,即z.b.b1= [ 'B2' ]

    3 应用测试

    创建 test-merge.js 文件,编写代码如下

    const merge = require('deepmerge')
    

    下面定义数组覆盖选项,当要实现数组覆盖的时候,可以在merge时使用该选项

    const overwriteMerge = (destinationArray, sourceArray, options) => sourceArray
    

    3.1 defaultConfig 的对象定义

    在 test-merge.js 文件里定义 defaultConfig 对象:

    const defaultConfig={
      title: '模拟考试',
      students: 50,
      subjects: 2,
      subjectList: [ '语文', '数学' ]
    }
    

    3.2 userConfig 的对象定义

    在 test-merge.js 文件里定义 userConfig 对象:

    const userConfig={
      title: 'xx年度会考',
      students: 60,
      subjectList: [ '地理', '生物' ]
    }
    

    3.3 deepMerge 合并生成 finalConfig 对象

    一般作为配置文件的合并,数组一般会使用覆盖模式,继续编写代码:

    const finalConfig = merge(defaultConfig, userConfig, { arrayMerge: overwriteMerge })
    console.log(finalConfig)
    

    运行后得到的结果为

    { 
      title: 'xx年度会考', 
      students: 60, 
      subjects: 2,
      subjectsList: [ '地理', '生物' ] 
    }
    

    完美实现默认配置和自定义配置的合并。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,039评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,223评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,916评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,009评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,030评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,011评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,934评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,754评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,202评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,433评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,590评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,321评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,917评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,568评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,738评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,583评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,482评论 2 352

推荐阅读更多精彩内容