模板字面量(Template Literal)是一种能够嵌入表达式的格式化字符串,有别于普通字符串,它使用反引号(`)包裹字符序列,而不是双引号或单引号。模板字面量包含特定形式的占位符(${expression}),由美元符号、大括号以及合法的表达式组成,合法的表达式(expression)可以是变量、算术或函数调用,甚至还可以是模板字面量。在ES6引入模板字面量后,就能避免用若干个加号来实现字符串拼接,而改用更为优雅的语法来替代,下面用新旧两种方式分别来组合字符串。

var name = "strick",
age = 29, str;
str = "My name is \"" + name + "\". My age is " + age + "."; //传统拼接方式
str = `My name is "${name}". My age is ${age}.`;      //模板字面量方式

  对比上面两条赋值语句可以得出,无论是代码可读性还是简洁性,新方式都要略胜一筹。因为旧方式中的普通字符串是用双引号包裹的,所以字符序列中的双引号要用反斜线(\)转义。而模板字面量则无需为双引号或单引号转义,但如果出现反引号,那么就得将其转义。

一、占位符

1)表达式

  在占位符内,表达式的计算结果会按照一定的规则转换成字符串,如果计算结果是数字、布尔值、对象或数组等,那么就调它们内置的toString()方法;而如果是null或undefined,那么就用String()函数实现类型转换,具体如下所示。

`${"abc"}`;         //"abc"
`${123}`; //"123"
`${true}`; //"true"
`${null}`; //"null"
`${undefined}`; //"undefined"
`${{ id: 1 }}`; //"[object Object]"
`${[1, 2, 3]}`; //"1,2,3"

  有一点要引起注意,那就是占位符中的变量必须先声明(可以不初始化),否则将抛出未定义的引用错误。下面的school变量就没有预先声明,而直接在模板字面量中使用。

`I am studying at ${school}.`;        //抛出未定义的引用错误

  前面曾提到过占位符中的表达式可以是模板字面量,这让占位符变得非常灵活,可以适应更多的场景,处理更为复杂的问题。例如有这么一个需求,为Chrome浏览器中的CSS属性添加浏览器前缀-webkit-,可以像下面这样编写。

let attr = "border-radius";
function isChrome() {
return true; //为了简化演示,省略了浏览器嗅探逻辑
}
attr = `${isChrome() ? `-webkit-${attr}` : attr}`;
console.log(attr); //"-webkit-border-radius"

2)作用域

  在占位符中的变量,它的作用域和定义模板字面量时所处的位置有关,而不是调用时的位置。以下面代码为例,有3个同名的scope变量,分别定义在全局作用域、outer1()函数和inner()函数中,模板字面量作为一个实参传递给inner()函数,最后在inner()函数中把模板字面量输出到控制台。

var scope = "global";     //全局变量
function outer1() {
var scope = "outer";
function inner(str) {
var scope = "inner";
console.log(str);
}
inner(`current ${scope}`);
}
outer1();   //"current outer"

  根据前面的作用域规则可知,得到的结果是“current outer”。如果模板字面量所处的作用域中没有该变量,那么就会沿着作用域链向上搜索,直到全局作用域为止。在下面的代码中,注释了outer2()函数中的scope变量,得到的结果为“current global”。

var scope = "global";     //全局变量
function outer2() {
//var scope = "outer";
function inner(str) {
var scope = "inner";
console.log(str);
}
inner(`current ${scope}`);
}
outer2();   //"current global"

二、多行字符串

  在ES6之前,如果要创建多行字符串,那么得像下面这样间接实现。

let multi1 = "first line \n" +
"second line \n" +
"third line";
let multi2 = "first line \n\
second line \n\
third line";

  第一种是将多段字符串用加号拼接,第二种是在换行之前使用反斜线(\)。虽然是两种实现方法,但两者都需要添加换行符(\n)。而在ES6引入了模板字面量后,就能原生支持多行字符串,不需要再用上述权宜之计了。具体如下所示,既不需要加号和反斜线,也不需要换行符,代码言简意赅。

let multi3 = `first line
second line
third line`;

  注意,模板字面量能识别空白符,像上面代码中的多行字符串,其第二行和第三行的开头都包含了空白符,因此在输出时都会有缩进。

三、标签模板

  模板字面量虽然强大,但也有它的局限性,例如下面两点:

(1)有可能会遭受XSS(跨站脚本攻击)攻击,因为无法转义HTML中的特殊字符(例如“<”、“>”等)。

(2)不能替代模板引擎(例如Mustache、Handlebars等),因为无法在占位符中使用if、while等语句。

1)调用

  为了解决上述问题,ES6引入了标签模板(Tagged Template)。标签模板并不是模板,而是一种特殊方式的函数调用,如下所示。

func`<p>${name}</p><p>${age}</p>`;

  调用func()函数的时候省略了圆括号,函数名后面直接跟模板字面量,这就是标签模板的调用方式。它一般会包含两个参数,第一个是由没有被替换的部分组成的数组,第二个是剩余参数,包含了所有占位符中的计算结果。下面是一个完整的标签模板的示例。

function func(literals, ...substitutions) {
console.log(literals); //["<p>", "</p><p>", "</p>", raw]
console.log(substitutions); //["strick", 29]
}
var name = "strick",
age = 29;
func`<p>${name}</p><p>${age}</p>`;

  注意观察两个参数的输出结果。literals数组包含三个元素和一个特殊的raw属性。三个元素分别是第一个占位符之前的部分,两个占位符之间的部分和第二个占位符后面的部分。剩余参数substitutions由两个占位符中所引用的变量name和age的值组成。

2)raw属性

  这里重点提一下raw属性,它也是一个数组,包含了literals数组中的三个元素所对应的原始信息,相当于为每个元素调用了一次String对象的raw()方法。注意,String.raw()是一个内置的标签模板,在调用时要用特殊的形式。

  下面用一个例子来演示String.raw()的功能,先定义一个包含水平制表符(\t)的字符串,然后在第一次输出的时候,“<p>”和“</p>”之间会有空格隔开,接着调用String.raw(),再次输出时就能把“\t”也一并显示。其实要在控制台显示第二条注释需要在“\t”前加一条反斜线(即“<p>\\t</p>”)做转义,这样才能把“\t”分成两个独立的字符:“\”和“t”,不再有水平制表符的效果。但此处为了便于理解,省略了反斜线。

let html = `<p>\t</p>`;
console.log(html); //"<p> </p>"
html = String.raw`<p>\t</p>`;
console.log(html); //"<p>\t</p>"

ES6躬行记(4)——模板字面量的更多相关文章

  1. ES6躬行记(1)——let和const

    古语云:“纸上得来终觉浅,绝知此事要躬行”.的确,不管看了多少本书,如果自己不实践,那么就很难领会其中的精髓.自己研读过许多ES6相关的书籍和资料,平时工作中也会用到,但在用到时经常需要上搜索引擎中查 ...

  2. ES6躬行记 笔记

    ES6躬行记(18)--迭代器 要实现以下接口## next() ,return,throw 可以用for-of保证迭代对象的正确性 例如 var str = "向

  3. ES6躬行记(5)——对象字面量的扩展

    一.简洁属性和方法 当创建对象字面量时,如果属性值是与属性同名的已定义的标识符(例如变量.常量等),那么ES6允许省略冒号和属性值,这样就能避免冗余的初始化.下面分别用传统的键值对和最新的简写方式创建 ...

  4. ES6躬行记(21)——类的继承

    ES6的继承依然是基于原型的继承,但语法更为简洁清晰.通过一个extends关键字,就能描述两个类之间的继承关系(如下代码所示),在此关键字之前的Man是子类(即派生类),而在其之后的People是父 ...

  5. ES6躬行记(3)——解构

    解构(destructuring)是一种赋值语法,可从数组中提取元素或从对象中提取属性,将其值赋给对应的变量或另一个对象的属性.解构地目的是简化提取数据的过程,增强代码的可读性.有两种解构语法,分别是 ...

  6. ES6躬行记(15)——箭头函数和尾调用优化

    一.箭头函数 箭头函数(Arrow Function)是ES6提供的一个很实用的新功能,与普通函数相比,不但在语法上更为简洁,而且在使用时也有更多注意点,下面列出了其中的三点: (1)由于不能作为构造 ...

  7. ES6躬行记(14)——函数

    在前面的章节中,已陆陆续续介绍了ES6为改良函数而引入的几个新特性,本章将会继续讲解ES6对函数的其余改进,包括默认参数.元属性.块级函数和箭头函数等. 一.默认参数 在ES5时代,只能在函数体中定义 ...

  8. ES6躬行记(20)——类

    ES6正式将类(Class)的概念在语法层面标准化,今后不必再用构造函数模拟类的行为.而ES6引入的类本质上只是个语法糖(即代码更为简洁.语义更为清晰),其大部分功能(例如继承.封装和复用等)均可在E ...

  9. ES6躬行记(2)——扩展运算符和剩余参数

    扩展运算符(Spread Operator)和剩余参数(Rest Parameter)的写法相同,都是在变量或字面量之前加三个点(...),并且只能用于包含Symbol.iterator属性的可迭代对 ...

随机推荐

  1. tomcat与iis公用80端口(已经发布.net项目现在开发Java项目时tomcat在eclipse中localhost:8080打不开问题)

    在开发过.net项目的电脑上安装eclipse配置tomcat运行时打不开页面问题描述,这也是本人亲生经历,找了好多资料网上大多都是tomcat配置问题描述,今天突然想到是不是IIS的问题,果然上网一 ...

  2. orm单表查询和模糊查询

    一.单表查询 1. 返回queryset对象的查询 all() 以列表形式返回全部queryset对象 filter(**kwargs) 筛选 exclude(**kwargs) 排除 reverse ...

  3. php方法传参

    带默认值的可以不传,function getColum($a=array(),$colum='id',$null=true,$colim2=null)这几个都可以不传,如果是:function get ...

  4. mongoDB根据_id进行查询

    var ObjectID = require('mongodb').ObjectID; whereStr = {_id:ObjectID(req.body._id)}

  5. 基于继承的 MethodInterceptor 动态代理(换种写法)

    net.sf.cglib.proxy.Enhancer Generates dynamic subclasses to enable method interception. This class s ...

  6. vue图片上传

    之前花了两个月用vue做了一个建筑照片我的webApp,前端就我一人,负责用vue写页面对接接口,后台一个程序员负责写接口,嵌套个安卓ios的壳.搞的是风风火火,过程也是很心累,大多不在技术,在于所谓 ...

  7. war和war exploded区别

    war和war exploded区别 在使用IDEA开发项目的时候,部署Tomcat的时候通常会出现下边的情况: 是选择war还是war exploded 这里首先看一下他们两个的区别: war模式: ...

  8. MySQL--自增列学习

    ##=====================================================================================## 在数据库表设计中会纠 ...

  9. 面试官问我,使用Dubbo有没有遇到一些坑?我笑了。

    前言 17年的时候,因为一时冲动没把持住(当然最近也有粉丝叫我再冲动一把再更新一波),结合面试题写了一个系列的Dubbo源码解析.目前公众号大部分粉丝都是之前的粉丝,这里不过多介绍. 根据我的面试经验 ...

  10. BP算法基本原理推导----《机器学习》笔记

    前言 多层网络的训练需要一种强大的学习算法,其中BP(errorBackPropagation)算法就是成功的代表,它是迄今最成功的神经网络学习算法. 今天就来探讨下BP算法的原理以及公式推导吧. 神 ...