JavaScript之對象的拷貝

前言:眾所周知JavaScript 里邊的數(shù)據(jù)類型大的分為:基本數(shù)據(jù)類型和引用數(shù)據(jù)類型屋谭。咱們今天說的對象拷貝就屬于引用數(shù)據(jù)類型里的東西。哪基本數(shù)據(jù)類型和引用數(shù)據(jù)類型之間又啥區(qū)別呢?

基本數(shù)據(jù)類型和引用數(shù)據(jù)類型的區(qū)別

基本數(shù)據(jù)類型存儲(chǔ)的是數(shù)據(jù)
引用數(shù)據(jù)類型存儲(chǔ)的是數(shù)據(jù)的地址(引用)
這就是他們之前的區(qū)別
舉個(gè)栗子

var user = {
    name: "張三",
    age: 18,
    address: {
        city: "大連"
    },
    cards: [
        {
            num: "6240**",
            name: "招商銀行"
        }
    ]
};
var user2 = user;

user2.address.city = "湖北";

console.log(user.address.city === user2.address.city) // ?

結(jié)果是 true掩驱,原因就是他們指向同一個(gè)引用(地址)。那如何避免這樣的問題呢?這就要提到咱們今天的重點(diǎn)了對象的拷貝

對象的拷貝

對象的拷貝又分為淺拷貝和深拷貝。深淺就說明了對象拷貝的層級(jí)。淺拷貝只拷貝第一層級(jí)睬愤,但深拷貝拷貝整個(gè)對象的所有層級(jí)。下面呢纹安,咱們就用代碼來展示一下淺拷貝尤辱。

淺拷貝

先介紹一下里邊用到的一些代碼(代碼解析)

  • Object.prototype.toString.call(target);

    每個(gè)對象都有一個(gè) toString() 方法,當(dāng)該對象被表示為一個(gè)文本值時(shí)钻蔑,或者一個(gè)對象以預(yù)期的字符串方式引用時(shí)自動(dòng)調(diào)用啥刻。默認(rèn)情況下,toString() 方法被每個(gè) Object 對象繼承咪笑。如果此方法在自定義對象中未被覆蓋可帽,toString() 返回 "[object type]",其中 type 是對象的類型窗怒。

    這主要是用于區(qū)分對象與數(shù)組映跟,在JavaScript中數(shù)組也屬于對象,所以用typeof是無法區(qū)分是對象還是數(shù)組扬虚。
    對象 [object Object]
    數(shù)組 [object Array]

  • Object.hasOwnProperty.call(target, key)
    對象本身是否含有該屬性努隙,不會(huì)查找原型鏈上的東西

  • Array.prototype.slice.call(target)
    返回一個(gè)新數(shù)組

/**
 * 對象拷貝
 * @param {*} target 目標(biāo)對象
 * @returns new Object
 */
function clone(target) {
    const type = Object.prototype.toString.call(target);
    if (type === "[object Object]") {
        // 代表這是一個(gè)對象
        const tempObj = {};
        for (const key in target) {
            if (Object.hasOwnProperty.call(target, key)) {
                // 拷貝
                tempObj[key] = target[key];
            }
        }
        return tempObj;
    } else if (type === "[object Array]") {
        // 代表這是一個(gè)數(shù)組
        return Array.prototype.slice.call(target);
    }
}

眼細(xì)的同學(xué)就發(fā)現(xiàn)了,淺拷貝都是僅僅拷貝了對象的第一層辜昵。而深拷貝呢就針對對象的所有層級(jí)進(jìn)行操作荸镊。來看下邊代碼。

深拷貝

先介紹一下里邊用到的一些代碼(代碼解析)

  • arguments.callee
    指向當(dāng)前函數(shù)的引用
/**
 * 對象深拷貝
 * @param {*} target 目標(biāo)對象
 * @returns new Object
 */
function clone(target) {
    if (typeof target !== "object") {
        return target;
    } else {
        const type = Object.prototype.toString.call(target);
        if (type === "[object Object]") {
            // 代表這是一個(gè)對象
            const tempObj = {};
            for (const key in target) {
                if (Object.hasOwnProperty.call(target, key)) {
                    // 深拷貝 不僅僅是針對對象第一次堪置,而是采用遞歸的方式逐個(gè)拷貝
                    tempObj[key] = arguments.callee(target[key]);
                }
            }
            return tempObj;
        } else if (type === "[object Array]") {
            // 代表這是一個(gè)數(shù)組
            const newArr = [];
            target.forEach(element => {
                newArr.push(arguments.callee(element));
            });
            return newArr;
        }
    }
}

開發(fā)中用的代碼

/**
 * 對象拷貝
 * @param {*} target 目標(biāo)對象
 * @param {*} deep 是否開啟深拷貝默認(rèn)為false
 * @returns new Object
 */
function clone(target, deep) {
    if (typeof target !== "object") {
        return target;
    } else {
        const type = Object.prototype.toString.call(target);
        if (type === "[object Object]") {
            // 代表這是一個(gè)對象
            const tempObj = {};
            for (const key in target) {
                if (Object.hasOwnProperty.call(target, key)) {
                    if (deep) {
                        // 深拷貝
                        tempObj[key] = arguments.callee(target[key], deep);
                    } else {
                        // 淺拷貝
                        tempObj[key] = target[key];
                    }
                }
            }
            return tempObj;
        } else if (type === "[object Array]") {
            // 代表這是一個(gè)數(shù)組
            if (deep) {
                const newArr = [];
                target.forEach(element => {
                    newArr.push(arguments.callee(element, deep));
                });
                return newArr;
            } else {
                return Array.prototype.slice.call(target);
            }
        }
    }
}

好了躬存,針對對象的深淺拷貝的解析今天就到這了,謝謝欣賞舀锨。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末岭洲,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子坎匿,更是在濱河造成了極大的恐慌盾剩,老刑警劉巖,帶你破解...
    沈念sama閱讀 207,248評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件替蔬,死亡現(xiàn)場離奇詭異告私,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)承桥,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,681評(píng)論 2 381
  • 文/潘曉璐 我一進(jìn)店門驻粟,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人快毛,你說我怎么就攤上這事格嗅。” “怎么了唠帝?”我有些...
    開封第一講書人閱讀 153,443評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵屯掖,是天一觀的道長。 經(jīng)常有香客問我襟衰,道長贴铜,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,475評(píng)論 1 279
  • 正文 為了忘掉前任瀑晒,我火速辦了婚禮绍坝,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘苔悦。我一直安慰自己轩褐,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,458評(píng)論 5 374
  • 文/花漫 我一把揭開白布玖详。 她就那樣靜靜地躺著把介,像睡著了一般。 火紅的嫁衣襯著肌膚如雪蟋座。 梳的紋絲不亂的頭發(fā)上拗踢,一...
    開封第一講書人閱讀 49,185評(píng)論 1 284
  • 那天,我揣著相機(jī)與錄音向臀,去河邊找鬼巢墅。 笑死,一個(gè)胖子當(dāng)著我的面吹牛券膀,可吹牛的內(nèi)容都是我干的君纫。 我是一名探鬼主播,決...
    沈念sama閱讀 38,451評(píng)論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼三娩,長吁一口氣:“原來是場噩夢啊……” “哼庵芭!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起雀监,我...
    開封第一講書人閱讀 37,112評(píng)論 0 261
  • 序言:老撾萬榮一對情侶失蹤双吆,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后会前,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體好乐,經(jīng)...
    沈念sama閱讀 43,609評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡瓦宜,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,083評(píng)論 2 325
  • 正文 我和宋清朗相戀三年蔚万,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片临庇。...
    茶點(diǎn)故事閱讀 38,163評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡昵慌,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出梧田,到底是詐尸還是另有隱情淳蔼,我是刑警寧澤,帶...
    沈念sama閱讀 33,803評(píng)論 4 323
  • 正文 年R本政府宣布存皂,位于F島的核電站,受9級(jí)特大地震影響司草,放射性物質(zhì)發(fā)生泄漏艰垂。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,357評(píng)論 3 307
  • 文/蒙蒙 一埋虹、第九天 我趴在偏房一處隱蔽的房頂上張望猜憎。 院中可真熱鬧,春花似錦搔课、人聲如沸胰柑。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,357評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽柬讨。三九已至,卻和暖如春袍啡,著一層夾襖步出監(jiān)牢的瞬間踩官,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,590評(píng)論 1 261
  • 我被黑心中介騙來泰國打工境输, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留蔗牡,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,636評(píng)論 2 355
  • 正文 我出身青樓嗅剖,卻偏偏與公主長得像辩越,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子信粮,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,925評(píng)論 2 344

推薦閱讀更多精彩內(nèi)容