字符串的扩展

本文详细介绍了JavaScript中字符串的扩展,包括字符的Unicode表示法,使用大括号处理超出双字节的字符;字符串实现遍历器接口,支持for...of循环;JSON.stringify()的改进,处理特殊码点字符;模板字符串的使用,包括多行字符串和变量嵌入;以及重要的标签模板,它是函数调用的一种形式,用于处理包含变量的模板字符串。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1.字符的Unicode表示法

ES6要用Unicode表示一个字符,采用\uxxxx形式,但是只限制码点在\u0000~\uFFFF之间的字符,超出了这个范围,比如表示一个汉字,需要用两个双字节的形式表示。

大括号表示法:
对超出双字节的字符,比如\u20BB7,用大括号括起来,\u{20BB7},就可以正确解读这个字符。

"\u{20BB7}"
// "?"

"\u{41}\u{42}\u{43}"
// "ABC"

let hello = 123;
hell\u{6F} // 123

'\u{1F680}' === '\uD83D\uDE80'
// true

2.字符串的遍历器接口

字符串可以解构赋值成数组的形式,所以字符串可以被for…of循环遍历。

for(let i of "hello"){
	console.log(i);  // h e l l o
}

// 对大于\uFFFF的码点进行遍历,比如汉字
let text = String.fromCodePoint(0x20BB7);

for(let i of text){
	console.log(i);  // ?
}

3.JSON.stringify()的改造

对于这个函数的使用,要注意一个问题。首先,JSON格式数据必须是UTF-8形式的,但是对于0xD800-0xDFFF之间的码点,UTF-8规定不能单独使用,必须配对使用,还不能颠倒顺序,以表示大于0xFFFF的字符,那怎么办呢?ES2019规定,如果遇到0xD800到0xDFFF之间的单个码点,或者不存在的配对形式,它会返回转义字符串,留给应用自己决定下一步的处理。

JSON.stringify('\u{D834}') // ""\\uD834""
JSON.stringify('\uDF06\uD834') // ""\\udf06\\ud834""

4.模板字符串

学过Django的人都应该知道,里面也有一个模板文件,和这个类似,方便后台的数据传输解析。

模板字符串增强版的字符串,用反引号(`)标识。它可以当作普通字符串使用,也可以用来定义多行字符串,或者在字符串中嵌入变量。很简单,就不举例子了。

模板字符串有一个trim()方法,可以注意下,用来消除里面的换行。
向其中嵌入变量的话,用${变量名/函数},为什么可以这样表示?归根于模板字符串大括号的本质就是执行JavaScript代码。

5.标签模板(重要)

标签模板其实不是模板,而是函数调用的一种形式。但是,如果模板里有变量,就不是简单的调用了,而是先将模板字符串处理成多个参数,再调用函数

// 标签模板其实不是模板,而是函数调用的一种形式
        let a=5,b=10;

        tag`Hello ${a+b} world ${a*b}`;
        // 等同于tag(['Hello','world',''],15,50)

        function tag(stringArr,value1,value2){
            console.log(stringArr[0]);
            console.log(stringArr[1]);
            console.log(stringArr[2]);
            console.log(value1);
            console.log(value2);

            return "OK";
        }

tag函数的第一个参数是一个数组,该数组的成员是模板字符串中那些没有变量替换的部分,也就是说,变量替换只发生在数组的第一个成员与第二个成员之间、第二个成员与第三个成员之间,以此类推。记住这个规律很重要,知道怎么拆开的才能知道怎么按照原来的位置拼合回去。

let sender = '<script>alert("abc")<\/script>';
let message = SaferHTML`<p>${sender} has sent you a message.</p>`;

function SaferHTML(templateData) {
    let s = templateData[0];
    for (let i = 1; i < arguments.length; i++) {
        let arg = String(arguments[i]);

        // Escape special characters in the substitution.
        s += arg.replace(/&/g, "&amp;")
                .replace(/</g, "&lt;")
                .replace(/>/g, "&gt;");

        // Don't escape special characters in the template.
        s += templateData[i];
    }
    return s;
}

console.log(message);

这里调试一下会发现,因为templateData是第一个参数,所以接收到的是字符串数组,传过来的值会放在arguments中。其中,第一个字符串数组,有一个raw属性,也是指向同一个数组,只不过raw会自动进行转义。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值