vue源码2

        vue之mustache库的机理其实是将模板字符串转化为tokens 然后再将 tokens 转化为 dom字符串,如下图

 对于一般的将模板字符串转化为dom字符串,这样不能实现复杂的功能

let data = {
            name:'小王',
            age:18
}
let templateStr = `
    <h1>我叫{{name}},我今年{{age}}岁<h1>
`
templateStr = templateStr.trim()
  
let htmlStr = templateStr.replace(/\{{(\w+)}}/g,function(match,$1,index){
    //第一个参数为他寻找的部分,第二个为捕获的东西,第三个所在的位置,第四个为该字符串
    return data[$1] 
    })
console.log(htmlStr) //我叫小王,我今年18岁

 将模板字符串转化为tokens

前面已经知道了musache的工作原理为 将模板字符串转化为tokens,然后再将tokens转化为BOM字符串所以此小节的任务为:

class Scanner {
    constructor (templateStr ){
        //将模板字符串写到实例身上
        this.templateStr = templateStr
        //指针
        this.pos = 0
        //尾巴,刚开始为字符串本身
        this.tail = templateStr
    }
    //让指针跳过目标,进而扫描后面的内容
    scan(target){
        this.pos += target.length
        this.tail = this.templateStr.substring(this.pos)

    }
    //扫描字符串,直到扫描到目标,返回目标之前的字符串
    scanUtil(target) {
        let recordPosValue = this.pos
        //如果该字符串的地一个元素即该目标的索引不为0时,说明指针还需要往右走
        while(this.tail.indexOf(target)!=0&&this.pos<this.templateStr.length){
          
            this.pos++;
            //尾巴变为pos后面的部分
            this.tail = this.templateStr.substring(this.pos)
        }
         return this.templateStr.substring(recordPosValue,this.pos)
    }
}
export default function becomeEasyToken (templateStr){
    let token = []
      //实例化一个扫描器,针对模板字符串工作
      let scanner = new Scanner(templateStr)
      while(scanner.pos<templateStr.length){
          let word;
          word = scanner.scanUtil('{{');
          if(word !=''){
            token.push(["text",word])
          }
         
          scanner.scan('{{')

          word = scanner.scanUtil("}}")
          if(word !=''){
            if(word[0] == "#"){
                token.push(["#",word.substring(1)])
            }else if(word[0]=="/"){
                token.push(['/',word.substring(1)])
            }else{
                token.push(["name",word])
            }
          }
          
          scanner.scan("}}")
          
      }
     
      return token
}

以上代码没有处理 "#" 的循环功能 ,所以还必须添加一个函数,并对该返回值稍加修改

import foldToken  from "./foldToken";
export default function becomeEasyToken (templateStr){
    let token = []
      //实例化一个扫描器,针对模板字符串工作
      let scanner = new Scanner(templateStr)
      while(scanner.pos<templateStr.length){
          let word;
          word = scanner.scanUtil('{{');
          if(word !=''){
            token.push(["text",word])
          }
         
          scanner.scan('{{')

          word = scanner.scanUtil("}}")
          if(word !=''){
            if(word[0] == "#"){
                token.push(["#",word.substring(1)])
            }else if(word[0]=="/"){
                token.push(['/',word.substring(1)])
            }else{
                token.push(["name",word])
            }
          }
          
          scanner.scan("}}")
          
      }
     
      return foldToken(token)
}
export default function foldToken(tokens) {
    //结果数组
    let nestedTokens = []
    //栈结构,存放小tokens
    let section = [];
    //与nestedTokens指向的是同一数组,该数组为一级数组
    let collentor = nestedTokens
    
    for (const item of tokens) {

        switch (item[0]) {
            case "#":
                //进栈
                section.push(item)
                
                collentor.push(item)
                //创建新一级的数组
                collentor = item[2] = [] 
                break;
            case "/":
                //出栈
                section.pop(item)
                //如果都出完了,则回到一级数组,还没出完则回到其上一级
                collentor =  section.length>0?section[section.length-1][2]:nestedTokens
                break;
            default:
                //仅负责给各级数组添加 "text" 元素
                collentor.push(item)
              
        }
    }
    return nestedTokens;
}

效果展示:

 

相关推荐

  1. vue2】模版编译

    2024-05-26 02:40:34       34 阅读
  2. Vue2】响应式原理

    2024-05-26 02:40:34       41 阅读
  3. vue2】阶段一:Vue 初始化

    2024-05-26 02:40:34       39 阅读
  4. Vue探索之Vue2.x分析(一)

    2024-05-26 02:40:34       47 阅读
  5. vue3、vue2中nextTick解析

    2024-05-26 02:40:34       50 阅读
  6. vue2解析---watch和computed

    2024-05-26 02:40:34       62 阅读
  7. Vue2梳理:关于vm.$mount的实现

    2024-05-26 02:40:34       44 阅读

最近更新

  1. docker php8.1+nginx base 镜像 dockerfile 配置

    2024-05-26 02:40:34       94 阅读
  2. Could not load dynamic library ‘cudart64_100.dll‘

    2024-05-26 02:40:34       101 阅读
  3. 在Django里面运行非项目文件

    2024-05-26 02:40:34       82 阅读
  4. Python语言-面向对象

    2024-05-26 02:40:34       91 阅读

热门阅读

  1. Effective C++(2)

    2024-05-26 02:40:34       33 阅读
  2. Midjourney绘画关键词参数汇总(一)

    2024-05-26 02:40:34       27 阅读
  3. 鸿蒙本机通信

    2024-05-26 02:40:34       32 阅读
  4. 设计一个完美的用户角色权限表

    2024-05-26 02:40:34       35 阅读
  5. react native 下载功能实现

    2024-05-26 02:40:34       35 阅读