您当前的位置:首页 > 网站建设 > javascript
| php | asp | css | H5 | javascript | Mysql | Dreamweaver | Delphi | 网站维护 | 帝国cms | React | 考试系统 | ajax | jQuery | 小程序 |

Vue

51自学网 2022-02-21 13:35:35
  javascript
  • 面试官:看过 Vue 的源码没?
  • 候选者:看过。
  • 面试官:那你说下 Vue data 中随意更改一个属性,视图都会被更新吗?
  • 候选者:不会。
  • 面试官:why?
  • 候选者:如果该属性没有被用到 template 中,就没有必要去更新视图,频繁这样性能不好。
  • 面试官:那 Vue 中是如何去实现该方案的?
  • 候选者:在实例初始化过程中,利用Object.defineProperty对 data 中的属性进行数据监听,如果在 template 中被使用到的属性,就被 Dep 类收集起来,等到属性被更改时会调用notify更新视图。
  • 面试官:那你怎么知道那些属性是在 template 被用到的呢?
  • 候选者:WTF。。。这个倒不是很清楚,您能解释下吗?
  • 面试官:OK,那我就简单解释下:

先写个简单的 demo,其中 data 中有 4 个属性a,b,c,d,在模板中被利用到的属性只有a,b。看看是不是只有a,b才会调用Dep收集起来呢?

new Vue({  el: '#app',  data() {    return {      a: 1,      b: 2,      c: 3,      d: 4,    };  },  created() {    console.log(this.b);    this.b = 'aaa';  },  template: '<div>Hello World{{a}}{{b}}</div>',});

在Vueinstance/state.js里面,会利用proxy把每个属性都 代理一遍

const keys = Object.keys(data)  const props = vm.$options.props  const methods = vm.$options.methods  let i = keys.length  while (i--) {    const key = keys[i]    if (props && hasOwn(props, key)) {      process.env.NODE_ENV !== 'production' && warn(        `The data property "${key}" is already declared as a prop. ` +        `Use prop default value instead.`,        vm      )    } else if (!isReserved(key)) {      // 代理对象的属性      proxy(vm, `_data`, key)    }  }  // observe data  observe(data, true /* asRootData */)

利用defineReactive对data中的每个属性进行劫持

observe(data, true /* asRootData */);// observeconst keys = Object.keys(obj);for (let i = 0; i < keys.length; i++) {  defineReactive(obj, keys[i]);}// defineReactiveObject.defineProperty(obj, key, {  enumerable: true,  configurable: true,  get: function reactiveGetter() {    const value = getter ? getter.call(obj) : val;    // 重点在这里,后续如果在模板中使用到的属性,都会被执行reactiveGetter函数    // 被Dep类 收集起来    if (Dep.target) {      console.log(`${key} 属性 被Dep类收集了`)      dep.depend();      if (childOb) {        childOb.dep.depend();        if (Array.isArray(value)) {          dependArray(value);        }      }    }    return value;  },  set: function reactiveSetter(newVal) {    const value = getter ? getter.call(obj) : val;    /* eslint-disable no-self-compare */    if (newVal === value || (newVal !== newVal && value !== value)) {      return;    }    if (setter) {      // 这里是处理computed set 函数      setter.call(obj, newVal);    } else {      val = newVal;    }    childOb = !shallow && observe(newVal);    // 如果我们在更改属性时,就会调用notify 异步更新视图    dep.notify();  },});

执行$mount进行视图挂载

if (vm.$options.el) {  vm.$mount(vm.$options.el);}

$mount 是调用 Vue 原型上的方法, 重点是最后一句 mount.call(this, el, hydrating)

Vue.prototype.$mount = function (  el?: string | Element,  hydrating?: boolean): Component {  el = el && query(el);  const options = this.$options;  // resolve template/el and convert to render function  /**   * 查看render 函数是否存在?如果不存在就解析template模板   * Vue渲染页面时,有两个方式 1. template,2. render,最终所有的模板类的都需要使用render去渲染   */  if (!options.render) {    let template = options.template;    if (template) {      if (typeof template === 'string') {        if (template.charAt(0) === '#') {          template = idToTemplate(template);          /* istanbul ignore if */          if (process.env.NODE_ENV !== 'production' && !template) {            warn(              `Template element not found or is empty: ${options.template}`,              this            );          }        }      } else if (template.nodeType) {        template = template.innerHTML;      } else {        if (process.env.NODE_ENV !== 'production') {          warn('invalid template option:' + template, this);        }        return this;      }    } else if (el) {      // 如果模板不存在,就创建一个默认的html模板      template = getOuterHTML(el);    }  }  // 重写了Vue.prototype.$mount ,最终调用缓存的mount方法完成对$mount的挂载  return mount.call(this, el, hydrating);};

这里mount调用了 mountComponent(this, el, hydrating) 方法,而 mountComponent是执行了 _render函数,最终_render是调用render 生成一个vnode。

const { render, _parentVnode } = vm.$options;vnode = render.call(vm._renderProxy, vm.$createElement);

最后一张图可以看到是render函数在渲染我们demo里面的template模板,最终只有a, b两个属性才会被Dep类收集起来。

如果文中有错误的地方,麻烦各位指出,我会持续改进的。谢谢, 需要调试源码的,这里点击这里,按照 readme操作即可。希望star下

到此这篇关于Vue data中随意改一个属性,视图都会更新?的文章就介绍到这了,更多相关Vue data 内容请搜索51zixue.net以前的文章或继续浏览下面的相关文章希望大家以后多多支持51zixue.net!


下载地址:
深入了解JavaScript中正则表达式的使用
手机安装GreasyFork油猴js脚本的教程
万事OK自学网:51自学网_软件自学网_CAD自学网自学excel、自学PS、自学CAD、自学C语言、自学css3实例,是一个通过网络自主学习工作技能的自学平台,网友喜欢的软件自学网站。