underscore.js 解读(敲源码)

2024-05-14 14:32
文章标签 源码 js 解读 underscore

本文主要是介绍underscore.js 解读(敲源码),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

我把第一遍敲的underscore的代码放在github上了,加上了备注和一些demo 感兴趣的可以去看看。
https://github.com/katoto/copyUnderscore

还有一个是和jQuery 相关的,也可以敲敲看。
https://github.com/katoto/Tur_jq

Underscore一个JavaScript实用库,提供了一整套函数式编程的实用功能。正因为如此,所以它的每一个方法都相对独立。适合初学者敲一敲,也可以学习学习。下面我将贴一些,自己认为比较重要的函数。

undersocre 如何避免作用域污染?

;(function(){var root = this;var previousUnderscore = root._;var Ctor = function(){};    var _= function(obj){                  //  ☆if(obj instanceof _)return obj;if(!(this instanceof _))return new _(obj);this._wrapped = obj;};  //          释放 _ 的控制权 要在_ 声明的后面_.noConflict = function(){root._ = previousUnderscore;return this;}if(typeof exports !== 'undefined'){if(typeof module !== 'undefined'&& module.exports){exports = module.exports = _;}exports._  = _;}else{root._ = _;}_.VERSION ='0.01'
//          传入全局变量 window or node
}.call(this))

jQuery如何避免作用域污染?

;(function(window ,undefined){
//Tur_jq 构造函数    定义undefined 是为了兼容性var Tur_jq = function Tur_jq ( selector ){return new Tur_jq.prototype.init( selector );};Tur_jq.fn = Tur_jq.prototype = {constructor:Tur_jq,selector:null,length:0,version:0.01,init:function( selector ) {//是空的和undefined 直接返回if (!selector) return this;// 字符串:选择器,htmlif (Tur_jq.isString(selector)) {if (selector.charAt(0) === '<') {Tur_jq.push.apply(this, Tur_jq.parseHTML(selector));} else {Tur_jq.push.apply(this, Tur_jq.select(selector));this.selector = selector;}return this;}//DOM 对象if (Tur_jq.isDOM( selector )){this[0] = selector;this.length =1;return this;}//Tur_jq 对象if( Tur_jq.isTur_jq( selector )){return selector;}// DOM 数组if( Tur_jq.isLikeArray( selector )){Tur_jq.push.apply( this ,selector );return this;}// 如果是函数的话 就是入口函数if( Tur_jq.isFunction( selector ) ){var oldFn = window.onload;if( typeof oldFn ==='function' ){window.onload = function(){oldFn();selector();};}else {window.onload = selector;}}},each:function( callback  ){Tur_jq.each( this ,callback );return this;}};Tur_jq.fn.init.prototype = Tur_jq.prototype;//可扩展Tur_jq.extend = Tur_jq.fn.extend = function( obj ){var k ;for( k in obj  ){this[k] = obj [k];}};
//  比如你要扩展函数Tur_jq.extend({isFunction:function(obj){return typeof obj === 'function';},isString:function( obj ){return typeof  obj ==='string';},isLikeArray: function( obj ){return obj && obj.length && obj.length >=0;},isTur_jq:function( obj ){return 'selector' in obj;},isDOM:function( obj ){return !!obj.nodeType;}});
//  同理就一直往下.....// 对外公开window.$ = window.jQurey = Tur_jq;
})( window );

underscore 对内置函数的处理

//实现更少的字节和作用域链查找   (可以再浏览数上看原型上有什么内置的方法)var ArrayProto = Array.prototype,ObjProto = Object.prototype,FuncProto = Function.prototype;var push = ArrayProto.push,slice = ArrayProto.slice,toString = ObjProto.toString,hasOwnProperty = ObjProto.hasOwnProperty;
//      ECMAScript 5 的原生方法  keys? bind?var nativeIsArray = Array.isArray,nativeKeys = Object.keys,nativeBind = FuncProto.bind,nativeCreate = Object.create;

而jQuery 一开始就会对当前的浏览器进行能力检测的。
下面贴一些 我认为很好的功能函数(有依赖,去看看源码就懂了)

//          洗牌算法       ☆_.shuffle = function(obj){var set = isArrayLike(obj)?obj:_.values(obj);var length = set.length,shuffled = Array(length);for(var index=0,rand;index<length;index++){rand = _.random(0,index);if(rand !== index) shuffled[index]=shuffled[rand];shuffled[rand]= set[index];}return shuffled;};
//      判断是否是isArray     ☆_.isArray = nativeIsArray || function(obj){return toString.call(obj) ==='[object Array]';}
//机智写法
_.each(['Arguments','Function','String','Number','Date','RegExp','Error'],function(name){_['is'+name]= function(obj){return toString.call(obj)=== '[object '+name+']' ;};});
//      ☆    nodeType 来判断   _.isElement = function(obj){return !!(obj && obj.nodeType ===1 );};
        _.isObject = function(obj){var type = typeof obj;//  0 || 1&& 1   1    0 || 1&& 0   0    return type ==='function'|| type==='object'&& !!obj}
//      !!!!  模板字符串           得认真看看才行_.templateSetting = {evaluate :/<%([\s\S]+?)%>/g,interpolate:/<%=([\s\S]+?)%>/g,escape :/<%-([\s\S]+?)%>/g};var noMatch = /(.)^/;var escapes = {"'":"'",'\\':'\\','\r':'r','\n':'n','\u2028':'u2028','\u2029':'u2029'};var escapeChar = function(match){return '\\'+escapes[match];}_.template = function(text,settings,oldSettings){if(!settings && oldSettings)settings = oldSettings;settings = _.defaults({},settings,_.templateSettings);var matcher = RegExp([(settings.escape || noMatch).source,(settings.interpolate|| noMatch).source,(settings.evaluate || noMatch).source].join('|')+'|$','g');var index = 0;var source = "__p+='";text.replace(matcher,function(match,escape,interpolate,evaluate,offset){source += text.slice(index,offset).replace(escaper,escapeChar);index = offset + match.length;if(escape){source +="'+\n((__t("+escape+"))==null?'':_.escape(__t))+\n'";}else if(interpolate){source +="'+\n((__t=("+interpolate+"))==null?'':__t)+\n'";}else if(evaluate){source +="';\n"+evaluate+"\n__p+='";}return match;})if(!settings.variable)source = 'with(obj||{}){\n'+source+'\n}';source = "var __t,__p='',__j=Array.prototype.join,"+"print=function(){__p+=__j.call(arguments,'');};\n"+source +'return __p;\n';try{var render = new Function(settings.variable || 'obj','_',source);}catch(e){e.source = source;throw e;}var template = function(data){return render.call(this,data,_);};var argument = settings.variable || 'obj';template.source = 'function('+argument+'){\n'+source+'}';return template;            }
//      Object.prototype.toString.call(2) // "[object Number]"
//      Object.prototype.toString.call('') // "[object String]"
//      Object.prototype.toString.call(true) // "[object Boolean]"
//      Object.prototype.toString.call(undefined) // "[object Undefined]"
//      Object.prototype.toString.call(null) // "[object Null]"
//      Object.prototype.toString.call(Math) // "[object Math]"
//      Object.prototype.toString.call({}) // "[object Object]"
//      Object.prototype.toString.call([]) // "[object Array]"

本想写的更好一点的,不过实在不懂如何下手,感觉也没人看,就算了。

underscore 里有个template 详细的可以查看这篇博客:

http://blog.csdn.net/fendouzhe123/article/details/39083367

详情可查看:
http://www.css88.com/doc/underscore/

阮一峰资料:
http://javascript.ruanyifeng.com/library/underscore.html

end 更多请敲源码!

这篇关于underscore.js 解读(敲源码)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/988999

相关文章

JS+HTML实现在线图片水印添加工具

《JS+HTML实现在线图片水印添加工具》在社交媒体和内容创作日益频繁的今天,如何保护原创内容、展示品牌身份成了一个不得不面对的问题,本文将实现一个完全基于HTML+CSS构建的现代化图片水印在线工具... 目录概述功能亮点使用方法技术解析延伸思考运行效果项目源码下载总结概述在社交媒体和内容创作日益频繁的

Node.js 数据库 CRUD 项目示例详解(完美解决方案)

《Node.js数据库CRUD项目示例详解(完美解决方案)》:本文主要介绍Node.js数据库CRUD项目示例详解(完美解决方案),本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考... 目录项目结构1. 初始化项目2. 配置数据库连接 (config/db.js)3. 创建模型 (models/

python3 gunicorn配置文件的用法解读

《python3gunicorn配置文件的用法解读》:本文主要介绍python3gunicorn配置文件的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录python3 gunicorn配置文件配置文件服务启动、重启、关闭启动重启关闭总结python3 gun

关于pandas的read_csv方法使用解读

《关于pandas的read_csv方法使用解读》:本文主要介绍关于pandas的read_csv方法使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录pandas的read_csv方法解读read_csv中的参数基本参数通用解析参数空值处理相关参数时间处理相关

使用Node.js制作图片上传服务的详细教程

《使用Node.js制作图片上传服务的详细教程》在现代Web应用开发中,图片上传是一项常见且重要的功能,借助Node.js强大的生态系统,我们可以轻松搭建高效的图片上传服务,本文将深入探讨如何使用No... 目录准备工作搭建 Express 服务器配置 multer 进行图片上传处理图片上传请求完整代码示例

Java调用C++动态库超详细步骤讲解(附源码)

《Java调用C++动态库超详细步骤讲解(附源码)》C语言因其高效和接近硬件的特性,时常会被用在性能要求较高或者需要直接操作硬件的场合,:本文主要介绍Java调用C++动态库的相关资料,文中通过代... 目录一、直接调用C++库第一步:动态库生成(vs2017+qt5.12.10)第二步:Java调用C++

用js控制视频播放进度基本示例代码

《用js控制视频播放进度基本示例代码》写前端的时候,很多的时候是需要支持要网页视频播放的功能,下面这篇文章主要给大家介绍了关于用js控制视频播放进度的相关资料,文中通过代码介绍的非常详细,需要的朋友可... 目录前言html部分:JavaScript部分:注意:总结前言在javascript中控制视频播放

java之Objects.nonNull用法代码解读

《java之Objects.nonNull用法代码解读》:本文主要介绍java之Objects.nonNull用法代码,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录Java之Objects.nonwww.chinasem.cnNull用法代码Objects.nonN

Python实现无痛修改第三方库源码的方法详解

《Python实现无痛修改第三方库源码的方法详解》很多时候,我们下载的第三方库是不会有需求不满足的情况,但也有极少的情况,第三方库没有兼顾到需求,本文将介绍几个修改源码的操作,大家可以根据需求进行选择... 目录需求不符合模拟示例 1. 修改源文件2. 继承修改3. 猴子补丁4. 追踪局部变量需求不符合很

SpringCloud负载均衡spring-cloud-starter-loadbalancer解读

《SpringCloud负载均衡spring-cloud-starter-loadbalancer解读》:本文主要介绍SpringCloud负载均衡spring-cloud-starter-loa... 目录简述主要特点使用负载均衡算法1. 轮询负载均衡策略(Round Robin)2. 随机负载均衡策略(