Home>Article>Web Front-end> Let’s talk about the 38 tool functions under the Vue3 shared module (source code reading)

Let’s talk about the 38 tool functions under the Vue3 shared module (source code reading)

青灯夜游
青灯夜游 forward
2022-12-12 19:55:49 2696browse

Compared with the tool functions of

Vue2Let’s talk about the 38 tool functions under the Vue3 shared module (source code reading), the tool functions of

##Vue3have changed a lot. My personal feeling is mainly reflected in the syntax. Fully embracees6; [Related recommendations:vuejs video tutorial,web front-end development]

Compared with the tool class, there are no functional changes How much, most of them are basically the same, but there are slight differences in syntax and implementation;

Source code address:

  • vue-next
  • vue-next/packages/shared/src/index.ts
I have introduced many reading methods before, but this time I skipped it and started directly. Read the source code;

All tool functions

  • makeMap: Generate an object similar toSet, use Used to determine whether a certain value exists
  • EMPTY_OBJ: Empty object
  • EMPTY_ARR: Empty array
  • NOOP: Empty function
  • NO: Function that returnsfalse
  • isOn: Determine whether it ison Events starting with
  • isModelListener: Determine the string starting withonUpdate
  • extend: Merge objects
  • remove: Remove a value from the array
  • hasOwn: Determine whether the object has a certain attribute
  • isArray: Determines whether it is an array
  • isMap: Determines whether it isMap
  • ##isSet
  • : Determines whether it is an arraySet
  • isDate
  • : Determine whether it isDate
  • isRegExp
  • : Determine whether it isRegExp
  • isFunction
  • : Determine whether it is a function
  • isString
  • : Determine whether it is a string
  • isSymbol
  • : Determine whether it isSymbol
  • isObject
  • : Determine whether it is an object
  • isPromise
  • : Determine whether IsPromise
  • objectToString
  • :Object.prototype.toString##toTypeString
  • :
  • Abbreviation of Object.prototype.toStringtoRawType
  • : Get the type of object
  • isPlainObject
  • : Determine whether it is a normal object
  • isIntegerKey
  • : Determine whether it is an integer
  • keyisReservedProp
  • : Determine whether it is a reserved property
  • isBuiltInDirective
  • : Determine whether it is a built-in instruction
  • camelize
  • : Convert the string to camel case
  • hyphenate
  • : Convert the string to concatenated case Characters
  • capitalize
  • : Capitalize the first letter of the string
  • toHandlerKey
  • : Convert the string to the
  • keyof the event handlerhasChanged
  • : Determine whether two values are equal
  • invokeArrayFns
  • : Call the function in the array
  • def
  • : Define the properties of the object
  • looseToNumber
  • : Convert a string to a number
  • toNumber
  • : Convert a string to a number
  • getGlobalThis
  • : Get the global object
  • genPropsAccessExp
  • : Generate the access expression of
  • propsthis Most of them are the same as the tool functions of
  • Vue2
, and the judgment of data type is done in the same way. Because of the previous reading experience of

Vue2, this Quick reading;If you want more details, you can read the article I wrote before:[Source code reading] Analysis of 36 utility functions in the Vue2 source code shared module

;

And this time it is the direct source code, thets

version, which is no longer processed into

js, so read thetssource code directly;Official start

makeMap

export function makeMap( str: string, expectsLowerCase?: boolean ): (key: string) => boolean { const map: Record = Object.create(null) const list: Array = str.split(',') for (let i = 0; i < list.length; i++) { map[list[i]] = true } return expectsLowerCase ? val => !!map[val.toLowerCase()] : val => !!map[val] }
makeMap

The source code is in the same directory as

makeMap.tsfile, directly use theexportkeyword to export after importing it. The implementation method is the same as the implementation method ofVue2;

EMPTY_OBJ & EMPTY_ARR

export const EMPTY_OBJ: { readonly [key: string]: any } = __DEV__ ? Object.freeze({}) : {} export const EMPTY_ARR = __DEV__ ? Object.freeze([]) : []

EMPTY_OBJEMPTY_ARR的实现方式和Vue2emptyObject相同,都是使用Object.freeze冻结对象,防止对象被修改;

NOOP

export const NOOP = () => {}

Vue2noop实现方式相同,都是一个空函数,移除了入参;

NO

/** * Always return false. */ export const NO = () => false

Vue2no实现方式相同,都是一个返回false的函数,移除了入参;

isOn

const onRE = /^on[^a-z]/ export const isOn = (key: string) => onRE.test(key)

判断是否是on开头的事件,并且on后面的第一个字符不是小写字母;

isModelListener

export const isModelListener = (key: string) => key.startsWith('onUpdate:')

判断是否是onUpdate:开头的字符串;

参考:startWith

extend

export const extend = Object.assign

直接拥抱es6Object.assignVue2的实现方式是使用for in循环;

remove

export const remove = (arr: T[], el: T) => { const i = arr.indexOf(el) if (i > -1) { arr.splice(i, 1) } }

对比于Vue2删除了一些代码,之前的快速删除最后一个元素的判断不见了;

猜测可能是因为有bug,因为大家都知道Vue2的数组响应式必须使用Arrayapi,那样操作可能会导致数组响应式失效;

hasOwn

const hasOwnProperty = Object.prototype.hasOwnProperty export const hasOwn = ( val: object, key: string | symbol ): key is keyof typeof val => hasOwnProperty.call(val, key)

使用的是Object.prototype.hasOwnProperty,和Vue2相同;

isArray

export const isArray = Array.isArray

使用的是Array.isArray,和Vue2相同;

isMap & isSet & isDate & isRegExp

export const isMap = (val: unknown): val is Map => toTypeString(val) === '[object Map]' export const isSet = (val: unknown): val is Set => toTypeString(val) === '[object Set]' export const isDate = (val: unknown): val is Date => toTypeString(val) === '[object Date]' export const isRegExp = (val: unknown): val is RegExp => toTypeString(val) === '[object RegExp]'

都是使用Object.toString来判断类型,对比于Vue2新增了isMapisSetisDate,实现方式没变;

isFunction & isString & isSymbol & isObject

export const isFunction = (val: unknown): val is Function => typeof val === 'function' export const isString = (val: unknown): val is string => typeof val === 'string' export const isSymbol = (val: unknown): val is symbol => typeof val === 'symbol' export const isObject = (val: unknown): val is Record => val !== null && typeof val === 'object'

Vue2的实现方式相同,都是使用typeof来判断类型,新增了isSymbol

isPromise

export const isPromise = (val: unknown): val is Promise => { return isObject(val) && isFunction(val.then) && isFunction(val.catch) }

Vue2对比修改了实现方式,但是判断逻辑没变;

objectToString

export const objectToString = Object.prototype.toString

直接是Object.prototype.toString

toTypeString

export const toTypeString = (value: unknown): string => objectToString.call(value)

对入参执行Object.prototype.toString

toRawType

export const toRawType = (value: unknown): string => { // extract "RawType" from strings like "[object RawType]" return toTypeString(value).slice(8, -1) }

Vue2的实现方式相同;

isPlainObject

export const isPlainObject = (val: unknown): val is object => toTypeString(val) === '[object Object]'

Vue2的实现方式相同;

isIntegerKey

export const isIntegerKey = (key: unknown) => isString(key) && key !== 'NaN' && key[0] !== '-' && '' + parseInt(key, 10) === key

判断一个字符串是不是由一个整数组成的;

isReservedProp

export const isReservedProp = /*#__PURE__*/ makeMap( // the leading comma is intentional so empty string "" is also included ',key,ref,ref_for,ref_key,' + 'onVnodeBeforeMount,onVnodeMounted,' + 'onVnodeBeforeUpdate,onVnodeUpdated,' + 'onVnodeBeforeUnmount,onVnodeUnmounted' )

使用makeMap生成一个对象,用于判断入参是否是内部保留的属性;

isBuiltInDirective

export const isBuiltInDirective = /*#__PURE__*/ makeMap( 'bind,cloak,else-if,else,for,html,if,model,on,once,pre,show,slot,text,memo' )

使用makeMap生成一个对象,用于判断入参是否是内置的指令;

cacheStringFunction

const cacheStringFunction =  string>(fn: T): T => { const cache: Record = Object.create(null) return ((str: string) => { const hit = cache[str] return hit || (cache[str] = fn(str)) }) as T }

Vue2cached相同,用于缓存字符串;

camelize

const camelizeRE = /-(\w)/g /** * @private */ export const camelize = cacheStringFunction((str: string): string => { return str.replace(camelizeRE, (_, c) => (c ? c.toUpperCase() : '')) })

-连接的字符串转换为驼峰式,同Vue2camelize相同;

capitalize

const hyphenateRE = /\B([A-Z])/g /** * @private */ export const hyphenate = cacheStringFunction((str: string) => str.replace(hyphenateRE, '-$1').toLowerCase() )

将驼峰式字符串转换为-连接的字符串,同Vue2hyphenate相同;

capitalize

/** * @private */ export const capitalize = cacheStringFunction( (str: string) => str.charAt(0).toUpperCase() + str.slice(1) )

将字符串首字母大写,同Vue2capitalize相同;

toHandlerKey

/** * @private */ export const toHandlerKey = cacheStringFunction((str: string) => str ? `on${capitalize(str)}` : `` )

将字符串首字母大写并在前面加上on

hasChanged

// compare whether a value has changed, accounting for NaN. export const hasChanged = (value: any, oldValue: any): boolean => !Object.is(value, oldValue)

Vue2相比,移除了polyfill,直接使用Object.is

invokeArrayFns

export const invokeArrayFns = (fns: Function[], arg?: any) => { for (let i = 0; i < fns.length; i++) { fns[i](arg) } }

批量调用传递过来的函数列表,如果有参数,会将参数传递给每个函数;

def

export const def = (obj: object, key: string | symbol, value: any) => { Object.defineProperty(obj, key, { configurable: true, enumerable: false, value }) }

使用Object.defineProperty定义一个属性,并使这个属性不可枚举;

looseToNumber

/** * "123-foo" will be parsed to 123 * This is used for the .number modifier in v-model */ export const looseToNumber = (val: any): any => { const n = parseFloat(val) return isNaN(n) ? val : n }

将字符串转换为数字,如果转换失败,返回原字符串;

通过注释知道主要用于v-model.number修饰符;

toNumber

/** * Only conerces number-like strings * "123-foo" will be returned as-is */ export const toNumber = (val: any): any => { const n = isString(val) ? Number(val) : NaN return isNaN(n) ? val : n }

将字符串转换为数字,如果转换失败,返回原数据;

getGlobalThis

let _globalThis: any export const getGlobalThis = (): any => { return ( _globalThis || (_globalThis = typeof globalThis !== 'undefined' ? globalThis : typeof self !== 'undefined' ? self : typeof window !== 'undefined' ? window : typeof global !== 'undefined' ? global : {}) ) }

获取全局对象,根据环境不同返回的对象也不同;

genPropsAccessExp

const identRE = /^[_$a-zA-Z\xA0-\uFFFF][_$a-zA-Z0-9\xA0-\uFFFF]*$/ export function genPropsAccessExp(name: string) { return identRE.test(name) ? `__props.${name}` : `__props[${JSON.stringify(name)}]` }

生成props的访问表达式,如果name是合法的标识符,直接返回__props.name,否则返回通过JSON.stringify转换后的__props[name]

总结

通过这次的源码阅读,我们巩固了一些基础知识,通过对比Vue2的工具函数,我们也了解了Vue3的一些变化;

这些变化个人感觉主要集中在拥抱es6,可以看到放弃ie是多么自由而奔放;

话外题,不知道大家有没有发现MDN上面的浏览器兼容性表格,已经没有了ie的相关信息。

(学习视频分享:vuejs入门教程编程基础视频

The above is the detailed content of Let’s talk about the 38 tool functions under the Vue3 shared module (source code reading). For more information, please follow other related articles on the PHP Chinese website!

Statement:
This article is reproduced at:juejin.cn. If there is any infringement, please contact admin@php.cn delete