Эта статья участвовала в "Проект «Звезда раскопок»”, чтобы выиграть творческий подарочный пакет и бросить вызов творческим поощрительным деньгам.
1. Введение
Привет всем, яВакагава. Добро пожаловать, чтобы следовать за мнойПубличный аккаунт Wakagawa Vision, недавно организованныйЧтение исходного кода, если вам интересно, вы можете добавить меня в WeChatruochuan12Участие продолжается уже больше двух месяцев, все вместе общались, учились и вместе двигались вперед, многие говорили, что многого добились.
Если вы хотите изучить исходный код, я настоятельно рекомендую то, что я написал ранее"Изучение серии "Общая архитектура исходного кода""ВключатьjQuery,underscore,lodash,vuex,sentry,axios,redux,koa,vue-devtools,vuex4,koa-compose,vue 3.2 发布,vue-this,create-vue,玩具viteБолее 10 статей исходного кода.
Эта статья Warehouse Vue-анализ, найти звезду ^ _ ^
недавно организованныйЧтение исходного кода, Давайте вместе изучим исходный код. Поэтому ищите разнообразный исходный код, который стоит изучить и содержит небольшое количество строк кода.
писал раньшеVue3Две связанные статьи.
это написаноКак изучить основы JavaScript,Рекомендуемые книги и учебные материалы,И немного из моего опыта, чтобы поделиться.
присоединитьсяЧитайте исходный код вместеотзывов читателей о том, что TA на самом деле все еще используетVue2. Вы можете написатьVue2Основные служебные функции. какблогер знанийСкромный Лорд ЧислаДухИзучайте то, что я узнаю, используйте это для себя, помогайте другимцель, поэтому я написал эту статью. считатьVue3вспомогательная функциясестра.
Прочитав эту статью, вы узнаете:
1. Vue2 源码 shared 模块中的几十个实用工具函数
2. 如何学习源码中优秀代码和思想,投入到自己的项目中
3. 如何学习 JavaScript 基础知识,会推荐很多学习资料
4. 我的一些经验分享
5. 等等
2. Подготовка окружающей среды
2.1 Прочтите Руководство по участию в проектах с открытым исходным кодом
Открытьvue-репозиторий,
Проекты с открытым исходным кодом обычно можно найти вREADME.mdили.github/contributing.mdНайдите рекомендации по содействию.
И Contribution Guide много пишет об участии в разработке проекта. Например, как запустить, какова структура каталогов проекта. Как инвестировать в развитие, какие нужны резервы знаний и т.д.
мы можемСтруктура каталогов проектаописание, найденоsharedмодуль.
shared: contains utilities shared across the entire codebase.
README.mdа такжеcontributing.mdОбычно на английском языке. Некоторые люди могут быть поставлены в тупик. На самом деле, если вы не можете понять это, вы можете использовать инструменты перевода, такие как перевод с разметкой слов, полностраничный перевод и перевод Google. Затем добавьте английский язык в план последующего обучения.
Эта статья оsharedмодуль, соответствующий путь к файлу:vue/vue/src/shared.
также можно использоватьgithub1sдоступ, быстрее.github1s vue/vue/src/shared
2.2 Чтобы упростить статью, непосредственно изучите упакованный исходный код.
исходный кодvue/vue/src/shared,использовалFlowтип, который может быть не совсем понятен.
Чтобы уменьшить сложность статьи, мы напрямую учимся из репозитория исходного кода.Упакованная строка dist/vue.js с 14 по 379..
Конечно, передняя часть может быть более подробной. я могу говорить прямо
3. 工具函数. Но благодаря моему вступлению выше даже новички могут понять исходный код некоторых проектов с открытым исходным кодом, и, возможно, у них будет определенное чувство выполненного долга. Кроме того, когда на собеседовании задаются аналогичные вопросы или письменные контрольные вопросы, вы говоритеVue2Если вы узнаете из исходного кода, интервьюер обязательно посмотрит на вас с восхищением.
3. Вспомогательные функции
Упакованная линия Vue.js 14 до 379, следующий должен объяснить эти методы.
3.1 emptyObject
/*!
* Vue.js v2.6.14
* (c) 2014-2021 Evan You
* Released under the MIT License.
*/
/* */
var emptyObject = Object.freeze({});
Заморозить объекты. Первый слой не может быть изменен. У объекта также есть метод, чтобы определить, замораживать или нет.
Object.isFrozen(emptyObject); // true
Об объектахAPIРекомендую прочитать мои предыдущие статьиВесь API-анализ объектов JavaScript
Вы также можете увидеть работы Учителя Жуань Ифэна.Вводная книга ES6 отражает
3.2 IsUndef не определен?
// These helpers produce better VM code in JS engines due to their
// explicitness and function inlining.
function isUndef (v) {
return v === undefined || v === null
}
3.3 Определен ли isDef?
JavaScriptЕсть шесть ложных значений.
false
null
undefined
0
'' (空字符串)
NaN
Чтобы судить точно, исходный код Vue2 инкапсулирован вisDef,isTrue,isFalseфункция точного суждения.
См. имя.
function isDef (v) {
return v !== undefined && v !== null
}
3.4 Верно ли isTrue?
См. имя.
function isTrue (v) {
return v === true
}
3.5 Является ли isFalse ложным
См. имя.
function isFalse (v) {
return v === false
}
3.6 isPrimitive определяет, является ли значение примитивным значением.
Определите, является ли это строкой, массивом илиsymbolили логическое значение.
/**
* Check if value is primitive.
*/
function isPrimitive (value) {
return (
typeof value === 'string' ||
typeof value === 'number' ||
// $flow-disable-line
typeof value === 'symbol' ||
typeof value === 'boolean'
)
}
3.7 Оценка isObject — это объект
потому чтоtypeof nullявляется «объектом». Массивы и т. д. также верны, если судить по этой функции.
/**
* Quick object check - this is primarily used to tell
* Objects from primitive values when we know the value
* is a JSON-compliant type.
*/
function isObject (obj) {
return obj !== null && typeof obj === 'object'
}
// 例子:
isObject([]) // true
// 有时不需要严格区分数组和对象
3.8 toRawType преобразуется в необработанный тип
Object.prototype.toString()Метод возвращает строку, представляющую объект.
спецификация экма, который описывает эти типы.
ECMAScript5.1 китайская версия
/**
* Get the raw type string of a value, e.g., [object Object].
*/
var _toString = Object.prototype.toString;
function toRawType (value) {
return _toString.call(value).slice(8, -1)
}
// 例子:
toRawType('') // 'String'
toRawType() // 'Undefined'
3.9 Является ли isPlainObject чистым объектом
/**
* Strict object type check. Only returns true
* for plain JavaScript objects.
*/
function isPlainObject (obj) {
return _toString.call(obj) === '[object Object]'
}
// 上文 isObject([]) 也是 true
// 这个就是判断对象是纯对象的方法。
// 例子:
isPlainObject([]) // false
isPlainObject({}) // true
3.10 Является ли isRegExp регулярным выражением
function isRegExp (v) {
return _toString.call(v) === '[object RegExp]'
}
// 例子:
// 判断是不是正则表达式
isRegExp(/ruochuan/) // true
3.11 Является ли isValidArrayIndex допустимым значением индекса массива?
Доступные значения индекса для массива: 0 ('0'), 1 ('1'), 2 ('2')...
/**
* Check if val is a valid array index.
*/
function isValidArrayIndex (val) {
var n = parseFloat(String(val));
return n >= 0 && Math.floor(n) === n && isFinite(val)
}
глобальныйisFinite()Функция используется для определения того, является ли переданное значение параметра конечным значением (finite number). При необходимости параметр сначала преобразуется в числовое значение.
isFinite(Infinity); // false
isFinite(NaN); // false
isFinite(-Infinity); // false
isFinite(0); // true
isFinite(2e64); // true, 在更强壮的Number.isFinite(null)中将会得到false
isFinite('0'); // true, 在更强壮的Number.isFinite('0')中将会得到false
3.12 isPromise определяет, является ли это обещанием
function isPromise (val) {
return (
isDef(val) &&
typeof val.then === 'function' &&
typeof val.catch === 'function'
)
}
// 例子:
isPromise(new Promise()) // true
использовать здесьisDefСуждение относительноisObjectНемного неточно судить. Но достаточно.
3.13 toString в строку
Преобразовать в строку. представляет собой массив или объект, а объектtoStringпутьObject.prototype.toString,использоватьJSON.stringifyконвертировать.
/**
* Convert a value to a string that is actually rendered.
*/
function toString (val) {
return val == null
? ''
: Array.isArray(val) || (isPlainObject(val) && val.toString === _toString)
? JSON.stringify(val, null, 2)
: String(val)
}
3.14 toNumber to Number
Преобразовать в числа. Если преобразование не удалось, возвращается исходная строка.
/**
* Convert an input value to a number for persistence.
* If the conversion fails, return original string.
*/
function toNumber (val) {
var n = parseFloat(val);
return isNaN(n) ? val : n
}
toNumber('a') // 'a'
toNumber('1') // 1
toNumber('1a') // 1
toNumber('a1') // 'a1'
3.15 makeMap генерирует карту (объект)
Передайте строку, разделенную запятыми, чтобы сгенерироватьmap(пара ключ-значение) и возвращает функцию для обнаруженияkeyЯвляется ли ценность в этомmapсередина. Второй параметр — это вариант нижнего регистра.
/**
* Make a map and return a function for checking if a key
* is in that map.
*/
function makeMap (
str,
expectsLowerCase
) {
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]; }
}
// Object.create(null) 没有原型链的空对象
3.16 Является ли isBuiltInTag встроенным тегом?
/**
* Check if a tag is a built-in tag.
*/
var isBuiltInTag = makeMap('slot,component', true);
// 返回的函数,第二个参数不区分大小写
isBuiltInTag('slot') // true
isBuiltInTag('component') // true
isBuiltInTag('Slot') // true
isBuiltInTag('Component') // true
3.17 Является ли isReservedAttribute зарезервированным атрибутом?
/**
* Check if an attribute is a reserved attribute.
*/
var isReservedAttribute = makeMap('key,ref,slot,slot-scope,is');
isReservedAttribute('key') // true
isReservedAttribute('ref') // true
isReservedAttribute('slot') // true
isReservedAttribute('slot-scope') // true
isReservedAttribute('is') // true
isReservedAttribute('IS') // undefined
3.18 удалить: удаляет элемент из массива.
/**
* Remove an item from an array.
*/
function remove (arr, item) {
if (arr.length) {
var index = arr.indexOf(item);
if (index > -1) {
return arr.splice(index, 1)
}
}
}
spliceНа самом деле, это очень ресурсоемкий метод. Чтобы удалить элемент в массиве, все остальные элементы должны быть перемещены.
Расширенный:axios InterceptorManagerИсходный код перехватчика, перехватчики хранятся в массиве. Но при фактическом удалении перехватчика просто установите перехватчик наnull. Вместо того, чтобы использоватьspliceУдалить. Последнее исполнениеnullне реализует, тот же эффект.axiosСледует сказать, что в сценарии с перехватчиком большое внимание уделялось производительности.Поскольку перехватчики определяются пользователем, теоретически их может быть бесконечное количество, поэтому необходимо учитывать производительность..
Смотри нижеaxiosПример кода перехватчика:
// 代码有删减
// 声明
this.handlers = [];
// 移除
if (this.handlers[id]) {
this.handlers[id] = null;
}
// 执行
if (h !== null) {
fn(h);
}
3.19 hasOwn определяет, является ли это его собственным свойством
/**
* Check whether an object has the property.
*/
var hasOwnProperty = Object.prototype.hasOwnProperty;
function hasOwn (obj, key) {
return hasOwnProperty.call(obj, key)
}
// 例子:
// 特别提醒:__proto__ 是浏览器实现的原型写法,后面还会用到
// 现在已经有提供好几个原型相关的API
// Object.getPrototypeOf
// Object.setPrototypeOf
// Object.isPrototypeOf
// .call 则是函数里 this 显示指定以为第一个参数,并执行函数。
hasOwn({__proto__: { a: 1 }}, 'a') // false
hasOwn({ a: undefined }, 'a') // true
hasOwn({}, 'a') // false
hasOwn({}, 'hasOwnProperty') // false
hasOwn({}, 'toString') // false
// 是自己的本身拥有的属性,不是通过原型链向上查找的。
3.20 кэшированный кеш
Используйте функцию закрытия для кэширования данных
/**
* Create a cached version of a pure function.
*/
function cached (fn) {
var cache = Object.create(null);
return (function cachedFn (str) {
var hit = cache[str];
return hit || (cache[str] = fn(str))
})
}
Регулярная рекомендация по обучению системыЛао Яо: «Мини-книга по регулярным выражениям JavaScript» вышла!, все сказали хорошо. Поэтому в этой статье не будут подробно описываться знания, связанные с регуляризацией.
3.21 верблюжий дефис вместо маленького верблюда
Дефис - CamelCase по клику => onClick
/**
* Camelize a hyphen-delimited string.
*/
var camelizeRE = /-(\w)/g;
var camelize = cached(function (str) {
return str.replace(camelizeRE, function (_, c) { return c ? c.toUpperCase() : ''; })
});
3.22 использовать заглавную первую букву
Сделать первую букву заглавной
/**
* Capitalize a string.
*/
var capitalize = cached(function (str) {
return str.charAt(0).toUpperCase() + str.slice(1)
});
3.23 дефис Небольшой горб к дефису
onClick => on-click
/**
* Hyphenate a camelCase string.
*/
var hyphenateRE = /\B([A-Z])/g;
var hyphenate = cached(function (str) {
return str.replace(hyphenateRE, '-$1').toLowerCase()
});
3.24 прокладка для привязки polyfillBind
/**
* Simple bind polyfill for environments that do not support it,
* e.g., PhantomJS 1.x. Technically, we don't need this anymore
* since native bind is now performant enough in most browsers.
* But removing it would mean breaking code that was able to run in
* PhantomJS 1.x, so this must be kept for backward compatibility.
*/
/* istanbul ignore next */
function polyfillBind (fn, ctx) {
function boundFn (a) {
var l = arguments.length;
return l
? l > 1
? fn.apply(ctx, arguments)
: fn.call(ctx, a)
: fn.call(ctx)
}
boundFn._length = fn.length;
return boundFn
}
function nativeBind (fn, ctx) {
return fn.bind(ctx)
}
var bind = Function.prototype.bind
? nativeBind
: polyfillBind;
Проще говоря, он совместим со старыми версиями браузеров, которые не поддерживают нативные.bindфункция. В то же время он совместим с методом записи, и оценивается количество параметров, и использованиеcallа такжеapplyРеализация, говорят, что параметры подходят для использованияapply,бесполезныйcallЛучшая производительность.
Если дляcall、apply、bindНезнакомые с использованием и реализацией, вы можете проверить мойИнтервьюер спрашивает сериюнаписано наИнтервьюер спросил: Могу ли я смоделировать вызов и применить методы JS?
Интервьюер спросил: Могу ли я смоделировать метод привязки, реализующий JS?
3.25 toArray преобразует массив, подобный массиву, в настоящий массив.
Преобразует массив, подобный массиву, в массив, поддерживает начало и по умолчанию начинает с 0.
/**
* Convert an Array-like object to a real Array.
*/
function toArray (list, start) {
start = start || 0;
var i = list.length - start;
var ret = new Array(i);
while (i--) {
ret[i] = list[i + start];
}
return ret
}
// 例子:
function fn(){
var arr1 = toArray(arguments);
console.log(arr1); // [1, 2, 3, 4, 5]
var arr2 = toArray(arguments, 2);
console.log(arr2); // [3, 4, 5]
}
fn(1,2,3,4,5);
3.26 расширение слияния
/**
* Mix properties into target object.
*/
function extend (to, _from) {
for (var key in _from) {
to[key] = _from[key];
}
return to
}
// 例子:
const data = { name: '若川' };
const data2 = extend(data, { mp: '若川视野', name: '是若川啊' });
console.log(data); // { name: "是若川啊", mp: "若川视野" }
console.log(data2); // { name: "是若川啊", mp: "若川视野" }
console.log(data === data2); // true
3.27 toObject к объекту
/**
* Merge an Array of Objects into a single Object.
*/
function toObject (arr) {
var res = {};
for (var i = 0; i < arr.length; i++) {
if (arr[i]) {
extend(res, arr[i]);
}
}
return res
}
// 数组转对象
toObject(['若川', '若川视野'])
// {0: '若', 1: '川', 2: '视', 3: '野'}
3.28 пустая функция noop
/* eslint-disable no-unused-vars */
/**
* Perform no operation.
* Stubbing args to make Flow happy without leaving useless transpiled code
* with ...rest (https://flow.org/blog/2017/05/07/Strict-Function-Call-Arity/).
*/
function noop (a, b, c) {}
// 初始化赋值
3.29 no всегда возвращает false
/**
* Always return false.
*/
var no = function (a, b, c) { return false; };
/* eslint-enable no-unused-vars */
3.30 identity возвращает сам параметр
/**
* Return the same value.
*/
var identity = function (_) { return _; };
3.31 genStaticKeys генерирует статические свойства
/**
* Generate a string containing static keys from compiler modules.
*/
function genStaticKeys (modules) {
return modules.reduce(function (keys, m) {
return keys.concat(m.staticKeys || [])
}, []).join(',')
}
3.32 свободное равенство свободное равенство
Поскольку массивы, объекты и т. д. являются ссылочными типами, два содержимого кажутся равными, а строгое равенство не равно.
var a = {};
var b = {};
a === b; // false
a == b; // false
Итак, эта функция представляет собой рекурсивное сравнение массивов, дат и объектов. Свободное равенство, если содержимое точно равно.
/**
* Check if two values are loosely equal - that is,
* if they are plain objects, do they have the same shape?
*/
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 (a instanceof Date && b instanceof Date) {
return a.getTime() === b.getTime()
} 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
}
}
3.33 свободный индекс
Эта функция реализует свободное равенство. оригинальныйindexOfстрого равны.
/**
* Return the first index at which a loosely equal value can be
* found in the array (if value is a plain object, the array must
* contain an object of the same shape), or -1 if it is not present.
*/
function looseIndexOf (arr, val) {
for (var i = 0; i < arr.length; i++) {
if (looseEqual(arr[i], val)) { return i }
}
return -1
}
3.34 Once гарантирует, что функция будет выполнена только один раз.
Используйте замыкания для хранения состояния
/**
* Ensure a function is called only once.
*/
function once (fn) {
var called = false;
return function () {
if (!called) {
called = true;
fn.apply(this, arguments);
}
}
}
const fn1 = once(function(){
console.log('哎嘿,无论你怎么调用,我只执行一次');
});
fn1(); // '哎嘿,无论你怎么调用,我只执行一次'
fn1(); // 不输出
fn1(); // 不输出
fn1(); // 不输出
3.35 Жизненный цикл и т. д.
var SSR_ATTR = 'data-server-rendered';
var ASSET_TYPES = [
'component',
'directive',
'filter'
];
[Vue 生命周期](https://cn.vuejs.org/v2/api/#%E9%80%89%E9%A1%B9-%E7%94%9F%E5%91%BD%E5%91%A8%E6%9C%9F%E9%92%A9%E5%AD%90)
var LIFECYCLE_HOOKS = [
'beforeCreate',
'created',
'beforeMount',
'mounted',
'beforeUpdate',
'updated',
'beforeDestroy',
'destroyed',
'activated',
'deactivated',
'errorCaptured',
'serverPrefetch'
];
4. Напоследок порекомендуйте несколько статей и книг
Эта часть совпадает со статьей о функциях инструмента Vue3, рекомендуется, поэтому скопируйте ее сюда.
Я рекомендую сначала я думаю, что это хорошоJavaScript APIнесколько статей и несколько книг, которые стоит прочитать.
Все API JavaScript String целая дешифровка
[Подробная длинная статья] Все API-интерфейсы массивов JavaScript полностью расшифрованы
Руководство пользователя внешнего интерфейса регулярных выражений
Лао Яо: «Мини-книга по регулярным выражениям JavaScript» вышла!
Lao Yao Talks: Как выучить JavaScript?
Весь API-анализ объектов JavaScript Биография Лу Синя 12.git ee.IO/ is -object- ах...
Продвинутое программирование с помощью JavaScript, 4-е издание
Полное руководство по JavaScript, 7-е издание
«Объектно-ориентированное программирование JavaScript 2»Объектно-ориентированный очень подробно.
Учитель Руан Ифэн: «Вводный курс по ES6»
Учебник по современному JavaScript
«JavaScript, которого вы не знаете», том 1
Шаблоны проектирования JavaScript и практика разработки
Я также испытал это, так как я не мог читать книги. Пишите статьи, чтобы поделиться.
Мой способ чтения книг: читать несколько книг одновременно и читать одни и те же и похожие главы, например функции. Прочитав эту книгу, вы можете ее не понять.Прочитав следующую книгу, вы в основном поймете ее после прочтения нескольких книг.Если вы не поняли ее один раз, прочитайте ее несколько раз, чтобы не забыть и закрепить соответствующие главы. Конечно, сначала было тяжело читать книгу, да и читать было сложно. Большинство из этих книг доступны для чтения в WeChat.Если вы привыкли читать бумажные книги, вы можете купить их.
Здесь вы можете посмотреть несколько видео и попрактиковаться в некоторых простых проектах.
Например: вы можете зарегистрироваться самостоятельноgithubАккаунт, разделенный на главы и подразделы, скопируйте код в книгу, отправьте наgithub, чем больше вы практикуетесь, тем больше вы чувствуете.
Другой примерСайт онлайн-обучения китайскому языку freeCodeCampВеб-сайт. Чтение книг — очень хороший способ систематического обучения. Позже я просто больше смотрел на исходный код и писал статьи, которыми делился со всеми.
5. Резюме
Эта статья после просмотраVue2в исходном кодеsharedмодульУпакованная строка dist/vue.js с 14 по 379..Исходный код не такой уж сложный, по крайней мере многое можно понять, например, функции инструментов. Трудность может быть трудностью: я не знаю сценария применения.
Vue2Функции инструментов хорошо названы, например:isсудить,toконвертировать,hasЕсть ли, чтобы разработчики могли сразу увидеть семантику функции.
Эти функции также очень одиночные, в основном функция делает только одну вещь.
Читателям и друзьям рекомендуется писать незнакомые функции от руки, что поможет закрепить базовые знания и заполнить пробелы.
Наконец, вы можете продолжать следовать за мной @ Ruo Chuan. добро пожаловать, чтобы добавить меня wechatruochuan12общаться, участвоватьЧитайте исходный код вместеДействия, все вместе изучают исходный код и вместе добиваются прогресса.
О группе && Exchange
недавно организованныйЧтение исходного кода, если вам интересно, вы можете добавить меня в WeChatruochuan12Участвуйте в долгосрочном обмене и обучении.
Автор: Чанг ИВакагаваНазвание смешано в реках и озерах. добро пожаловать, чтобы добавить меня wechatruochuan12. По дороге на фронт | Знаю очень мало, только хорошо учусь.
Обратите внимание на паблик аккаунта Ruochuan Vision, Изучайте исходный код вместе каждую неделю, учитесь читать исходный код и переходите к расширенному интерфейсу.
Блог Вакагавы
segmentfaultКолонна Вакагава Видение, открылВидение ВакагаваКолонка, добро пожаловать на внимание ~
Колонка самородков, добро пожаловать, обратите внимание~
Колонна видений Чжиху Руочуань, открылВидение ВакагаваКолонка, добро пожаловать на внимание ~
github blog, спроситьstar^_^~