jQuery 的简易实现思路

概述

总所周知,jQuery是一个早期非常强大的js库,它小巧、快速、特性丰富,它在HTML的文档遍历、DOM操作,事件冒泡,动画以及AJAX使用方面提供了非常便捷易用的API。本文从学习的角度简要分析jQuery的封装机制。

实现思路

当给定需求如下:

window.jQuery = ???
window.$ = jQuery

var $div = $('div')
$div.addClass('red')     // 可将所有 div 的 class 添加一个 red
$div.setText('hi')       // 可将所有 div 的 textContent 变为 hi

在只知道常用DOM API的情况下,为了模拟一个简易jQuery的封装机制,首先能想到的是在Node的原型上添加方法addClasssetText,但这样就破坏了DOM本身的封装,并不是一个好方法。更好的思路是设计一个新的函数来对原node添加装饰功能并返回。具体如下:

(1)version 1.0:

window.jQuery = function (node1) {
          var node = node1

            node.addClass = function (className) {
                node.classList.add(className)
            }

            node.setText = function (text) {
                node.textContent = text
            }

            return node
        }

(2)version 2.0:类型检测(传入字符串或节点对象)

window.jQuery = function (node1) {
            var node

            if (typeof node1 === "string") {
                node = document.querySelector(node1)
            } else {
                node = node1
            }

            node.addClass = function (className) {
                node.classList.add(className)
            }

            node.setText = function (text) {
                node.textContent = text
            }

            return node
        }

(3)version 3.0: 支持操作多个元素

window.jQuery = function (node1) {
            var node

            if (typeof node1 === "string") {
                node = document.querySelectorAll(node1)
            } else {
                node = {}
                node[0] = node
                node["length"] = 1
            }

            node.addClass = function (className) {
                for (var i = 0; i < node.length; i++) {
                    node[i].classList.add(className)
                }
            }

            node.setText = function (text) {
                for (var i = 0; i < node.length; i++) {
                    node[i].textContent = text
                }
            }

            return node
        }

(4)version 4.0:优化代码

 window.jQuery = function (nodeOrSelector) {
            let nodes = {}

            // 返回一个纯净的数组
            if (typeof nodeOrSelector === "string") {
                let temp = document.querySelectorAll(nodeOrSelector)
                for (let i = 0; i < temp.length; i++) {
                    nodes[i] = temp[i]
                }
                nodes.length = temp.length
            } else if (nodeOrSelector instanceof Node) {
                nodes = {
                    0: nodeOrSelector,
                    length: 1
                }

            }

            // 同时添加多个类(classes可以是一个字符串,也可以是一个数组集合了需要添加的类)
            nodes.addClass = function (classes) {

                if (typeof classes === "string") {
                    for (let i = 0; i < nodes.length; i++) {
                        nodes[i].classList.add(classes)
                    }
                } else if (classes instanceof Array) {
                    classes.forEach(value => {
                        for (let i = 0; i < nodes.length; i++) {
                            nodes[i].classList.add(value)
                        }
                    })
                }
            }

            nodes.setText = function (text) {
                for (let i = 0; i < nodes.length; i++) {
                    nodes[i].textContent = text
                }
            }

            return nodes
        }

window.$ = window.jQuery

在4.0版里,我们能够得到

  • 更纯净的node(原型链里不再有Node.prototype
  • 可同时添加多个类的addClass方法,接受的参数可以是一个字符串,即添加一个类,也可以是多个类组成的数组(因为错误处理我还没了解,所以其他类型抛出错误现在没有实现)
  • 同时给得到的所有节点添加文字内容的setText方法

总结

以上的过程,旨在学习jQuery的封装机制。即jQuery是一个构造函数,它封装了原生js的属性和方法,以更于易用的api供我们使用。本文是对jQuery的一个简易版实现,理解了其中的原理,对于我们学习其他框架也是大有裨益的。

加油!

你可能感兴趣的:(jQuery 的简易实现思路)