vue源码中值得学习的方法


作者:chinamasters   https://segmentfault.com/a/1190000025157159

最近在深入研究vue源码,把学习过程中,看到的一些好玩的的函数方法收集起来做分享,希望对大家对深入学习js有所帮助。如果大家都能一眼看懂这些函数,说明技术还是不错的哦。

1. 数据类型判断

Object.prototype.toString.call()返回的数据格式为 [object Object]类型,然后用slice截取第8位到倒一位,得到结果为 Object

var _toString = Object.prototype.toString;
function toRawType (value) {
  return _toString.call(value).slice(8, -1)
}

运行结果测试

toRawType({}) //  Object 
toRawType([])  // Array    
toRawType(true) // Boolean
toRawType(undefined) // Undefined
toRawType(null) // Null
toRawType(function(){}) // Function

2. 利用闭包构造map缓存数据

vue中判断我们写的组件名是不是html内置标签的时候,如果用数组类遍历那么将要循环很多次获取结果,如果把数组转为对象,把标签名设置为对象的key,那么不用依次遍历查找,只需要查找一次就能获取结果,提高了查找效率。

function makeMap (str, expectsLowerCase) {
    // 构建闭包集合map
    var map = Object.create(null);
    var list = str.split(',');
    for (var i = 0; i < list.length; i++) {
      map[list[i]] = true;
    }
    return expectsLowerCase
      ? function (val) { return map[val.toLowerCase()]; }
      : function (val) { return map[val]; }
}
// 利用闭包,每次判断是否是内置标签只需调用isHTMLTag
var isHTMLTag = makeMap('html,body,base,head,link,meta,style,title')
console.log('res', isHTMLTag('body')) // true

3. 二维数组扁平化

vue中_createElement格式化传入的children的时候用到了simpleNormalizeChildren函数,原来是为了拍平数组,使二维数组扁平化,类似lodash中的flatten方法。

// 先看lodash中的flatten
_.flatten([1, [2, [3, [4]], 5]])
// 得到结果为  [1, 2, [3, [4]], 5]

// vue中
function simpleNormalizeChildren (children) {
  for (var i = 0; i < children.length; i++) {
    if (Array.isArray(children[i])) {
      return Array.prototype.concat.apply([], children)
    }
  }
  return children
}

// es6中 等价于
function simpleNormalizeChildren (children) {
   return [].concat(...children)
}

4. 方法拦截

vue中利用Object.defineProperty收集依赖,从而触发更新视图,但是数组却无法监测到数据的变化,但是为什么数组在使用push pop等方法的时候可以触发页面更新呢,那是因为vue内部拦截了这些方法。

 // 重写push等方法,然后再把原型指回原方法
  var ARRAY_METHOD = [ 'push', 'pop', 'shift', 'unshift', 'reverse',  'sort', 'splice' ];
  var array_methods = Object.create(Array.prototype);
  ARRAY_METHOD.forEach(method => {
    array_methods[method] = function () {
      // 拦截方法
      console.log('调用的是拦截的 ' + method + ' 方法,进行依赖收集');
      return Array.prototype[method].apply(this, arguments);
    }
  });

运行结果测试

var arr = [1,2,3]
arr.__proto__ = array_methods // 改变arr的原型
arr.unshift(6) // 打印结果: 调用的是拦截的 unshift 方法,进行依赖收集

5. 继承的实现

vue中调用Vue.extend实例化组件,Vue.extend就是VueComponent构造函数,而VueComponent利用Object.create继承Vue,所以在平常开发中VueVue.extend区别不是很大。这边主要学习用es5原生方法实现继承的,当然了,es6中 class类直接用extends继承。

  // 继承方法 
  function inheritPrototype(Son, Father) {
    var prototype = Object.create(Father.prototype)
    prototype.constructor = Son
    // 把Father.prototype赋值给 Son.prototype
    Son.prototype = prototype
  }
  function Father(name) {
    this.name = name
    this.arr = [1,2,3]
  }
  Father.prototype.getName = function() {
    console.log(this.name)
  }
  function Son(name, age) {
    Father.call(this, name)
    this.age = age
  }
  inheritPrototype(Son, Father)
  Son.prototype.getAge = function() {
    console.log(this.age)
  }

运行结果测试

var son1 = new Son("AAA", 23)
son1.getName()            //AAA
son1.getAge()             //23
son1.arr.push(4)          
console.log(son1.arr)     //1,2,3,4

var son2 = new Son("BBB", 24)
son2.getName()            //BBB
son2.getAge()             //24
console.log(son2.arr)     //1,2,3

6. 执行一次

once 方法相对比较简单,直接利用闭包实现就好了

function once (fn) {
  var called = false;
  return function () {
    if (!called) {
      called = true;
      fn.apply(this, arguments);
    }
  }
}

7. 浅拷贝

简单的深拷贝我们可以用 JSON.stringify() 来实现,不过vue源码中的looseEqual 浅拷贝写的也很有意思,先类型判断再递归调用,总体也不难,学一下思路。

function looseEqual (a, b) {
  if (a === b) { return true }
  var isObjectA = isObject(a);
  var isObjectB = isObject(b);
  if (isObjectA && isObjectB) {
    try {
      var isArrayA = Array.isArray(a);
      var isArrayB = Array.isArray(b);
      if (isArrayA && isArrayB) {
        return a.length === b.length && a.every(function (e, i) {
          return looseEqual(e, b[i])
        })
      } else if (!isArrayA && !isArrayB) {
        var keysA = Object.keys(a);
        var keysB = Object.keys(b);
        return keysA.length === keysB.length && keysA.every(function (key) {
          return looseEqual(a[key], b[key])
        })
      } else {
        /* istanbul ignore next */
        return false
      }
    } catch (e) {
      /* istanbul ignore next */
      return false
    }
  } else if (!isObjectA && !isObjectB) {
    return String(a) === String(b)
  } else {
    return false
  }
}
function isObject (obj) {
  return obj !== null && typeof obj === 'object'
}

就先分享这些函数,其他函数,后面继续补充,如有不对欢迎指正,谢谢!


最后

  • 欢迎加我微信(winty230),拉你进技术群,长期交流学习...

  • 欢迎关注「前端Q」,认真学前端,做个专业的技术人...

点个在看支持我吧

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Django和Vue是两个流行的Web开发框架,Django是用Python编写的后端框架,Vue是用JavaScript编写的前端框架。 Django和Vue源码包含了许多有用的组件和工具,可以帮助开发者创建高效、可靠和安全的Web应用程序。它们的源码都是开源的,意味着任何人都可以查看、修改和贡献改进代码。 Django的源码包括了许多模板、间件和视图,这些都是用来协助开发者构建Web应用程序的基础组件。它还包括了许多与安全、表单处理和用户身份验证相关的功能,这些对于创建安全的Web应用程序至关重要。 Vue源码包括了许多组件,如路由、状态管理和指令等,这些都是用来构建复杂的交互式Web应用程序的基础组件。它还提供了一些有用的开发工具,如调试器和测试工具,用于简化开发过程。 Django和Vue源码可以相互结合使用,以创建全栈Web应用程序,其Django作为后端,用来处理业务逻辑和数据持久化,Vue作为前端,用来呈现数据和实现交互。使用这种方法,开发者可以创建现代化、高效和灵活的Web应用程序,满足不同的商业需求。 ### 回答2: Django Vue源码指的是使用Django作为后端框架,Vue作为前端框架,结合开发的web应用程序的源代码。Django是一个高度灵活的Python框架,可以轻易地进行数据库、表单、用户身份验证等操作,Vue则是一个开源的JavaScript框架,用于开发单页面应用程序。 Django Vue源码的优点在于,它结合了Django和Vue的两种技术,使得开发人员可以轻松构建功能强大、易于维护的Web应用程序。借助Vue的庞大生态系统,开发人员可以选择一系列易于集成的开源组件,从而加快开发速度。通过使用Vue提供的UI库,可以快速创建出高度响应的界面,增强用户体验。 此外,Django Vue源码具有高度可配置性和可定制性。Django提供了诸如REST框架、ORM等功能,让开发人员可以按照自己的需求来构建API接口,Vue则提供了Vuex和Vue Router等组件,帮助人员实现更加复杂的Web应用程序功能。 总的来说,Django Vue源码是一种强大的技术组合,能够让开发人员更加便捷地开发高度可定制化的Web应用程序。 ### 回答3: Django Vue源码是一个结合了Django和Vue框架的开源项目,它的主要目的是实现快速构建现代化Web应用程序。该项目的源码可以在Github上找到。 Django Vue源码是基于Django REST框架和Vue.js构建的,它提供了一种方便的方法来构建Web应用程序,可以轻松地创建SPA(单页应用程序)或MPA(多页应用程序),并且同时提供了Django框架的可靠性和Vue框架的灵活性。 该项目的前端部分是基于Vuejs框架构建的,使得开发人员可以利用Vuejs框架的各种功能特性,如组件化、数据驱动、虚拟DOM等等,来开发丰富的用户界面。同时,后端采用Django REST框架,通过RESTful API的方式与前端进行交互。 通过Django Vue源码,开发人员可以快速搭建一个完整的Web应用程序,通过Vue.js构建前端用户交互界面,并且可以实现与后端API交互,完成数据的传输和操作。 值得一提的是,Django Vue源码还与Bootstrap框架结合使用,提供了丰富的UI组件库,使得开发界面更加快捷和美观。 总的来说,Django Vue源码是一个提供了高效、灵活、可靠的方案,用于快速搭建现代化Web应用程序的开源项目,值得开发人员深入学习和探索。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值