关于Object.defineProperty getter setter碰见的地址引用问题

        最近回顾vue响应式原理的时候,敲代码敲到了一行很疑惑的地方,自己单独拿出来做了一个demo,此处记录一下

情况一:

let obj = {
  a: "333",
};

let value = obj.a;
Object.defineProperty(obj, "a", {
  get: () => {
    console.log(111);
    return value;
  },
  set: (val) => {
    debugger;
    value = val;
    debugger;
  },
});
value = "789";
console.log(obj)

按我个人之前的理解,代码此时输出应当为

let obj = {
  a: "333",
};

let value = obj.a;
Object.defineProperty(obj, "a", {
  get: () => {
    console.log(111);
    return value;
  },
  set: (val) => {
    debugger; 
    value = val;
    debugger; 
  },
});
value = "789";
console.log(obj); // obj:{a:'333'}

但经过断点调试,发现结果为

let obj = {
  a: "333",
};

let value = obj.a;
Object.defineProperty(obj, "a", {
  get: () => {
    console.log(111);
    return value;
  },
  set: (val) => {
    debugger; 
    value = val;
    debugger; 
  },
});
value = "789";
console.log(obj); // obj:{a:'789'}

附截图

如果将 value=val 注释,结果如下:

 但如果不给obj.a添加拦截

let obj = {
  a: "333",
};

let value = obj.a;
// Object.defineProperty(obj, "a", {
//   get: () => {
//     console.log(111);
//     return value;
//   },
//   set: (val) => {
//     debugger;
//     // value = val;
//     debugger;
//   },
// });
value = "789";
console.log(obj);

输出结果如下:

这就衍生出来了第一个问题,value为什么变化后,obj.a跟着也变化了

情况二:

还是这套代码,但是我做一下小小的改动:

let obj = {
  a: "333",
};

let value = obj.a;
Object.defineProperty(obj, "a", {
  get: () => {
    console.log(111);
    return value;
  },
  set: (val) => {
    debugger;
    value = val;
    debugger;
  },
});
// value = "789";
console.log(obj);
obj.a = "789";

此时debug结果如下:

 

 从输出中我们可以看到,value=val前,obj.a并没有变,value=val后,obj.a被赋值为了value的值

我们改一下代码:

let obj = {
  a: "333",
};

let value = obj.a;
Object.defineProperty(obj, "a", {
  get: () => {
    console.log(111);
    return value;
  },
  set: (val) => {
    debugger;
    // value = val;
    debugger;
  },
});
// value = "789";
console.log(obj);
obj.a = "789";

输出结果应当很明显,结果如下:

 失去了value=val后,setter中obj.a的值并没有变化

这就衍生出了第二个问题,为啥在setter中,value的值还是和obj.a的值是关联的

猜测一下原因:

不管是情况一,还是情况二,我们可以发现,修改的变量虽然是string类型的,但是两个变量却修改一个会影响另一个,此处的合理猜测为由于受Object.defineProperty()方法的影响,局部变量value与obj.a虽然为string类型,但二者指向的内存存储地址为同一个,但是基础类型在内存中是以栈的形式存储的,并且是按值存放,按值访问,复杂类型存储在堆中,以引用地址访问,所以这也只是一种猜测,未来如果找到原因我会更新这条问题,此处作为一个记录。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值