ES6 Features系列:Template Strings & Tagged Template Strings

1. Brief

ES6(ECMAScript 6th edition)于2015年7月份发布,虽然各大浏览器仍未全面支持ES6,但我们可以在后端通过Node.js 0.12和io.js,而前端则通过Traceur或Babel这类Transpiler将ES6语法预转译为ES5语法,来提前兴奋一把。而仅需适配IE9+的朋友们现在更是可以开始撸ES6了,而不必为学哪门JavaScript超集语言而烦恼。(ES6又名为ECMAScript 2015或JavaScript.next,ES4的部分较为激进的特性被调用到该版本中实现。)

ES6带给我们很多惊喜,如class、module、export和import等。但在学习和运用到项目中时,我们需要注意以下两点:

1. ES6包含的是语法糖和语言、库的bug fix,对工程性问题没有太大的帮助;

2. 由于Traceur和Babel无法对ES6的所有特性进行完整高效的polyfill,因此我们无法完全享用ES6的各项特性。

最近接手一个项目的前端改造,正在尝试全新的技术栈(Riot+ES6+Glup+Webpack),本系列文章将作为理论+项目实践的笔记供日后查阅。

2. What is Template Strings?

一言以蔽之,Template Strings就是让我们减少字符串手工拼接的工作量。

  2.1. Before ES6

// Sample 1: 单行字符串拼接
var operand1 = 1
  , operand2 = 2.1
var tpl1 = operand1 + ‘ + ‘ + operand2 + ‘~=‘ + parseInt(operand1+operand2)
var tpl2 = [operand1, ‘ + ‘ , operand2, ‘~=‘, parseInt(operand1 + operand2)].join(‘‘)
// Sample 2: 多行字符串拼接
var name = ‘fsjohnhuang‘
  , id = ‘region‘
var tpl1 = ‘<div id="‘ + id + ‘">‘
  + ‘<a>‘ + name + ‘</a>‘
  + ‘</div>‘
var tpl2 = ‘<div id=" ‘ + id + ‘ ">\
  <a>‘ + name + ‘</a>  </div>‘

2.2. Embracing ES6

// Sample 1: 单行字符串拼接
var operand1 = 1
  , operand2 = 2.1
var tpl1 = `${operand1}+${operand2}~=${parseInt(operand1+operand2)}`
// Sample 2: 多行字符串拼接
var name = ‘fsjohnhuang‘
  , id = ‘region‘
var tpl1 = `<div id="${id}">
  <a>${name}</a>
  </div>`

假若了解过CoffeeScript,那么会发现ES6的Template Strings怎么这么眼熟。Template Strings由两部分组成:

1. 模板起始符—— `` ,称为沉音符/反引号(grave accent),其内容被识别为字符串模板。

2. 表达式占位符—— ${<expression>} ,<expression>为JavaScript的有效表达式(如 name, 1==2等),因此 ${<expression>} 并不是简单的占位符那么简单了。

  2.3. Cautions

1.  ${<expression>} 中可访问当前作用域所能访问到变量和函数,如

var x = 1

(function(){
  var y = 2
  (function(b){
    var tpl = `${x},${y},${a},${b}` // 结果是 "1,2,undefined,5"
  }(5))
  var a = 3
  let c = 4 // 由于采用let来声明c变量,因此不会发生variable hoist
}())

2.  ${<expression>} 是即时计算(real-time computing)的,通过函数加壳可实现延迟计算(lazy evaluation)

//real-time computing
var tpl = `${x},${y}`
var x = 1, y = 2
console.log(tpl) // "undefined, undefined"

// lazy evaluation
var tpl = ctx => `${ctx.x},${ctx.y}`
console.log(tpl({x:1, y:2})) // "1, 2"

3. 多行陷阱(pitfall of multiline),在编写HTML模板时我习惯如下写法

var tpl = ‘<div>\
    <h3>${title}</h3>    <span>${subtitle}</span>  </div>‘
// 然后是模板引擎解析tpl

那现在是否就可以毫无顾虑地改用Template Strings呢?

var tpl = ctx => `<div>
    <h3>${ctx.title}</h3>
    <span>${ctx.subtitle}</span>
  </div>`
// 直接调用tpl函数

答案是否定的

原因是通过正斜杠( \ )定义的多行字符串实际输出还是一行字符串而已,但通过反引号( `` )定义的是真实的多行字符串,且通过换行符( \n )分隔每一行。

// 通过\定义多行的结果
<div>    <h3>${ctx.title}</h3>    <span>${ctx.subtitle}</span>  </div>

// 通过反引号定义多行的结果
<div>\n
    <h3>${ctx.title}</h3>\n
    <span>${ctx.subtitle}</span>\n
  </div>

那么当使用jQuery将反引号定义的HTML模板来生产DOM元素时就会直接报错了,这时我们需要删除这些控制字符。

var removeCtlChar = raw => raw.replace(/[\r\n\t\v\f]/ig, ‘‘)

3. What is Tagged Template Strings?

从上文我们了解到Template Strings是以整体为单位进行即时计算,也就是说留给我们的自主操控能力是十分有限的。而Tagged Template Strings则大大增强了我们的操控欲望。

其实Tagged Template Strings实质上是对Template Strings进行Tokenize操作,从而细化我们的可操作粒度。而词法类型分为 字符串表达式占位符的运算结果

var x = 1, y = 2
var tpl = ‘hello${x}:${y+1}‘

// Tokenize后的结果
var tokens = [‘hello‘, 1, ‘:‘, 3, ‘‘] 

具体玩法如下:

// 语法
<Tagged Function><Template Strings>

/** Sample **/
/* 定义<Tagged Function>
 * @param {Array.<DOMString>} strings - 字符串类型的tokens
 * @param {...Any} vals - 表达式占位符的运算结果tokens
 * @returns {Any}
 */
var taggedFunc = (strings, ...vals){
  var ret = []
  for(let i = 0, len = strings.length ; i < len; ++i)
    ret.push(strings.raw[i], vals[i] || ‘‘)
  return ret
}

// 定义Template Strings
var x = 1, y =2
var ret = taggedFunc`\tHello${x}:${y+1}`
console.log(ret) // 显示 "\tHello1:3"
console.log(`\tHello${x}:${y+1}`) // 显示 "    Hello1:3"

<Tagged Function>函数 有两个入参分别代表两类token。 {Array.<DOMString>} strings 为字符串类型的tokens,而 {...Any} vals 则为表达式占位符运算结果tokens。

而需要注意的是: strings.length === vals.length + 1

另外我们看到最后两行代码会发现 `\tHello${x}:${y+1}` 中的制表符将在输出结果中起效,而经过Tagged Function处理的则按普通字符输出而已。其实这是通过 {Array.<DOMString>}strings.raw属性 操作strings中token的结果,也就是说strings.raw属性将对控制符进行转义从而实现按照普通字符输出。

   3.1. 内置的Tagged Function——String.raw

其作用与上述的taggedFunc一样,就是将按普通字符输出Template Strings中的控制符。

   3.2. Cautions

1. Tagge Template Strings的语法是Template Strings紧跟在Tagged Function后面,两者间不能有空格或制表符等。

2. vals是运算后的实际值,若要延迟计算依然需要加壳。

3. @ruanyifeng老师说可通过Tagged Function来自定义带流程控制的模板语言

// 下面的hashTemplate函数
// 是一个自定义的模板处理函数
var libraryHtml = hashTemplate`
  <ul>
    #for book in ${myBooks}
      <li><i>#{book.title}</i> by #{book.author}</li>
    #end
  </ul>
`;

本人觉得这种用法不可取,Tagged Function本来就按照自身规则对模板进行Tokenize,然后我们在此基础上对结果进行二次Tokenize,那还不如直接按自己定义的规则来做词法分析更省心。

4. Conclusion

Template Strings和Tagged Template Strings 均可通过Traceur和Babel做transpile,所以我们现在就可以撸起了,开干吧各位!

尊重原创,转载请注明来自:http://www.cnblogs.com/fsjohnhuang/p/4601200.html  肥子John^_^

5. Thanks

http://es6.ruanyifeng.com/#docs/string

http://www.sitepoint.com/understanding-ecmascript-6-template-strings/

时间: 2024-08-02 11:00:47

ES6 Features系列:Template Strings & Tagged Template Strings的相关文章

ES6特性-带标签的模板字符串(tagged template)

tagged template: 加在模板字符串前面加一个标签(函数). let dessert = = '甜品' drink = '茶' let breakfast = kitchen`今天的早餐是${dessert} 与 ${drink}!` function kitchen(strings,...values){ console.log(strings); console.log(values); } strings和value都是数组,里面分别是string和value: 原文地址:ht

【探秘ES6】系列专栏(一):ES6简介

摘要:新一代JavaScript标准,ES6即将发布.[探秘ES6]系列专栏将一一剖析ES6的诸多新特性,让Web开发者对此有清晰全面的了解.本文为系列的第一篇,带你了解ES6到底是什么以及有哪些令人期待的地方. ES6作为新一代JavaScript标准,即将与广大前端开发者见面.为了让大家对ES6的诸多新特性有更深入的了解,Mozilla Web开发者博客推出了<ES6 In Depth>系列文章.CSDN已获授权,将持续对该系列进行翻译,组织成[探秘ES6]系列专栏,供大家学习借鉴.本文为

angularjs学习笔记--主html&amp;template html&amp;module&amp;template js、模块、控制器、双向数据绑定、过滤器

// Register the `phoneList` component on the `phoneList` module, angular. module('phoneList'). component('phoneList', {...}); // Define the `phonecatApp` module angular.module('phonecatApp', [ // ...which depends on the `phoneList` module 'phoneList'

ES6入门系列三(特性总览下)

0.导言 最近从coffee切换到js,代码量一下子变大了不少,也多了些许陌生感.为了在JS代码中,更合理的使用ES6的新特性,特在此对ES6的特性做一个简单的总览. 1.模块(Module) --Chrome测试不可用 在ES6中,有class的概念,不过这只是语法糖,并没有解决模块化问题.Module功能则是为了解决模块化问题而提出的. 我们可以使用如下方式定义模块: 11_lib.js文件内容 // 导出属性和方法 export var PI = 3.1415926; export fun

ES6笔记系列

ES6,即ECMAScript 6.0,ES6的第一个版本是在2015年发布的,所以又称作ECMAScript 2015 如今距ES6发布已经一年多的时间,这时候才去学,作为一个JSer,羞愧羞愧,还是尽早去学习吧 主要参照阮一峰的 ES6标准入门 及其他资料,系统地学习ES6,好记星不如烂笔头,记录一下成文 以下为系列文章

Index Template和Dynamic Template

目录 什么是Index Template Index Template的工作方式 什么是Index Template Index Template - 帮助你设定Mappings 和 Settings,并按照一定的规则,自动匹配到新创建的索引之上 模板仅在一个索引被新创建时,才会产生作用.修改模板不会影响到自己创建的索引 可以设定多个模板,这些设置会被"merge"在一起 可以指定"order"的数值,控制"mergine"的过程 Index T

ES6 入门系列 - 函数的扩展

1函数参数的默认值 基本用法 在ES6之前,不能直接为函数的参数指定默认值,只能采用变通的方法. function log(x, y) { y = y || 'World'; console.log(x, y); } log('Hello') // Hello World log('Hello', 'China') // Hello China log('Hello', '') // Hello World 上面代码检查函数log的参数y有没有赋值,如果没有,则指定默认值为World.这种写法的

ES6入门系列四(测试题分析)

0.导言 ES6中新增了不少的新特性,来点测试题热热身.具体题目来源请看:http://perfectionkills.com/javascript-quiz-es6/. 以下将一题一题来解析what和why. 1.题目一 (function(x, f = () => x) { var x; var y = x; x = 2; return [x, y, f()]; })(1) A. [2, 1, 1] B. [2, undefined, 1] C. [2, 1, 2] D. [2, undef

ES6入门系列一(基础)

1.let命令 Tips: 块级作用域(只在当前块中有效) 不会变量提升(必须先申明在使用) 让变量独占该块,不再受外部影响 不允许重复声明 总之:let更像我们熟知的静态语言的的变量声明指令 ES6新增了let命令,用来声明变量.用法类似于var,但所声明的变量,只能在let命令所在的代码块内有效. let声明的变量只有块级作用域 'use strict' { let a = 1; } console.log(a); //结果是什么? 看一段熟悉的代码: var a = []; for (va