jQuery的实质, 自己写一个jQuery

jQuery 的实质:

  • 在了解了python的装饰器之后, 发现jQuery的本质就是一个装饰器, 只是它接受了不同的参数.
    • python装饰器接受一个函数, 在装饰器内部用一个函数(wrapper)将其包装, 并返回处理后的结果.
    • jQuery接受一个css选择器或dom对象作为参数, 返回一个包装后的对象(jQuery对象).

自定义DOM API

  • 实现一个获取兄弟元素的API
    • 通过定义 length属性, 让函数返回值可迭代, 成为一个类数组对象.
 function getSiblings(node){
    var allChildren = node.parentNode.children
    var array = {length:0}
    for (let i =0; i<allChildren.length; i++){
    if(allChildren[i] !== item3){
      array[array.length] = allChildren[i]
      array.length += 1
    }
  }
return array
}
  • 实现一个可批量增加/删除 类的API
    我们可以使用点记法 obj.attr 或者 obj[attr] 来调用一个对象的成员
function addClass(node, classes){
  for(let key in classes){
      var value = classes[key]
      var methodName = value?'add':'remove'
      node.classList[methodName](key)      
    }
}
  • 实现一个命名空间
    • JS中并不提供原生的命名空间支持
    • 我们可以创建一个简单对象字面量来打包所有的相关函数和变量。
      • 这个简单对象字面量模拟了命名空间的作用。
      • 避免覆盖全局变量
      • 为我们自定的API提供了一个统一的入口
window.myDom ={}
myDom.addClass = addClass
myDom.getSiblings = getSiblings
  • 让一个dom对象具有我们定义的API

    • 方法1:
      • 扩展 Node 接口, 在 Node.prototype上加函数.
      • 这个方法是不好的
  • 方法2:

    • 使用一个wrapper函数来包装这个对象并返回它, jQuery就是用这种的方式实现的
      • 接收一个dom对象, 返回一个新的jQuery对象.
      • jQuery 还能够区分传入的是字符串或是node节点
window.Node2 = function(nodeOrSelector) {
  let node
  if (typeof nodeOrSelector === 'string') {
    node = document.querySelector(nodeOrSelector)
  } else {
    node = nodeOrSelector
  }
  return {
    getSiblings: function() {
      var allChildren = node.parentNode.children
      var array = {
        length: 0
      }
      for (let i = 0; i < allChildren.length; i++) {
        if (allChildren[i] !== item3) {
          array[array.length] = allChildren[i]
          array.length += 1
        }
      }
      return array
    },
    addClass: function(classes) {
      for (let key in classes) {
        var value = classes[key]
        var methodName = value ? 'add' : 'remove'
        node.classList[methodName](key)
      }
    }
  }
}
  • 实现一个jQuery选择器
    • 我们要实现类似jQuery的功能, 每次都要能够获得一个jQuery对象这样的类数组对象.
      • 如果参数是一个字符串, 我们就通过 querySelectorAll 获取一个类数组对象
        • 迭代获得的对象, 将它的每一个元素放到新的对象中.
        • 要保证自定义的类数组对象有index下标和length属性. 这样它才是可迭代的
    • 如果参数是一个节点, 我们同样它放到一个具有 index 和 length的自定义类数组对象中.
window.jQuery = function(nodeOrSelector) {
  let nodes = {}
  if (typeof nodeOrSelector === 'string') {
    let temp = document.querySelectorAll(nodeOrSelector)
      //我们不要nodeList, 要一个自定的类数组元素
    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
    }
  }
  • 对选择器对象设置 getText 和 setText 方法
    • getText方法要返回一个可迭代对象.
      • 其中每个元素存储一个选择器元素的 textContent值
    • setText 方法要对选择器进行统一设置
      • 对选择器的内一个元素的 textContent 进行设置.
  nodes.getText = function() {
    let text = []
    for (let i = 0; i < nodes.length; i++) {
      text.push(nodes[i].textContent)
    }
    return text
  }

  nodes.setText = function(text) {
    for (let i = 0; i < nodes.length; i++) {
      nodes[i].textContent = text
    }
    return nodes
  }
  • jQuery 中的 getText 和 setText 方法
    • 在jQuery中, 两种方法被合并了, 通过判断是否有传参数, 来执行不同的代码语句
 nodes.text = function(text) {
    if (text === undefined) {
      let text = []
      for (let i = 0; i < nodes.length; i++) {
        text.push(nodes[i].textContent)
      }
      return text
    } else {
      for (let i = 0; i < nodes.length; i++) {
        nodes[i].textContent = text
      }
      return nodes
    }
  }
  • 这其实类似python中的 @property 和 @attr.setter, 为getter方法和setter方法 提供了一个统一的接口

    • @property装饰器
  • nodes.text 函数和闭包

    • nodes.text 函数内部没有声明 nodes变量, 而是引用了函数外部的变量, nodes.text 函数和 nodes变量就构成了一个闭包.
  • 自定义jQuery的完整代码:

window.jQuery = function(nodeOrSelector) {
  let nodes = {}
  if (typeof nodeOrSelector === 'string') {
    let temp = document.querySelectorAll(nodeOrSelector)
      //我们不要nodeList, 要一个自定的类数组元素
    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
    }
  }

  nodes.addClass = function(classes) {
    for (let key in classes) {
      var value = classes[key]
      var methodName = value ? 'add' : 'remove'
      for (let i = 0; i < nodes.length; i++) {
        nodes[i].classList[methodName](key)
      }
    }
  }

  nodes.text = function(text) {
    if (text === undefined) {
      let text = []
      for (let i = 0; i < nodes.length; i++) {
        text.push(nodes[i].textContent)
      }
      return text
    } else {
      for (let i = 0; i < nodes.length; i++) {
        nodes[i].textContent = text
      }
      return nodes
    }
  }
  return nodes
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351