preface
Hi, I’m Wakawa. This is the second chapter to learn the overall architecture of source code. Overall architecture this word seems to be a bit big, let’s say it is the overall structure of the source code, the main is to learn the overall structure of the code, do not go into other is not the main line of the specific function implementation. This article is about code packaged and integrated, not code broken down in the actual repository.
Learn the overall architecture of the source code series articles as follows:
1. Learn the overall architecture of jQuery source code, and create your own JS class library 2. Learn to use the whole framework of the source code to create their own functional programming library 3. Learning loDASH source code overall architecture, to create their own functional programming class library 4. Learn sentry source code overall architecture, build their own front-end exception monitoring SDK 5. Learn vuEX source code overall architecture, to create their own state management library 6. Learning axiOS source code overall architecture, to create their own request library 7. Learning koA source code overall architecture, a brief analysis of KOA onion model principle and CO principle 8. Learn redux source code overall architecture, in-depth understanding of Redux and its middleware principles
Interested readers can click to read.
I’ve read a lot of articles about underscore. Js analysis classes, but I always feel like there’s something missing. This may be the paper come zhongjue shallow, must know this matter to practice. I decided to write an article about the overall architecture of underscore.js.
Version of this article is V1.9.1. Source address: unpkg.com https://unpkg.com/[email protected]/underscore.js
While most people have never used underscore.js, a look at the official documentation should tell you how to use it.
Start with a simple example from the official document _.chain:
_.chain([1, 2, 3]).reverse().value();
// => [3, 2, 1]
Copy the code
As you can see from the example, this supports chained calls.
Readers can also follow the idea of the article, open the download source code for debugging, so more impressive.
Chain calls
_. Chain function source:
_.chain = function(obj) {
var instance = _(obj);
instance._chain = true;
return instance;
};
Copy the code
The simple function is to call _() by passing obj. But the return value variable turns out to be the instance instance object. Add attribute _chain with value true and return the intance object. But in this example, the instance object can actually call the reverse method and then call the value method. Guess OOP (object oriented) calls are supported.
With the question, I looked at the code that defines the _ function object.
_
Function object supportOOP
var _ = function(obj) {
if (obj instanceof _) return obj;
if(! (this instanceof _))return new _(obj);
this._wrapped = obj;
};
Copy the code
Return obj if the argument obj is already an instance of _. If this is not an instance of _, then manually new _(obj); When we call new again, we assign the object obj to the _wrapped property. {_wrapped: ‘obj’,} __proto__ is _(obj).__proto__ is _. Prototype;
For those of you who are unfamiliar with this area, take a look at the following diagramThe interviewer asks:JS inheritance
Drawing).
Continue with the official _.chain example. This example is broken down and written in three steps.
var part1 = _.chain([1, 2, 3]);
var part2 = part1.reverse();
var part3 = part2.value();
// There is no subsequent part1.reverse() operationconsole.log(part1); // {__wrapped: [1, 2, 3], _chain: true} console.log(part2); // {__wrapped: [3, 2, 1], _chain: true} console.log(part3); / / [3, 2, 1)Copy the code
Reverse is an Array. Prototype method. Why do we support chained calls? Search for Reverse, and you’ll see the following code:
Plug the example into this code and you get:
_. Chain ([1, 2, 3]). The reverse (). The value ()Copy the code
var ArrayProto = Array.prototype;
// Iterates over the array. prototype methods and assigns them to _.prototype_.each(['pop'.'push'.'reverse'.'shift'.'sort'.'splice'.'unshift'].function(name) {
// Here 'method' is the reverse function var method = ArrayProto[name];
_.prototype[name] = function() { // 这里的obj 就是数组 [1, 2, 3] var obj = this._wrapped; // Arguments is the set of arguments, specify the reverse this to obj, argument to arguments, and execute the function. Obj is [3, 2, 1] method.apply(obj, arguments); if ((name === 'shift' || name === 'splice') && obj.length === 0) delete obj[0]; // The chainResult function is important here. return chainResult(this, obj); }; }); Copy the code
// Helper function to continue chaining intermediate results.
var chainResult = function(instance, obj) {
// If the instance has _chain astrueReturns the instance object {_chain that supports chained calls:true_wrapped: [3, 2, 1]}, this._wrapped: [3, 2, 1]} return instance._chain ? _(obj).chain() : obj;
};
Copy the code
if ((name === ‘shift’ || name === ‘splice’) && obj.length === 0) delete obj[0]; Mention the above source code in this sentence, see this sentence is puzzled. So search github for the phrase with “” in double quotes. Represents all searches.
Search for two issues in the official library, probably means compatible with IE low version of writing. If you’re interested, you can click on it.
I don’t understand the meaning of this sentence.
why delete obj[0]
Flow-based programming
That completes the analysis of the chained call _.chain() and _ function objects. {_wrapped: ”, _chain: true} {_wrapped: ”, _chain: true} {_wrapped: ”, _chain: true} {wrapped: ”, _chain: true} {wrapped: ”, _chain: true}} {wrapped: ”, _chain: true}} This is called stream-based programming.
And then at the end of the data processing, what do I do to return this data. Underscore provides a value method.
_.prototype.value = function() { return this._wrapped;
}
Copy the code
Several aliases are provided in passing. ToJSON, the valueOf. _.prototype.valueOf = _.prototype.toJSON = _.prototype.value;
The toString method is also provided.
_.prototype.toString = function() {
return String(this._wrapped);
};
Copy the code
String() has the same effect as new String(). You can guess that the internal implementation is similar to the _ function object.
var String = function() { if(! (this instanceOf String))return new String(obj);
}
Copy the code
var chainResult = function(instance, obj) {
return instance._chain ? _(obj).chain() : obj;
};
Copy the code
The careful reader will notice how _(obj).chain() in the chainResult function implements chained calls.
{_wrapped: obj} _(obj) {_wrapped: obj} How can there be a chain() method? There must be somewhere to mount the method on _.prototype or something, and that’s _.mixin().
_.mixin
Mount all static methods to_.prototype
, or you can mount custom methods
_. Mixed with mixins. But it is too intrusive and often prone to problems such as coverage. Remember when React had mixins and Vue had mixins. However, after the version iteration and update, mixins are not recommended or supported.
_.mixin = function(obj) {
// Iterate over all methods on the object _.each(_.functions(obj), function(name) {
// for example, chain, obj['chain'] function, which is custom, is assigned to _[name]. Func is the function. That is, custom methods are available not only on _ function objects, but also on '_. Prototype' var func = _[name] = obj[name];
_.prototype[name] = function() { // The data object to be processed var args = [this._wrapped]; // Combine arguments with data objects to process push.apply(args, arguments); // The chain-apply (_, args) argument has been added with the _chain attribute to support chain-calling. // _.chain = function(obj) { // var instance = _(obj); // instance._chain = true; // return instance; }; return chainResult(this, func.apply(_, args)); }; }); // Finally return _ function object. return _; }; _.mixin(_); Copy the code
_mixin(_) mounts static methods to _.prototype, which is the _.prototype.chain method, which is the _.chain method.
So _(obj).chain(obj) has the same effect as _(obj).chain().
On the above chain calls, the author drew a picture, the so-called picture is worth a thousand words.
_.mixin mounts custom methods
Mount custom methods: Here is an example:
_.mixin({
log: function() { console.log('Oops, I've been called.');
}
})
_.log() // Oops, I'm being called_ ().log() // Oops, I've been calledCopy the code
_.functions(obj)
_.functions = _.methods = function(obj) {
var names = [];
for (var key in obj) {
if (_.isFunction(obj[key])) names.push(key);
}
return names.sort(); }; Copy the code
_. Functions and _. Methods, which iterate over the methods on the object, put them in an array, and sort them. Returns the sorted array.
underscore.js
What the hell_
and_.prototype
How many methods and properties are mounted
Let’s take a look at how many static methods and properties underscore.js mounts on the _ function object, and how many methods and properties underscore.prototype mounts.
Try it out using the for in loop. Look at this code:
var staticMethods = [];
var staticProperty = [];
for(var name in_) { if(typeof _[name] === 'function') { staticMethods.push(name);
} else{ staticProperty.push(name); } } console.log(staticProperty); / / /"VERSION"."templateSettings"] twoconsole.log(staticMethods); / / /"after"."all"."allKeys"."any"."assign". ] 138Copy the code
var prototypeMethods = [];
var prototypeProperty = [];
for(var name in _.prototype){
if(typeof _.prototype[name] === 'function') { prototypeMethods.push(name);
} else{ prototypeProperty.push(name); } } console.log(prototypeProperty); / / []console.log(prototypeMethods); / / /"after"."all"."allKeys"."any"."assign". ] 152Copy the code
Underscore: underscore: underscore: underscore: underscore: underscore: underscore: underscore: underscore: underscore
Overview of overall architecture
Anonymous function self-executing
(function() {
} ());Copy the code
Such assurance does not pollute the external environment, at the same time isolation of the external environment, is not external influence internal environment.
The outside world can’t access the variables and functions inside, inside can access the outside variables, but it defines its own variables, will not access the outside variables. Anonymous functions wrap the code in, preventing it from colliding with other code and contaminating the global environment. Readers who are not familiar with self-executing functions can refer to this article. Execute function expressions immediately (IIFE)
Root processing
var root = typeof self == 'object' && self.self === self && self ||
typeof global == 'object' && global.global === global && global ||
this ||
{};
Copy the code
Support browser environment, Node, Web Worker, Node VM, wechat applet.
export
if(typeof exports ! ='undefined' && !exports.nodeType) {
if(typeof module ! ='undefined' && !module.nodeType && module.exports) {
exports = module.exports = _;
}
exports._ = _;
} else { root._ = _; } Copy the code
Regarding root processing and export of these two sections of code explanation, recommend to see this article Hu: underscore series how to write their own underscore, speak really too good. I will not repeat here. All in all, the UI. Js writers didn’t do these things overnight, but they also accumulated slowly and improved after the ISSUE was mentioned by others.
supportamd
Modularity specification
if (typeof define == 'function' && define.amd) {
define('underscore'[],function() {
return _;
});
}
Copy the code
_. NoConflict Anti-conflict function
Source:
// Assign the value back when noConflict is executedvar previousUnderscore = root._;
_.noConflict = function() {
root._ = previousUnderscore;
return this;
}; Copy the code
Use:
<script>
var _ = 'I am what I am, not the same fireworks, other don't cover me.';
</script>
<script src="https://unpkg.com/[email protected]/underscore.js">
</script>
<script> var underscore = _.noConflict(); console.log(_); // 'I am what I am, not the same fireworks, other don't cover me.' underscore.isArray([]) // true </script> Copy the code
conclusion
_.chain([1, 2, 3]).reverse().value(); More in-depth debugging and tracing code, analysis of chained calls (_.chain() and _(obj).chain()), OOP, streaming based programming, and _.mixin(_) mount methods in _.prototype, and finally overall architecture analysis. Learn the overall architecture of underscore. Js to help you create your own functional programming library.
Article analysis of the source code overall structure.
(function() {
var root = typeof self == 'object' && self.self === self && self ||
typeof global == 'object' && global.global === global && global ||
this ||
{};
var previousUnderscore = root._; var _ = function(obj) { if (obj instanceof _) return obj; if(! (this instanceof _))return new _(obj); this._wrapped = obj; }; if(typeof exports ! ='undefined' && !exports.nodeType) { if(typeof module ! ='undefined' && !module.nodeType && module.exports) { exports = module.exports = _; } exports._ = _; } else { root._ = _; } _.VERSION = '1.9.1'; _.chain = function(obj) { var instance = _(obj); instance._chain = true; return instance; }; var chainResult = function(instance, obj) { return instance._chain ? _(obj).chain() : obj; }; _.mixin = function(obj) { _.each(_.functions(obj), function(name) { var func = _[name] = obj[name]; _.prototype[name] = function() { var args = [this._wrapped]; push.apply(args, arguments); return chainResult(this, func.apply(_, args)); }; }); return _; }; _.mixin(_); _.each(['pop'.'push'.'reverse'.'shift'.'sort'.'splice'.'unshift'].function(name) { var method = ArrayProto[name]; _.prototype[name] = function() { var obj = this._wrapped; method.apply(obj, arguments); if ((name === 'shift' || name === 'splice') && obj.length === 0) delete obj[0]; return chainResult(this, obj); }; }); _.each(['concat'.'join'.'slice'].function(name) { var method = ArrayProto[name]; _.prototype[name] = function() { return chainResult(this, method.apply(this._wrapped, arguments)); }; }); _.prototype.value = function() { return this._wrapped; }; _.prototype.valueOf = _.prototype.toJSON = _.prototype.value; _.prototype.toString = function() { return String(this._wrapped); }; if (typeof define == 'function' && define.amd) { define('underscore'[],function() { return _; }); } } ());Copy the code
The next article is to learn the overall source architecture of Lodash. Learn loDASH source code overall architecture, to create their own functional programming library
Readers are welcome to comment on any problems or improvements they find. In addition, I think it is well written. I can like, comment and forward it, which is also a kind of support for the author.
Recommended reading
Underscore — Underscore — Underscore — Underscore — Underscore
The author’s previous article
Q: Can you simulate the JS call and apply method? Q: Can you simulate the JS bind method? Can simulate the implementation of the JS new operator front end using puppeteer crawler to generate the React. JS small book PDF and merge
about
Author: often with the name of Ruochuan mixed in rivers and lakes. The front road lovers | | PPT know little, only good study. Ruakawa’s blog has been reconstructed by Vuepress, and the reading experience may be many gold-digging columns. Welcome to pay attention to the front view column of ~ SegmentFault, open the front view column, welcome to pay attention to the front view column of ~ Zhihu, open the front view column, welcome to pay attention to the front view column of ~ Chatfinch, add the column of Chatfinch, Welcome to follow ~ github blog, related source code and resources are put here, ask a star^_^~
Wechat public account Ruochuan Vision
May be more interesting wechat public number, long press scan code concern. You can also add wechatruochuan12
, indicate the source, pull you into [front view communication group].
This article was typeset using MDNICE