2020-08-27

基本數(shù)據(jù)類型與引用數(shù)據(jù)類型

  • 基本數(shù)據(jù)類型:String珠月、Number、Boolean楔敌、Null啤挎、Undefined、Symbol卵凑。這些類型在內(nèi)存中分別占有固定大小的空間庆聘,他們的值保存在棧空間勺卢,通過按值訪問伙判、拷貝和比較。
  • 引用數(shù)據(jù)類型:Array黑忱、Object宴抚。引用類型的值是對象勒魔,保存在堆內(nèi)存中,而棧內(nèi)存存儲的是對象的變量標(biāo)識符以及對象在堆內(nèi)存中的存儲地址

對于引用類型變量酱塔,棧內(nèi)存中存放的是該對象的訪問地址沥邻,在堆內(nèi)存中為該值分配空間危虱,由于這種值的大小不固定羊娃,因此不能把他們保存到棧內(nèi)存中;但內(nèi)存地址大小是固定的埃跷,因此可以將堆內(nèi)存地址保存到棧內(nèi)存中蕊玷。這樣,當(dāng)查詢引用類型的變量時(shí)弥雹,就先從棧中讀取堆內(nèi)存地址垃帅,然后再根據(jù)該地址取出對應(yīng)的值。

不同類型的賦值方式

<figcaption></figcaption>

基本類型:從一個(gè)變量向另外一個(gè)新變量復(fù)制基本類型的值剪勿,會創(chuàng)建這個(gè)值的一個(gè)副本贸诚,并將該副本復(fù)制給新變量,不會影響到原數(shù)據(jù)[圖片上傳失敗...(image-f7518e-1598515384776)]

<figcaption></figcaption>

引用類型:從一個(gè)變量向另一個(gè)新變量復(fù)制引用類型的值厕吉,其實(shí)復(fù)制的是指針酱固,最終兩個(gè)變量最終都指向同一個(gè)對象,會影響到原數(shù)據(jù)

image.png

深拷貝&淺拷貝

  • 淺拷貝:僅僅是復(fù)制了引用头朱,修改基本數(shù)據(jù)類型不會影響原有數(shù)據(jù)的基本數(shù)據(jù)類型运悲,修改引用數(shù)據(jù)類型會影響原有的數(shù)據(jù)類型
  • 深拷貝:在堆中重新分配內(nèi)存,不同的地址项钮,相同的值班眯,互不影響

淺拷貝實(shí)現(xiàn)

封裝方法實(shí)現(xiàn)
let arr1 = [1, {'name': 'a'}]

let shallowClone = (arr) => {
  let newArr = []
  for (let prop in arr) {
    if (arr.hasOwnProperty(prop)) {
      newArr[prop] = arr[prop]
    }
  }

  return newArr
}

let arr2 = shallowClone(arr1)
arr2[0] = 2
arr2[1].name = 'b'

console.log(JSON.stringify({arr1})) // {"arr1":[1,{"name":"b"}]}
console.log(JSON.stringify({arr2})) // {"arr2":[2,{"name":"b"}]}
// 基本數(shù)據(jù)類型未受到影響、引用數(shù)據(jù)類型還是受到影響了
復(fù)制代碼
Array.prototype.concat()
let arr1 = [1, {'name': 'a'}]
let arr2 = arr1.concat()
arr2[0] = 2
arr2[1].name = 'b'

console.log(JSON.stringify({arr1})) // {"arr1":[1,{"name":"b"}]}
console.log(JSON.stringify({arr2})) // {"arr1":[2,{"name":"b"}]}
復(fù)制代碼
Array.prototype.slice()
let arr1 = [1, {'name': 'a'}]
let arr2 = arr1.slice()
arr2[0] = 2
arr2[1].name = 'b'

console.log(JSON.stringify({arr1})) // {"arr1":[1,{"name":"b"}]}
console.log(JSON.stringify({arr2})) // {"arr2":[2,{"name":"b"}]}
復(fù)制代碼
Object.assign()
let obj1 = {
  'name': 'k',
  'msg': {
    'age': '10',
    'sex': '男'
  },
}

let obj2 = Object.assign({}, obj1)
obj2.name = 'kk'
obj2.msg.age = '20'

console.log(JSON.stringify({obj1})) // {"obj1":{"name":"k","msg":{"age":"20","sex":"男"}}}
console.log(JSON.stringify({obj2})) // {"obj2":{"name":"kk","msg":{"age":"20","sex":"男"}}}
復(fù)制代碼
展開運(yùn)算符
let arr1 = [1, 2, 3, {'name': 'kk'}]
let arr2 = [...arr1]
arr2.push(4); 
arr2[3].name = 'kkk'

console.log(JSON.stringify({arr1})) // {"arr1":[1,2,3,{"name":"kkk"}]}
console.log(JSON.stringify({arr2})) // {"arr2":[1,2,3,{"name":"kkk"},4]}
復(fù)制代碼

深拷貝實(shí)現(xiàn)

let arr1 = [1, {'name': 'a'}]

// 判斷數(shù)據(jù)類型
let checkType = (data) => {
  return Object.prototype.toString.call(data).slice(8, -1)
}

let deepClone = (data) => {
  let newData, targetType = checkType(data)
  if (targetType === 'Object') {
    newData = {}
  }
  else if (targetType === 'Array') {
    newData = []
  }
  else {
    return data
  }

  // 遍歷
  for (let i in data) {
    let val = data[i]
    if (checkType(val) === 'Object' || checkType(val) === 'Array') {
      newData[i] = deepClone(val)
    } else {
      newData[i] = val
    }
  }

  return newData
}

let arr2 = deepClone(arr1)
arr2[0] = 2
arr2[1].name = 'b'

console.log(JSON.stringify({arr1})) // {"arr1":[1,{"name":"a"}]}
console.log(JSON.stringify({arr2})) // {"arr2":[2,{"name":"b"}]}
復(fù)制代碼

JSON.parse(JSON.stringify())

該方法存在局限性

  • 無法存放函數(shù)或undefined烁巫,有丟失風(fēng)險(xiǎn)
  • 無法存放NaN署隘,會變成null

作者:toki
鏈接:https://juejin.im/post/6844903983941697549
來源:掘金
著作權(quán)歸作者所有。商業(yè)轉(zhuǎn)載請聯(lián)系作者獲得授權(quán)亚隙,非商業(yè)轉(zhuǎn)載請注明出處磁餐。


個(gè)人備注:

封裝方法中提到了arr.hasOwnProperty(prop)
這個(gè)方法可以用來檢測一個(gè)對象是否含有特定的自身屬性

語法:obj.hasOwnProperty(prop)
參數(shù):要檢測的屬性 字符串 名稱或者 Symbol
返回值: 用來判斷某個(gè)對象是否含有指定的屬性的 Boolean

具體例子:

function ObjWithProto(){
    this.foo = 'foo_val'
}
ObjWithProto.prototype.bar = 'bar_val'
var dict = new ObjWithProto()
dict.foobar = 'foobar_val'

dict.hasOwnProperty('foo') // true
dict.hasOwnProperty('foobar') // true
dict.hasOwnProperty('bar') // false
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市恃鞋,隨后出現(xiàn)的幾起案子崖媚,更是在濱河造成了極大的恐慌,老刑警劉巖恤浪,帶你破解...
    沈念sama閱讀 222,252評論 6 516
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件畅哑,死亡現(xiàn)場離奇詭異,居然都是意外死亡水由,警方通過查閱死者的電腦和手機(jī)荠呐,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,886評論 3 399
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人泥张,你說我怎么就攤上這事呵恢。” “怎么了媚创?”我有些...
    開封第一講書人閱讀 168,814評論 0 361
  • 文/不壞的土叔 我叫張陵渗钉,是天一觀的道長。 經(jīng)常有香客問我钞钙,道長鳄橘,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 59,869評論 1 299
  • 正文 為了忘掉前任芒炼,我火速辦了婚禮瘫怜,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘本刽。我一直安慰自己鲸湃,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,888評論 6 398
  • 文/花漫 我一把揭開白布子寓。 她就那樣靜靜地躺著暗挑,像睡著了一般。 火紅的嫁衣襯著肌膚如雪别瞭。 梳的紋絲不亂的頭發(fā)上窿祥,一...
    開封第一講書人閱讀 52,475評論 1 312
  • 那天,我揣著相機(jī)與錄音蝙寨,去河邊找鬼晒衩。 笑死,一個(gè)胖子當(dāng)著我的面吹牛墙歪,可吹牛的內(nèi)容都是我干的听系。 我是一名探鬼主播,決...
    沈念sama閱讀 41,010評論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼虹菲,長吁一口氣:“原來是場噩夢啊……” “哼靠胜!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起毕源,我...
    開封第一講書人閱讀 39,924評論 0 277
  • 序言:老撾萬榮一對情侶失蹤浪漠,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后霎褐,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體址愿,經(jīng)...
    沈念sama閱讀 46,469評論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,552評論 3 342
  • 正文 我和宋清朗相戀三年冻璃,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了响谓。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片损合。...
    茶點(diǎn)故事閱讀 40,680評論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖娘纷,靈堂內(nèi)的尸體忽然破棺而出嫁审,到底是詐尸還是另有隱情,我是刑警寧澤赖晶,帶...
    沈念sama閱讀 36,362評論 5 351
  • 正文 年R本政府宣布律适,位于F島的核電站,受9級特大地震影響嬉探,放射性物質(zhì)發(fā)生泄漏擦耀。R本人自食惡果不足惜棉圈,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,037評論 3 335
  • 文/蒙蒙 一涩堤、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧分瘾,春花似錦胎围、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,519評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至上岗,卻和暖如春福荸,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背肴掷。 一陣腳步聲響...
    開封第一講書人閱讀 33,621評論 1 274
  • 我被黑心中介騙來泰國打工敬锐, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人呆瞻。 一個(gè)月前我還...
    沈念sama閱讀 49,099評論 3 378
  • 正文 我出身青樓台夺,卻偏偏與公主長得像,于是被迫代替她去往敵國和親痴脾。 傳聞我的和親對象是個(gè)殘疾皇子颤介,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,691評論 2 361