十三、jQuery过时的今天,你还会使用它吗

简介: 早几年学习前端,大家都非常热衷于研究jQuery源码。我至今还记得当初从jQuery源码中学到一星半点应用技巧的时候常会有一种发自内心的惊叹,“原来JavaScript居然可以这样用!”但是随着前端的迅猛发展,另外几种前端框架的崛起,jQuery慢慢变得不再是必须。所有人对jQuery的热情都降低了许多。jQuery在前端史上有它非常超然的历史地位,许多从中学到的技巧在实践开发中仍然非常好用。简单的了解它有助于我们更加深入的理解JavaScript。如果你能够从中看明白jquery是如何一步步被取代的,那么,我想你的收益远不止学会使用了一个库那么简单。

早几年学习前端,大家都非常热衷于研究jQuery源码。


我至今还记得当初从jQuery源码中学到一星半点应用技巧的时候常会有一种发自内心的惊叹,“原来JavaScript居然可以这样用!”


但是随着前端的迅猛发展,另外几种前端框架的崛起,jQuery慢慢变得不再是必须。所有人对jQuery的热情都降低了许多。


jQuery在前端史上有它非常超然的历史地位,许多从中学到的技巧在实践开发中仍然非常好用。简单的了解它有助于我们更加深入的理解JavaScript。如果你能够从中看明白jquery是如何一步步被取代的,那么,我想你的收益远不止学会使用了一个库那么简单。


因此,我的态度是,项目中你可以不用,但是我仍然建议你学。


这篇文章的主要目的,是从面向对象的角度,跟大家分享jquery对象是如何封装的。算是对大家进一步学习jQuery源码的一个抛砖引玉。


1


使用jQuery对象时,我们这样写:


// 声明一个jQuery对象
$('.target')
// 获取元素的css属性
$('.target').css('width')
// 获取元素的位置信息
$('.target').offset()


在使用之初可能会有许多疑问,比如$是怎么回事?为什么不用new就可以直接声明一个对象?等等。了解之后,才知道原来这正是jQuery对象创建的巧妙之处。


先直接用代码展示出来,再用图跟大家解释是怎么回事。


;
(function (ROOT) {
  // 构造函数
  var jQuery = function (selector) {
    // 在jQuery中直接返回new过的实例,这里的init是jQuery的真正构造函数
    return new jQuery.fn.init(selector)
  }
  jQuery.fn = jQuery.prototype = {
    constructor: jQuery,
    version: '1.0.0',
    init: function (selector) {
      // 在jquery中这里有一个复杂的判断,但是这里我做了简化
      var elem, selector;
      elem = document.querySelector(selector);
      this[0] = elem;
      // 在jquery中返回一个由所有原型属性方法组成的数组,我们这里简化,直接返回this即可
      // return jQuery.makeArray(selector, this);
      return this;
    },
    // 在原型上添加一堆方法
    toArray: function () { },
    get: function () { },
    each: function () { },
    ready: function () { },
    first: function () { },
    slice: function () { }
    // ... ...
  }
  jQuery.fn.init.prototype = jQuery.fn;
  // 实现jQuery的两种扩展方式
  jQuery.extend = jQuery.fn.extend = function (options) {
    // 在jquery源码中会根据参数不同进行很多判断,我们这里就直接走一种方式,所以就不用判断了
    var target = this;
    var copy;
    for (name in options) {
      copy = options[name];
      target[name] = copy;
    }
    return target;
  }
  // jQuery中利用上面实现的扩展机制,添加了许多方法,其中
  // 直接添加在构造函数上,被称为工具方法
  jQuery.extend({
    isFunction: function () { },
    type: function () { },
    parseHTML: function () { },
    parseJSON: function () { },
    ajax: function () { }
    // ...
  })
  // 添加到原型上
  jQuery.fn.extend({
    queue: function () { },
    promise: function () { },
    attr: function () { },
    prop: function () { },
    addClass: function () { },
    removeClass: function () { },
    val: function () { },
    css: function () { }
    // ...
  })
  // $符号的由来,实际上它就是jQuery,一个简化的写法,在这里我们还可以替换成其他可用字符
  ROOT.jQuery = ROOT.$ = jQuery;
})(window);


在上面的代码中,我封装了一个简化版的jQuery对象。


它向大家简单展示了jQuery的整体骨架。


在代码中可以看到,jQuery自身对于原型的处理使用了一些巧妙的方式,比如jQuery.fn = jQuery.prototypejQuery.fn.init.prototype = jQuery.fn;等,这几句正是jQuery对象的关键所在。看图分析。


微信图片_20220510233558.jpg


2


对象封装分析


在上面的实现中,首先在jQuery构造函数里声明了一个fn属性,并将其指向了原型jQuery.prototype。然后在原型中添加了init方法。


jQuery.fn = jQuery.prototype = {
  init: {}
}


之后又将init的原型,指向了jQuery.prototype。


jQuery.fn.init.prototype = jQuery.fn;


而在构造函数jQuery中,返回了init的实例对象。


var jQuery = function (selector) {
  // 在jQuery中直接返回new过的实例,这里的init是jQuery的真正构造函数
  return new jQuery.fn.init(selector)
}


最后对外暴露入口时,将字符$jQuery对等起来。


ROOT.jQuery = ROOT.$ = jQuery;


因此当我们直接使用$('#test')创建一个对象时,实际上是创建了一个init的实例,这里的真正构造函数是原型中的init方法。


注意:许多对jQuery内部实现不太了解的盆友,常常会毫无节制使用$(),比如对于同一个元素的不同操作:


var width = parseInt($('#test').css('width'));
if(width > 20) {
  $('#test').css('backgroundColor', 'red');
}


通过我们上面的一系列分析,我们知道每当我们执行$()时,就会重新生成一个init的实例对象,因此当我们这样没有节制的使用jQuery是非常不正确的,虽然看上去方便了一些,但是对于内存的消耗非常大。正确的做法是既然是同一个对象,那么就用一个变量保存起来后续使用即可。


var $test = $('#test');
var width = parseInt($test.css('width'));
if(width > 20) {
  $test.css('backgroundColor', 'red');
}


3


扩展方法分析


在上面的代码实现中,我还简单实现了两个扩展方法。


jQuery.extend = jQuery.fn.extend = function (options) {
  // 在jquery源码中会根据参数不同进行很多判断,我们这里就直接走一种方式,所以就不用判断了
  var target = this;
  var copy;
  for (name in options) {
    copy = options[name];
    target[name] = copy;
  }
  return target;
}


要理解它的实现,首先要明确知道this的指向。如果你搞不清楚,可以回头去看看我们之前关于this指向的讲解。


传入的参数options为一个key: value模式的对象,我通过for in遍历options,将key作为jQuery的新属性,value作为该新属性所对应的新方法,分别添加到jQuery方法和jQuery.fn中。


也就是说,当我们通过jQuery.extend扩展jQuery时,方法被添加到了jQuery构造函数中,而当我们通过jQuery.fn.extend扩展jQuery时,方法被添加到了jQuery原型中。

上面的例子中,我也简单展示了在jQuery内部,许多方法的实现都是通过这两个扩展方法来完成的。


当我们通过上面的知识了解了jQuery的大体框架之后,我们对于jQuery的学习就可以具体到诸如css/val/attr等方法是如何实现这样的程度,那么源码学习起来就会轻松很多,节省更多的时间。也给一些对于源码敬而远之的朋友提供一个学习的可能。


4


有一个朋友留言给我,说她被静态方法,工具方法和实例方法这几个概念困扰了很久,到底他们有什么区别?


其实在上一篇文章中,关于封装一个对象,我跟大家分享了一个非常非常干,但是却只有少数几个读者大佬get到的知识,那就是在封装对象时,属性和方法可以具体放置的三个位置,并且对于这三个位置的不同做了一个详细的解读。


微信图片_20220510233601.jpg


在实现jQuery扩展方法时,一部分方法需要扩展到构造函数中,一部分方法需要扩展到原型中,当我们通读jQuery源码时,还发现有一些方法放在了模块作用域中,至于为什么会有这样的区别,建议大家回过头去读读前一篇文章。


这里用一个例子简单区分一下。


// 模块内部
const a = 20;
function Person(name, age) {
  this.name = name;
  this.age = age;
  // 构造函数方法,每声明一个实例,都会重新创建一次,属于实例独有
  this.getName = function() {
    return this.name;
  }
}
// 原型方法,仅在原型创建时声明一次,属于所有实例共享
Person.prototype.getAge = function() {
  return this.age;
}
// 工具方法,直接挂载在构造函数名上,仅声明一次,无法直接访问实例内部属性与方法
Person.each = function() {}


如上例中,each就是一个工具方法,或者说静态方法。


工具方法的特性也和工具一词非常贴近,他们与实例的自身属性毫无关联,仅仅只是实现一些通用的功能,我们可以通过$.each$('div').each这2个方法来体会工具方法与实例方法的不同之处。


在实际开发中,我们运用得非常多的一个工具库就是lodash.js,大家如果时间充裕一定要去学习一下他的使用。


$.ajax()
$.isFunction()
$.each()
... ...


放在原型中的方法,在使用时必须创建了一个新的实例对象才能访问,因此这样的方法叫做实例方法。也正是因为这一点,他的使用成本会比工具方法高一些。但是相比构造函数方法,原型方法更节省内存。


$('#test').css()
$('#test').attr()
$('div').each()


这样,那位同学的疑问就被搞定啦。我们在学习的时候,一定不要过分去纠结一些概念,而要明白这些概念背后的具体场景是怎么回事儿,那么学习这件事情就不会在一些奇奇怪怪的地方卡住了。


所以通过$.extend扩展的方法,其实就是对工具方法的扩展,而通过$.fn.extend扩展的方法,就是对于实例方法的扩展。


5


jQuery插件的实现


我在初级阶段时,觉得要自己编写一个jQuery插件是超级高大上的事情,可望不可即。但是通过对于上面的理解,我就知道扩展jQuery插件其实是一件我们自己也可以完成的事情。


在前面我跟大家分享了jQuery如何实现,以及他们的方法如何扩展,并且前一篇文章分享了拖拽对象的具体实现。所以建议大家暂时不要阅读下去,自己动手尝试将拖拽扩展成为jQuery的一个实例方法,这就是一个jQuery插件了。


具体也没有什么可多说的了,大家看了代码就可以明白一切。


// Drag对象简化代码,完整源码可在上一篇文章中查看
;
(function () {
  // 构造
  function Drag(selector) { }
  // 原型
  Drag.prototype = {
    constructor: Drag,
    init: function () {
      // 初始时需要做些什么事情
      this.setDrag();
    },
    // 稍作改造,仅用于获取当前元素的属性,类似于getName
    getStyle: function (property) { },
    // 用来获取当前元素的位置信息,注意与之前的不同之处
    getPosition: function () { },
    // 用来设置当前元素的位置
    setPostion: function (pos) { },
    // 该方法用来绑定事件
    setDrag: function () { }
  }
  // 一种对外暴露的方式
  window.Drag = Drag;
})();
// 通过扩展方法将拖拽扩展为jQuery的一个实例方法
(function ($) {
  $.fn.extend({
    becomeDrag: function () {
      new Drag(this[0]);
      return this;   // 注意:为了保证jQuery所有的方法都能够链式访问,每一个方法的最后都需要返回this,即返回jQuery实例
    }
  })
})(jQuery);
相关文章
|
4月前
|
存储 JavaScript 前端开发
JavaScript:揭秘网页背后的魔法,一探究竟JS的神奇力量!
【8月更文挑战第22天】JavaScript(JS)始于1995年,以网页动态效果闻名。随Node.js等技术发展,JS现广泛用于服务器端、桌面及移动应用开发。JS是解释型语言,在浏览器中直接运行。基本语法涵盖变量声明、数据类型、运算符及控制结构。变量可用`var`、`let`或`const`声明,支持多种数据类型如`Number`、`String`。函数是代码的基本单元,支持匿名及箭头函数。对象用于表示复杂数据结构。ES6引入了类、模块等新特性。异步编程通过回调、Promises及async/await实现。掌握这些基础知识,有助于开发者构建高质量的Web应用。
56 0
|
JavaScript 前端开发 CDN
jQuery补充
jQuery补充
225 0
|
JSON JavaScript 前端开发
jQuery ——(很全、很清晰,三小时带你学会 jQuery 基础)
jQuery ——(很全、很清晰,三小时带你学会 jQuery 基础)
357 0
|
JavaScript 前端开发 API
jQuery重点知识整理
jQuery重点知识整理
147 0
jQuery重点知识整理
|
开发框架 JavaScript 前端开发
jQuery 已“死”?为清除技术债,我们删掉了前端所有 jQuery 依赖
近期,英国公共部门信息网站 GOV.UK 前端开发主管 Matt Hobbs 宣布该公司删除了 jQuery 作为所有前端应用程序的依赖项,这意味着“在所有 13 个 FE 应用程序中,JS 大小减少了 32 KB(31% ~49% 之间)”。
144 0
jQuery 已“死”?为清除技术债,我们删掉了前端所有 jQuery 依赖
|
JavaScript 前端开发 API
终于有人对 jQuery下手了,一键移除项目对它的依赖
大家好,我是零一。虽然现在很多前端项目都在用Vue、React,但依赖jquery的项目也不少,尤其是年代比较久远的项目,那些还正在维护jquery项目的你,是否想将jquery从你的项目中移除?毕竟这个库那么大,你能用到的代码也就只有15%~30%,而且jquery对各个浏览器的兼容性也做了很大的处理(代码量up up),但其实很多老项目也不会去考虑兼容很边缘的浏览器了,所以其实jquery中兼容处理的代码也没太大必要 最近新发现了一个有意思的工具,仅上线2周,就有600+的Star,它说能帮助你的项目脱离对jquery的依赖,感觉是个不错的想法,一起来看看吧~
157 0
终于有人对 jQuery下手了,一键移除项目对它的依赖
|
存储 JavaScript 前端开发
不依赖jQuery也能做好动画
在开发者社区中有种错误的观念——认为在web中,CSS动画是唯一高性能的动画方式。这使很多开发者放弃基于JavaScript的动画。所以导致——(1)强制使用大量样式表来完成复杂的UI交互,(2)不能很好地支持IE8、9,(3)放弃只有JS才能完成的完美物理运动效果。
146 0