2つのJavaScriptオブジェクトのプロパティを動的にマージするにはどうすればよいですか?


2519

実行時に2つの(非常に単純な)JavaScriptオブジェクトをマージできる必要があります。たとえば、次のようにします。

var obj1 = { food: 'pizza', car: 'ford' }
var obj2 = { animal: 'dog' }

obj1.merge(obj2);

//obj1 now has three properties: food, car, and animal

このためのスクリプトを持っている人や、これを行うための組み込みの方法を知っている人はいますか?再帰は必要ありません。関数をマージする必要はありません。フラットオブジェクトのメソッドだけです。


「1レベル下」をマージする方法を示す同様の質問に対するこの回答には注目に値します。つまり、(最初の値を2番目の値で上書きする代わりに)重複するキーの値をマージしますが、それ以上再帰しません。IMHO、そのタスクに適したクリーンなコード。
ToolmakerSteve

ところで、上位のいくつかの回答は「浅い」マージを行います。同じキーがobj1とobj2の両方に存在する場合、obj2の値は保持され、obj1の値は削除されます。たとえば、質問の例にがあったvar obj2 = { animal: 'dog', food: 'bone' };場合、マージはになります{ food: 'bone', car: 'ford', animal: 'dog' }。「ネストされたデータ」で作業していて、「ディープマージ」が必要な場合は、「ディープマージ」または「再帰」に関する回答を探してください。の値がある場合はarraysここで説明されているように、github "TehShrike / deepmerge"の "arrayMerge"オプションを使用します
ToolmakerSteve、

リンクの下に従ってください stackoverflow.com/questions/171251/... ES6バージョンの新機能で広がりオペレータ
Dev216

回答:


2908

ECMAScript 2018標準メソッド

オブジェクトスプレッドを使用します

let merged = {...obj1, ...obj2};

merged今の労働組合であるobj1obj2。のプロパティはのプロパティobj2を上書きしobj1ます。

/** There's no limit to the number of objects you can merge.
 *  Later properties overwrite earlier properties with the same name. */
const allRules = {...obj1, ...obj2, ...obj3};

この構文のMDNドキュメントもここにあります。babelを使用している場合、それを機能させるにはbabel-plugin-transform-object-rest-spreadプラグインが必要です。

ECMAScript 2015(ES6)標準メソッド

/* For the case in question, you would do: */
Object.assign(obj1, obj2);

/** There's no limit to the number of objects you can merge.
 *  All objects get merged into the first object. 
 *  Only the object in the first argument is mutated and returned.
 *  Later properties overwrite earlier properties with the same name. */
const allRules = Object.assign({}, obj1, obj2, obj3, etc);

MDN JavaScriptリファレンスを参照


ES5以前の方法

for (var attrname in obj2) { obj1[attrname] = obj2[attrname]; }

これは、変更さobj2れてobj1いないものを使用したい場合に、必要な属性ではないすべての属性を単に追加することに注意してくださいobj1

プロトタイプ全体をクラップするフレームワークを使用している場合は、などのチェックでより洗練されたものhasOwnPropertyにする必要がありますが、そのコードは99%のケースで機能します。

関数の例:

/**
 * Overwrites obj1's values with obj2's and adds obj2's if non existent in obj1
 * @param obj1
 * @param obj2
 * @returns obj3 a new object based on obj1 and obj2
 */
function merge_options(obj1,obj2){
    var obj3 = {};
    for (var attrname in obj1) { obj3[attrname] = obj1[attrname]; }
    for (var attrname in obj2) { obj3[attrname] = obj2[attrname]; }
    return obj3;
}

90
オブジェクトに同じ名前の属性がある場合、これは機能せず、属性をマージすることもできます。
謝Jìléi

69
これは浅いコピー/マージのみを行います。多くの要素を破壊する可能性があります。
ジェイテイラー

45
一部の貧しい魂はプロトタイプ全体でがらくたフレームワークを使用せざるを得ないことを認めるための+1 ...
thejonwithnoh

4
「そのため、プロパティを割り当てるのではなく、単に新しいプロパティをコピーまたは定義するだけです。これにより、マージソースにゲッターが含まれている場合、新しいプロパティをプロトタイプにマージするのに適さなくなる可能性があります。(developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/…)。私は使用しなければなりませんでしたvar merged = {...obj1, ...obj2}
モルグラー2018年

2
@ Ze'ev{...obj1, ...{animal: 'dog'}}
backslash112

1191

jQueryにはこのためのユーティリティもあります:http : //api.jquery.com/jQuery.extend/

jQueryドキュメントから取得:

// Merge options object into settings object
var settings = { validate: false, limit: 5, name: "foo" };
var options  = { validate: true, name: "bar" };
jQuery.extend(settings, options);

// Now the content of settings object is the following:
// { validate: true, limit: 5, name: "bar" }

上記のコードは、という名前の既存のオブジェクトを変更しますsettings


どちらの引数も変更せずに新しいオブジェクトを作成する場合は、次のようにします。

var defaults = { validate: false, limit: 5, name: "foo" };
var options = { validate: true, name: "bar" };

/* Merge defaults and options, without modifying defaults */
var settings = $.extend({}, defaults, options);

// The content of settings variable is now the following:
// {validate: true, limit: 5, name: "bar"}
// The 'defaults' and 'options' variables remained the same.

166
注意:ただし、変数「設定」は変更されます。jQueryは新しいインスタンスを返しません。これ(および名前付け)の理由は、.extend()が、オブジェクトをまとめて変更するのではなく、オブジェクトを拡張するために開発されたためです。新しいオブジェクトが必要な場合(たとえば、設定は触れたくないデフォルトです)、いつでもjQuery.extend({}、settings、options);を使用できます。
webmat

1
右に!本当にありがとう。以前のコメントで述べたように、名前は不十分です。私はjQueryドキュメントを前後に検索しましたが、実行されませんでした。
マイクスタロフ2011年

28
ちなみに、jQuery.extendにも深い(ブール値)設定があります。jQuery.extend(true,settings,override)これは、設定のプロパティがオブジェクトを保持し、オーバーライドにそのオブジェクトの一部しか含まれていない場合に重要です。一致しないプロパティを削除する代わりに、deep設定はそれが存在する場所でのみ更新されます。デフォルトはfalseです。
vol7ron、2011年

19
ジー・ウィリカーズ、彼らは本当にこれを良い名前で付けた。Javascriptオブジェクトを拡張する方法を検索すると、すぐに見つかります。ただし、JavaScriptオブジェクトを融合または統合しようとしている場合は、それに気付かない可能性があります。
デレク・グリア

2
ほんの数行のバニラJSが望みどおりの動作をする場合は、ライブラリメソッドの使用を検討するように人々に言わないでください。jQueryの前に時間がありました!
CrazyMerlin

348

ハーモニーのECMAScript 2015(ES6)を指定Object.assignこれを行います。

Object.assign(obj1, obj2);

現在のブラウザのサポートは向上していますが、サポートのないブラウザ用に開発している場合は、ポリフィルを使用できます。


35
これは浅いマージにすぎないことに注意してください
Joachim Lous

私はその間考えるObject.assign:かなりまともな報道があるkangax.github.io/compat-table/es6/...
LazerBass

13
これが正しい答えになるはずです。最近の人々は、この種のことをサポートしないまれなブラウザー(IE11)と互換性があるようにコードをコンパイルしています。補足:obj2をobj1に追加したくない場合は、次のObject.assign({}, obj1, obj2)
コマンド

6
@Duvrai私が見たレポートによると、2016
IE11

3
@Kermani OPは、再帰が不要であることを明確に指摘しています。したがって、このソリューションが浅いマージを実行することを知ることは有用ですが、この答えはそのままで十分です。JoachimLouのコメントには十分な賛成票が寄せられ、必要に応じてその追加情報を提供しています。深いマージと浅いマージ、および同様のトピックの違いは、この議論の範囲を超えており、他のSOの質問により適していると思います。
NanoWizard 2016

264

オブジェクトのプロパティをマージするコードを探して、ここにたどりつきました しかし、再帰的なマージのためのコードがなかったので、私は自分で書きました。(多分jQuery拡張は再帰的なBTWですか?)とにかく、うまくいけば、誰か他の人もそれが役立つと思うでしょう。

(今、コードは使用しませんObject.prototype:)

コード

/*
* Recursively merge properties of two objects 
*/
function MergeRecursive(obj1, obj2) {

  for (var p in obj2) {
    try {
      // Property in destination object set; update its value.
      if ( obj2[p].constructor==Object ) {
        obj1[p] = MergeRecursive(obj1[p], obj2[p]);

      } else {
        obj1[p] = obj2[p];

      }

    } catch(e) {
      // Property in destination object not set; create it and set its value.
      obj1[p] = obj2[p];

    }
  }

  return obj1;
}

o1 = {  a : 1,
        b : 2,
        c : {
          ca : 1,
          cb : 2,
          cc : {
            cca : 100,
            ccb : 200 } } };

o2 = {  a : 10,
        c : {
          ca : 10,
          cb : 20, 
          cc : {
            cca : 101,
            ccb : 202 } } };

o3 = MergeRecursive(o1, o2);

次のようなオブジェクトo3を作成します

o3 = {  a : 10,
        b : 2,
        c : {
          ca : 10,
          cb : 20,
          cc : { 
            cca : 101,
            ccb : 202 } } };

11
いいですが、最初にオブジェクトのディープコピーを作成します。この方法では、オブジェクトが参照渡しされるため、o1も変更されます。
スケリット

1
これは私が探していたものでした。try catchステートメントに進む前に、obj2.hasOwnProperty(p)があるかどうかを確認してください。そうしないと、プロトタイプチェーンの上位から他のがらくたになってしまう可能性があります。
アダム、

3
Markusに感謝します。o1もMergeRecursiveによって変更されるため、最後に、o1とo3は同一であることに注意してください。何も返さない方が直感的な場合があるため、ユーザーは、提供しているもの(o1)が変更されて結果になることを知っています。また、あなたの例では、もともとo1に含まれていないものをo2に追加する価値があるかもしれませんo2プロパティがo1に挿入されることを示すためにo1-しかし、あなたはこの点で機能します)。
パンケーキ

7
これは良い答えですが、2つの問題があります。1)例外に基づくロジックはありません。この場合、スローされた例外はキャッチされ、予期しない結果が発生します。2)元のオブジェクトが変更されているため、何も返さないという@pancakeのコメントに同意します。例外ロジックなしのjsFiddleの例を次に示します。jsfiddle.net/jlowery2663/z8at6knn/4 – Jeff Lowery
Jeff Lowery

3
また、obj2 [p] .constructor == Arrayは、オブジェクトの配列がある場合に必要です。
Composer

175

underscore.js's extend-methodはこれを1行で行うことに注意してください。

_.extend({name : 'moe'}, {age : 50});
=> {name : 'moe', age : 50}

36
この例は匿名オブジェクトを扱っているので問題ありませんが、そうでない場合、アンダースコアも宛先オブジェクトを変更するため、jQueryの回答に関する突然変異に関するwebmatのコメントがここに適用されます。jQueryの答えのように、空のオブジェクトにこの変異のないただのマージを行うには:_({}).extend(obj1, obj2);
阿部Voelker

8
達成しようとしていることに応じて、関連する可能性のある別の _.defaults(object, *defaults)プロパティがあります。「オブジェクトのnullおよび未定義のプロパティをデフォルトオブジェクトの値で埋め、オブジェクトを返します。」
08に拒否

多くの人が宛先オブジェクトの変更についてコメントしていますが、メソッドに新しいオブジェクトを作成させるのではなく、一般的なパターン(たとえばdojo)はdojo.mixin(dojo.clone(myobj)、{newpropertys: "to add to myobj "})
コリンD

7
アンダースコアは任意の数のオブジェクトを取ることができるため、を実行することにより、元のオブジェクトの変更を回避できますvar mergedObj = _.extend({}, obj1, obj2)
lobati

84

jQuery extend()と同様に、AngularJSにも同じ関数があります。

// Merge the 'options' object into the 'settings' object
var settings = {validate: false, limit: 5, name: "foo"};
var options  = {validate: true, name: "bar"};
angular.extend(settings, options);

1
@naXaこのfnのためだけにlibを追加したくない場合は、オプションだと思います。
juanmf 2016年

67

今日、オブジェクトをマージする必要があります。この質問(および回答)は非常に役に立ちました。私はいくつかの答えを試しましたが、どれも自分のニーズに合いませんでしたので、いくつかの答えを組み合わせ、自分で何かを追加し、新しいマージ機能を思いつきました。ここにあります:

var merge = function() {
    var obj = {},
        i = 0,
        il = arguments.length,
        key;
    for (; i < il; i++) {
        for (key in arguments[i]) {
            if (arguments[i].hasOwnProperty(key)) {
                obj[key] = arguments[i][key];
            }
        }
    }
    return obj;
};

いくつかの使用例:

var t1 = {
    key1: 1,
    key2: "test",
    key3: [5, 2, 76, 21]
};
var t2 = {
    key1: {
        ik1: "hello",
        ik2: "world",
        ik3: 3
    }
};
var t3 = {
    key2: 3,
    key3: {
        t1: 1,
        t2: 2,
        t3: {
            a1: 1,
            a2: 3,
            a4: [21, 3, 42, "asd"]
        }
    }
};

console.log(merge(t1, t2));
console.log(merge(t1, t3));
console.log(merge(t2, t3));
console.log(merge(t1, t2, t3));
console.log(merge({}, t1, { key1: 1 }));

3
良い答えですが、プロパティが最初と2番目に存在し、1番目と2番目をマージして新しいオブジェクトを作成しようとすると、最初のオブジェクトに存在する一意のオブジェクトは失われます
Strikers

2
しかし、それは予想される動作ではありませんか?この関数の目的は、値を引数順にオーバーライドすることです。次に、現在のものを上書きします。どのようにして一意の値を保持できますか?私の例から、あなたは何を期待していmerge({}, t1, { key1: 1 })ますか?
Emre Erkan、

私の期待は、非オブジェクト型の値のみをマージすることです。
AGamePlayer 2015

予想される結果が次のとおりの場合、失敗します:gist.github.com/ceremcem/a54f90923c6e6ec42c7daf24cf2768bd
ceremcem

これは、node.jsプロジェクトのストリクトモードで動作しました。これまでのところ、これがこの投稿の最良の答えです。
クレウィス2018年

48

オブジェクトスプレッドプロパティを使用できます。現在、ステージ3 ECMAScriptの提案です。

const obj1 = { food: 'pizza', car: 'ford' };
const obj2 = { animal: 'dog' };

const obj3 = { ...obj1, ...obj2 };
console.log(obj3);


ブラウザのサポート?
Alph.Dev 2017年

1
@ Alph.Dev caniuse.comをご存知ですか?
connexo 2018年

node.jsで3ドット構文を機能させることができません。ノードはそれについて文句を言います。
クレウィス

41

指定されたソリューションは、割り当てる前source.hasOwnProperty(property)for..inループをチェックインするように変更する必要があります。そうしないと、プロトタイプチェーン全体のプロパティがコピーされてしまいます。


40

1行のコードでN個のオブジェクトのプロパティをマージする

Object.assignこの方法は、ECMAScriptの2015(ES6)規格の一部であり、あなたが必要な正確に何を行います。(IEサポートされていません)

var clone = Object.assign({}, obj);

Object.assign()メソッドは、列挙可能なすべての独自プロパティの値を1つ以上のソースオブジェクトからターゲットオブジェクトにコピーするために使用されます。

続きを読む...

古いブラウザをサポートするためのポリフィル

if (!Object.assign) {
  Object.defineProperty(Object, 'assign', {
    enumerable: false,
    configurable: true,
    writable: true,
    value: function(target) {
      'use strict';
      if (target === undefined || target === null) {
        throw new TypeError('Cannot convert first argument to object');
      }

      var to = Object(target);
      for (var i = 1; i < arguments.length; i++) {
        var nextSource = arguments[i];
        if (nextSource === undefined || nextSource === null) {
          continue;
        }
        nextSource = Object(nextSource);

        var keysArray = Object.keys(nextSource);
        for (var nextIndex = 0, len = keysArray.length; nextIndex < len; nextIndex++) {
          var nextKey = keysArray[nextIndex];
          var desc = Object.getOwnPropertyDescriptor(nextSource, nextKey);
          if (desc !== undefined && desc.enumerable) {
            to[nextKey] = nextSource[nextKey];
          }
        }
      }
      return to;
    }
  });
}

古いブラウザで 'use strict'は何をするのか、またはなぜそこにあるのか object [defineProperty;をサポートするブラウザ キー; getOwnPropertyDescriptor; など]、正確に古いわけではありません。これらはgen.5 UAの以前のバージョンです。
Bekim Bacaj

Objects.assign他の回答が返さないマージされたオブジェクトを返すことを明確にしていただきありがとうございます。それはより関数型のプログラミングスタイルです。
Sridhar Sarnobat

36

次の2つはおそらく良い出発点です。lodashには、これらの特別なニーズに対応するカスタマイザ機能もあります。

_.extendhttp://underscorejs.org/#extend
_.mergehttps://lodash.com/docs#merge


4
上記のメソッドは元のオブジェクトを変更することに注意してください。
Wtower 2016

2つのオブジェクトをマージし、1つまたは2つのレベルよりも深くネストされたプロパティを保持したいので、lodashメソッドが機能しました(単にオブジェクトを置き換える/ワイプするのではなく)。
SamBrick 2017

真の@Wtower。このためにバグがありました。常に..merge({}、...のように始まることを確認してください
Martin Cremer

29

ちなみに、あなたがしているのは、マージではなく、プロパティを上書きすることです...

これは、JavaScriptオブジェクト領域が実際にマージtoされる方法fromです。オブジェクト自体ではないオブジェクトのキーのみがによって上書きされます。その他はすべて実際にマージされます。もちろん、この動作を変更してto[n] is undefined、if などの場合にのみ存在するものを上書きしないようにすることができます。

var realMerge = function (to, from) {

    for (n in from) {

        if (typeof to[n] != 'object') {
            to[n] = from[n];
        } else if (typeof from[n] == 'object') {
            to[n] = realMerge(to[n], from[n]);
        }
    }
    return to;
};

使用法:

var merged = realMerge(obj1, obj2);

3
アンダースコアjs拡張はマージではなく実際に上書きされました
David Cumps '26 / 02/26

1
typeof配列とtypeof nullは 'object'を返し、これを壊すことに注意してください。
Salakar

@ u01jmg3ここで私の答えを参照してください:stackoverflow.com/questions/27936772/…-isObject関数
Salakar

これは、node.jsプロジェクトでstrictモードを使用している場合は機能しません
klewis

28

これが私の刺しです

  1. 深いマージをサポート
  2. 引数を変更しません
  3. 任意の数の引数を取ります
  4. オブジェクトのプロトタイプを拡張しません
  5. 別のライブラリー(jQueryMooToolsUnderscore.jsなど)に依存しない
  6. hasOwnPropertyのチェックを含む
  7. それは足りない :)

    /*
        Recursively merge properties and return new object
        obj1 &lt;- obj2 [ &lt;- ... ]
    */
    function merge () {
        var dst = {}
            ,src
            ,p
            ,args = [].splice.call(arguments, 0)
        ;
    
        while (args.length > 0) {
            src = args.splice(0, 1)[0];
            if (toString.call(src) == '[object Object]') {
                for (p in src) {
                    if (src.hasOwnProperty(p)) {
                        if (toString.call(src[p]) == '[object Object]') {
                            dst[p] = merge(dst[p] || {}, src[p]);
                        } else {
                            dst[p] = src[p];
                        }
                    }
                }
            }
        }
    
       return dst;
    }

例:

a = {
    "p1": "p1a",
    "p2": [
        "a",
        "b",
        "c"
    ],
    "p3": true,
    "p5": null,
    "p6": {
        "p61": "p61a",
        "p62": "p62a",
        "p63": [
            "aa",
            "bb",
            "cc"
        ],
        "p64": {
            "p641": "p641a"
        }
    }
};

b = {
    "p1": "p1b",
    "p2": [
        "d",
        "e",
        "f"
    ],
    "p3": false,
    "p4": true,
    "p6": {
        "p61": "p61b",
        "p64": {
            "p642": "p642b"
        }
    }
};

c = {
    "p1": "p1c",
    "p3": null,
    "p6": {
        "p62": "p62c",
        "p64": {
            "p643": "p641c"
        }
    }
};

d = merge(a, b, c);


/*
    d = {
        "p1": "p1c",
        "p2": [
            "d",
            "e",
            "f"
        ],
        "p3": null,
        "p5": null,
        "p6": {
            "p61": "p61b",
            "p62": "p62c",
            "p63": [
                "aa",
                "bb",
                "cc"
            ],
            "p64": {
                "p641": "p641a",
                "p642": "p642b",
                "p643": "p641c"
            }
        },
        "p4": true
    };
*/

このページでテストした他の関数と比較すると、この関数は本当に再帰的(深いマージを行う)であり、jQuery.extend()が実際にうまく機能していることを模倣しています。ただし、最初のオブジェクト/引数を変更して、ユーザーが空のオブジェクト{}を最初のパラメーターとして渡すか、関数で元のオブジェクトを変更するかを決定できるようにした方がよいでしょう。したがって、これに変更dst = {}するdst = arguments[0]と、渡した最初のオブジェクトがマージされたオブジェクトに変更されます
Jasdeep Khalsa

3
今では、Chromeでの動作が停止しています。toString.call(src) == '[object Object]'オブジェクトの有無を確認するためにこのような構造を使用することは悪い考えだと思います。typeof srcはるかに優れています。さらに、それは配列をオブジェクトに変換します(少なくとも、最新のchromeでそのような動作をしています)。
m03geek

私は貪欲なループの中でこれを行う必要がありました、私はこの関数とこれまで使用していた関数を_.merge(object、[sources])をlodash(一種のアンダースコアライブラリ)からベンチマークしました:関数はほぼ2倍高速です相棒、ありがとな。
Arnaud Bouchot

20

Object.assign()

ECMAScript 2015(ES6)

これはECMAScript 2015(ES6)標準の一部である新しいテクノロジーです。この技術の仕様は確定していますが、さまざまなブラウザでの使用状況と実装状況については、互換性の表を確認してください。

Object.assign()メソッドは、列挙可能なすべての独自プロパティの値を1つ以上のソースオブジェクトからターゲットオブジェクトにコピーするために使用されます。ターゲットオブジェクトを返します。

var o1 = { a: 1 };
var o2 = { b: 2 };
var o3 = { c: 3 };

var obj = Object.assign(o1, o2, o3);
console.log(obj); // { a: 1, b: 2, c: 3 }
console.log(o1);  // { a: 1, b: 2, c: 3 }, target object itself is changed.

19

あまり複雑でないオブジェクトの場合は、JSONを使用できます

var obj1 = { food: 'pizza', car: 'ford' }
var obj2 = { animal: 'dog', car: 'chevy'}
var objMerge;

objMerge = JSON.stringify(obj1) + JSON.stringify(obj2);

// {"food": "pizza","car":"ford"}{"animal":"dog","car":"chevy"}

objMerge = objMerge.replace(/\}\{/, ","); //  \_ replace with comma for valid JSON

objMerge = JSON.parse(objMerge); // { food: 'pizza', animal: 'dog', car: 'chevy'}
// Of same keys in both objects, the last object's value is retained_/

この例では、文字列内で「} {」が発生してはならないことに注意してください。


4
var so1 = JSON.stringify(obj1); var so2 = JSON.stringify(obj1); objMerge = so1.substr(0, so1.length-1)+","+so2.substr(1); console.info(objMerge);
Quamis 2013

function merge(obj1, obj2) { return JSON.parse((JSON.stringify(obj1) + JSON.stringify(obj2)) .replace(/\}\{/g, ',').replace(/,\}/g, '}').replace(/\{,/g, '{')); }
Alex Ivasyuv 2015年

またはワンライナーとして:objMerge = JSON.parse((JSON.stringify(obj1) + JSON.stringify(obj2)).replace(/\}\{/, ", "));
ラビShuki Gur 2017年

1
@Charles、申し訳ありませんが、「この方法は非常に危険」ですが、楽しい方法でもあります。これは、可能な限り安全です。1つ目は、JSONを使用してオブジェクトの内容を読み取るという重い作業を行います。2つ目は、最も安全なJSONオブジェクトを使用します。 JSON互換のオブジェクト文字列リテラルからJSOを再解析するように最適化されたマシン。また、その標準を実装する最初のブラウザーであるIE8との下位互換性もあります。「なんでそんなバカなことを言って逃げなきゃいけないんだろう?」
Bekim Bacaj 2017

stringifyは使用できないため、オブジェクトの関数はここで機能しなくなると思いますが、他の欠点を明確にできますか?
yeahdixon 2018年

18

呼ばれるライブラリがありますdeepmergeGitHubのいくつかのトラクションを得ているように見えます:。これはスタンドアロンで、npmとbowerの両方のパッケージマネージャーから利用できます。

回答からコードをコピーして貼り付けるのではなく、これを使用または改善する傾向があります。


17

これを行う最良の方法は、Object.definePropertyを使用して列挙できない適切なプロパティを追加することです。

この方法でも、Object.prototype.extendを使用してプロパティを作成した場合に得られる、新しく作成された「拡張」がなくても、オブジェクトのプロパティを繰り返し処理できます。

うまくいけば、これが役立ちます:

Object.defineProperty(Object.prototype、 "extend"、{
    列挙可能:false、
    値:function(from){
        var props = Object.getOwnPropertyNames(from);
        var dest = this;
        props.forEach(function(name){
            if(宛先名){
                var destination = Object.getOwnPropertyDescriptor(from、name);
                Object.defineProperty(dest、name、destination);
            }
        });
        これを返す;
    }
});

それが機能したら、次のことができます。

var obj = {
    名前: 'スタック'、
    終了:「オーバーフロー」
}
var replacement = {
    名前: '在庫'
};

obj.extend(replacement);

私はそれについてここにブログ記事を書いたところです:http : //onemoredigit.com/post/1527191998/extending-objects-in-node-js


少々お待ちください---コードが機能しません!:(他のマージアルゴリズムと比較するためにjsperfに挿入しましたが、コードは失敗します-関数extendは存在しません
Jens Roland

16

あなたは単にjQueryを使うことができます extend

var obj1 = { val1: false, limit: 5, name: "foo" };
var obj2 = { val2: true, name: "bar" };

jQuery.extend(obj1, obj2);

さてobj1含まのすべての値obj1およびobj2


2
obj1には、obj2ではなくすべての値が含まれます。最初のオブジェクトを拡張しています。jQuery.extend( target [, object1 ] [, objectN ] )
ruuter、

5
この質問はjQueryに関するものではありません。 JavaScript!= jQuery
Jorge Riv

1
私はJavaScriptをグーグル化しましたが、これはより簡単な方法です
Ehsan

13

プロトタイプにはこれがあります:

Object.extend = function(destination,source) {
    for (var property in source)
        destination[property] = source[property];
    return destination;
}

obj1.extend(obj2) あなたがしたいことをします。


6
あなたはObject.prototype.extendを意味しましたか?いずれにせよ、Object.prototypeを拡張することはお勧めできません。-1。
SolutionYogi

考えてみてください。そうObject.prototypeである必要はありませんObject。良い考えかどうか、これはprototype.jsフレームワークが行うことなので、それを使用している場合、またはそれに依存する別のフレームワークを使用している場合は、Object.prototypeすでにで拡張されextendます。
ephemient 2009

2
私はあなたにうんざりするつもりはありませんが、prototype.jsが行う議論は不十分なので、それはokieだと言っています。Prototype.jsは人気がありますが、JavaScriptの実行方法の手本となるわけではありません。JSプロによるプロトタイプライブラリのこの詳細なレビューを確認してください。dhtmlkitchen.com/?category=/JavaScript/&date=2008/06/17/…–
SolutionYogi

7
誰がその正誤を気にしていますか?それが機能する場合、それは機能します。契約を守ろうとしたり、新しいクライアントを獲得したり、期限を守ったりする場合を除いて、完璧なソリューションを待つのは素晴らしいことです。プログラミングに情熱を持っている人なら誰でも無料でお金をあげれば、最終的にはすべてが「正しい」方法で行われるようになります。
デビッド

1
みんなどこで見つけたのObject.prototypeObject.extendオブジェクトに干渉することはありませんObject。関数をオブジェクトにアタッチするだけです。そしてobj1.extend(obj2)、関数を実行する正しい方法ではありませんObject.extend(obj1, obj2)。instedを使用してください
artnikpro

13

**オブジェクトのマージは、Object.assignまたはスプレッド...演算子を使用して簡単です**

var obj1 = { food: 'pizza', car: 'ford' }
var obj2 = { animal: 'dog', car: 'BMW' }
var obj3 = {a: "A"}


var mergedObj = Object.assign(obj1,obj2,obj3)
 // or using the Spread operator (...)
var mergedObj = {...obj1,...obj2,...obj3}

console.log(mergedObj);

オブジェクトは右から左にマージされます。つまり、右のオブジェクトと同じプロパティを持つオブジェクトは上書きされます。

この例では、obj2.carオーバーライドしますobj1.car


12

誰かがGoogle Closure Libraryを使用している場合:

goog.require('goog.object');
var a = {'a': 1, 'b': 2};
var b = {'b': 3, 'c': 4};
goog.object.extend(a, b);
// Now object a == {'a': 1, 'b': 3, 'c': 4};

同様のヘルパー関数が配列にも存在します:

var a = [1, 2];
var b = [3, 4];
goog.array.extend(a, b); // Extends array 'a'
goog.array.concat(a, b); // Returns concatenation of array 'a' and 'b'

10

David Coallierの方法を拡張しました。

  • 複数のオブジェクトをマージする可能性を追加
  • 深いオブジェクトをサポート
  • オーバーライドパラメータ(最後のパラメータがブール値の場合に検出されます)

オーバーライドがfalseの場合、プロパティはオーバーライドされませんが、新しいプロパティが追加されます。

使用法:obj.merge(merges ... [、override]);

これが私のコードです:

Object.defineProperty(Object.prototype, "merge", {
    enumerable: false,
    value: function () {
        var override = true,
            dest = this,
            len = arguments.length,
            props, merge, i, from;

        if (typeof(arguments[arguments.length - 1]) === "boolean") {
            override = arguments[arguments.length - 1];
            len = arguments.length - 1;
        }

        for (i = 0; i < len; i++) {
            from = arguments[i];
            if (from != null) {
                Object.getOwnPropertyNames(from).forEach(function (name) {
                    var descriptor;

                    // nesting
                    if ((typeof(dest[name]) === "object" || typeof(dest[name]) === "undefined")
                            && typeof(from[name]) === "object") {

                        // ensure proper types (Array rsp Object)
                        if (typeof(dest[name]) === "undefined") {
                            dest[name] = Array.isArray(from[name]) ? [] : {};
                        }
                        if (override) {
                            if (!Array.isArray(dest[name]) && Array.isArray(from[name])) {
                                dest[name] = [];
                            }
                            else if (Array.isArray(dest[name]) && !Array.isArray(from[name])) {
                                dest[name] = {};
                            }
                        }
                        dest[name].merge(from[name], override);
                    } 

                    // flat properties
                    else if ((name in dest && override) || !(name in dest)) {
                        descriptor = Object.getOwnPropertyDescriptor(from, name);
                        if (descriptor.configurable) {
                            Object.defineProperty(dest, name, descriptor);
                        }
                    }
                });
            }
        }
        return this;
    }
});

例とテストケース:

function clone (obj) {
    return JSON.parse(JSON.stringify(obj));
}
var obj = {
    name : "trick",
    value : "value"
};

var mergeObj = {
    name : "truck",
    value2 : "value2"
};

var mergeObj2 = {
    name : "track",
    value : "mergeObj2",
    value2 : "value2-mergeObj2",
    value3 : "value3"
};

assertTrue("Standard", clone(obj).merge(mergeObj).equals({
    name : "truck",
    value : "value",
    value2 : "value2"
}));

assertTrue("Standard no Override", clone(obj).merge(mergeObj, false).equals({
    name : "trick",
    value : "value",
    value2 : "value2"
}));

assertTrue("Multiple", clone(obj).merge(mergeObj, mergeObj2).equals({
    name : "track",
    value : "mergeObj2",
    value2 : "value2-mergeObj2",
    value3 : "value3"
}));

assertTrue("Multiple no Override", clone(obj).merge(mergeObj, mergeObj2, false).equals({
    name : "trick",
    value : "value",
    value2 : "value2",
    value3 : "value3"
}));

var deep = {
    first : {
        name : "trick",
        val : "value"
    },
    second : {
        foo : "bar"
    }
};

var deepMerge = {
    first : {
        name : "track",
        anotherVal : "wohoo"
    },
    second : {
        foo : "baz",
        bar : "bam"
    },
    v : "on first layer"
};

assertTrue("Deep merges", clone(deep).merge(deepMerge).equals({
    first : {
        name : "track",
        val : "value",
        anotherVal : "wohoo"
    },
    second : {
        foo : "baz",
        bar : "bam"
    },
    v : "on first layer"
}));

assertTrue("Deep merges no override", clone(deep).merge(deepMerge, false).equals({
    first : {
        name : "trick",
        val : "value",
        anotherVal : "wohoo"
    },
    second : {
        foo : "bar",
        bar : "bam"
    },
    v : "on first layer"
}));

var obj1 = {a: 1, b: "hello"};
obj1.merge({c: 3});
assertTrue(obj1.equals({a: 1, b: "hello", c: 3}));

obj1.merge({a: 2, b: "mom", d: "new property"}, false);
assertTrue(obj1.equals({a: 1, b: "hello", c: 3, d: "new property"}));

var obj2 = {};
obj2.merge({a: 1}, {b: 2}, {a: 3});
assertTrue(obj2.equals({a: 3, b: 2}));

var a = [];
var b = [1, [2, 3], 4];
a.merge(b);
assertEquals(1, a[0]);
assertEquals([2, 3], a[1]);
assertEquals(4, a[2]);


var o1 = {};
var o2 = {a: 1, b: {c: 2}};
var o3 = {d: 3};
o1.merge(o2, o3);
assertTrue(o1.equals({a: 1, b: {c: 2}, d: 3}));
o1.b.c = 99;
assertTrue(o2.equals({a: 1, b: {c: 2}}));

// checking types with arrays and objects
var bo;
a = [];
bo = [1, {0:2, 1:3}, 4];
b = [1, [2, 3], 4];

a.merge(b);
assertTrue("Array stays Array?", Array.isArray(a[1]));

a = [];
a.merge(bo);
assertTrue("Object stays Object?", !Array.isArray(a[1]));

a = [];
a.merge(b);
a.merge(bo);
assertTrue("Object overrides Array", !Array.isArray(a[1]));

a = [];
a.merge(b);
a.merge(bo, false);
assertTrue("Object does not override Array", Array.isArray(a[1]));

a = [];
a.merge(bo);
a.merge(b);
assertTrue("Array overrides Object", Array.isArray(a[1]));

a = [];
a.merge(bo);
a.merge(b, false);
assertTrue("Array does not override Object", !Array.isArray(a[1]));

私のequalsメソッドはここにあります:JavaScriptでのオブジェクト比較



9

Ext JS 4次のようにそれを行うことができます。

var mergedObject = Ext.Object.merge(object1, object2)

// Or shorter:
var mergedObject2 = Ext.merge(object1, object2)

merge(object):Objectを参照してください。


1
まだ2年後、まだ解決していないとして、バグEXTJS-9173 EXT.Object.mergeでの用心
クリス・

9

うわー。これは、私が複数のページで見た最初のStackOverflow投稿です。別の「答え」を追加するための謝罪

この方法はES5以前のものです-ES6については他にもたくさんの答えがあります。

プロパティを利用して「深い」オブジェクトがマージされることはありませんでしたarguments。ここに私の答えは- コンパクト再帰的な、無制限のオブジェクトの引数を渡すことができます。

function extend() {
    for (var o = {}, i = 0; i < arguments.length; i++) {
        // if (arguments[i].constructor !== Object) continue;
        for (var k in arguments[i]) {
            if (arguments[i].hasOwnProperty(k)) {
                o[k] = arguments[i][k].constructor === Object ? extend(o[k] || {}, arguments[i][k]) : arguments[i][k];
            }
        }
    }
    return o;
}

コメントアウトされている部分はオプションです。オブジェクトではない渡された引数をスキップします(エラーを防止します)。

例:

extend({
    api: 1,
    params: {
        query: 'hello'
    }
}, {
    params: {
        query: 'there'
    }
});

// outputs {api: 1, params: {query: 'there'}}

この答えは今、海に落ちています...


うまくいく最短の答え!それはより多くの賛成票に値します!
JensTörnell

8
var obj1 = { food: 'pizza', car: 'ford' }
var obj2 = { animal: 'dog' }

// result
result: {food: "pizza", car: "ford", animal: "dog"}

jQuery.extend()の使用 - リンク

// Merge obj1 & obj2 to result
var result1 = $.extend( {}, obj1, obj2 );

_.merge()の使用 - リンク

// Merge obj1 & obj2 to result
var result2 = _.merge( {}, obj1, obj2 );

_.extend()の使用 - リンク

// Merge obj1 & obj2 to result
var result3 = _.extend( {}, obj1, obj2 );

Object.assign()ECMAScript 2015(ES6)の使用 - リンク

// Merge obj1 & obj2 to result
var result4 = Object.assign( {}, obj1, obj2 );

すべての出力

obj1: { animal: 'dog' }
obj2: { food: 'pizza', car: 'ford' }
result1: {food: "pizza", car: "ford", animal: "dog"}
result2: {food: "pizza", car: "ford", animal: "dog"}
result3: {food: "pizza", car: "ford", animal: "dog"}
result4: {food: "pizza", car: "ford", animal: "dog"}

7

Markusvsyncの回答に基づいて、これは拡張バージョンです。この関数は任意の数の引数を取ります。DOMのプロパティを設定するために使用できますノードに、値の詳細なコピーを作成するます。ただし、最初の引数は参照によって与えられます。

DOMノードを検出するには、isDOMNode()関数を使用します(スタックオーバーフローの質問JavaScript isDOMを参照— JavaScriptオブジェクトがDOMオブジェクトであるかどうかをどのように確認しますか?

Opera 11、Firefox 6、Internet Explorer 8でテストされました、Google Chrome 16でます。

コード

function mergeRecursive() {

  // _mergeRecursive does the actual job with two arguments.
  var _mergeRecursive = function (dst, src) {
    if (isDOMNode(src) || typeof src !== 'object' || src === null) {
      return dst;
    }

    for (var p in src) {
      if (!src.hasOwnProperty(p))
        continue;
      if (src[p] === undefined)
        continue;
      if ( typeof src[p] !== 'object' || src[p] === null) {
        dst[p] = src[p];
      } else if (typeof dst[p]!=='object' || dst[p] === null) {
        dst[p] = _mergeRecursive(src[p].constructor===Array ? [] : {}, src[p]);
      } else {
        _mergeRecursive(dst[p], src[p]);
      }
    }
    return dst;
  }

  // Loop through arguments and merge them into the first argument.
  var out = arguments[0];
  if (typeof out !== 'object' || out === null)
    return out;
  for (var i = 1, il = arguments.length; i < il; i++) {
    _mergeRecursive(out, arguments[i]);
  }
  return out;
}

いくつかの例

innerHTMLとHTML要素のスタイルを設定する

mergeRecursive(
  document.getElementById('mydiv'),
  {style: {border: '5px solid green', color: 'red'}},
  {innerHTML: 'Hello world!'});

配列とオブジェクトをマージします。undefinedを使用すると、左側の配列/オブジェクトの値を保持できます。

o = mergeRecursive({a:'a'}, [1,2,3], [undefined, null, [30,31]], {a:undefined, b:'b'});
// o = {0:1, 1:null, 2:[30,31], a:'a', b:'b'}

JavaScriptオブジェクト以外の引数(nullを含む)は無視されます。最初の引数を除いて、DOMノードも破棄されます。つまり、new String()のように作成された文字列は実際にはオブジェクトであることに注意してください。

o = mergeRecursive({a:'a'}, 1, true, null, undefined, [1,2,3], 'bc', new String('de'));
// o = {0:'d', 1:'e', 2:3, a:'a'}

2つのオブジェクトを新しいオブジェクトにマージする場合(2つのオブジェクトのいずれにも影響を与えない)、最初の引数として{}を指定します

var a={}, b={b:'abc'}, c={c:'cde'}, o;
o = mergeRecursive(a, b, c);
// o===a is true, o===b is false, o===c is false

編集(ReaperSoonによる):

配列もマージするには

function mergeRecursive(obj1, obj2) {
  if (Array.isArray(obj2)) { return obj1.concat(obj2); }
  for (var p in obj2) {
    try {
      // Property in destination object set; update its value.
      if ( obj2[p].constructor==Object ) {
        obj1[p] = mergeRecursive(obj1[p], obj2[p]);
      } else if (Array.isArray(obj2[p])) {
        obj1[p] = obj1[p].concat(obj2[p]);
      } else {
        obj1[p] = obj2[p];
      }
    } catch(e) {
      // Property in destination object not set; create it and set its value.
      obj1[p] = obj2[p];
    }
  }
  return obj1;
}


5

lodashのデフォルトを使用する必要があります

_.defaultsDeep({ 'user': { 'name': 'barney' } }, { 'user': { 'name': 'fred', 'age': 36 } });
// → { 'user': { 'name': 'barney', 'age': 36 } }

5

Underscore.js、オブジェクトの配列をマージするために行います。

var arrayOfObjects = [ {a:1}, {b:2, c:3}, {d:4} ];
_(arrayOfObjects).reduce(function(memo, o) { return _(memo).extend(o); });

その結果:

Object {a: 1, b: 2, c: 3, d: 4}
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.