欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

ECMAScript6入门第五章-字符串的新增方法

程序员文章站 2022-03-08 22:17:58
...

String.fromCodePoint():

ES5提供String.fromCharCode()方法,用于从Unicode码点返回对应字符串,但是这个方法不能识别码点大于0xFFFF的字符。

String.formCharCode(0x20BB7);// "ஷ"

上面代码中,String.fromCharCode()不能识别大于0xFFFF的码点,所以0x20BB7就发生了溢出,最高位2被舍弃了,最后返回码点U+0BB7对应的字符,而不是码点U+20BB7对应的字符。

ES6提供了String.formCodePoint()方法,可以识别大于0xFFFF的字符。如果String.fromCodePoint方法有多个参数,则它们会被合并成一个字符串返回。

在作用上,与codePointAt()方法相反。
formCodePoint方法定义在String对象上,而codePointAt方法定义在字符串的实例对象上。

String.fromCodePoint(0x20BB7)
// "????"

String.fromCodePoint(0x78, 0x1f680, 0x79) === 'x\uD83D\uDE80y'
// true

String.raw():

该方法返回一个斜杠都被转义的字符串(即斜杠前面再加一个斜杠),往往用于模板字符串的处理方法。

String.raw()方法可以作为处理模板字符串的基本方法,它会将所有变量替换掉,而且对斜杠进行转义,方便下一步作为字符串来使用。

String.raw`Hi\n${2+3}!`
// 实际返回 "Hi\\n5!",显示的是转义后的结果 "Hi\n5!"

如果原字符串的斜杠已经转义,那么String.raw()会再次进行转义。

String.raw`Hi\\n` === "Hi\\\\n" // true

String.raw()本质上是一个正常的函数,只是专用于模板字符串的标签函数。如果写成正常函数的形式,它的第一个参数,应该是一个具有raw属性的对象,且raw属性的值应该是一个数组,对应模板字符串解析后的值。

`foo${1 + 2}bar`
// 等同于
String.raw({ raw: ['foo', 'bar'] }, 1 + 2) // "foo3bar"

作为函数,String.raw()的代码实现基本如下:

String.raw = function (strings, ...values) {
  let output = '';
  let index;
  for (index = 0; index < values.length; index++) {
    output += strings.raw[index] + values[index];
  }

  output += strings.raw[index]
  return output;
}

实例方法:codePointAt():

JavaScript内部,字符以UTF-16的格式储存,每个字符固定为2个字节。
对于那些需要4个字节储存的字符(Unicode码点大于0xFFFF的字符),JavaScript会认为它们是两个字符。

var s = "????";

s.length // 2
s.charAt(0) // ''
s.charAt(1) // ''
s.charCodeAt(0) // 55362
s.charCodeAt(1) // 57271

上面代码中,“????”的码点是0x20BB7,UTF-16 编码为0xD842 0xDFB7(十进制为55362 57271),需要4个字节储存。对于这种4个字节的字符,JavaScript不能正确处理,字符串长度会误判为2;而且charAt()方法无法读取整个字符,charCodeAt()方法只能分别返回前两个字节和后两个字节的值。

ES6提供了codePointAt()方法,能够正确处理4个字节储存的字符,返回一个字符的码点。对于那些两个字节储存的常规字符,它的返回结果与charCodeAt()方法相同。

codePointAt()方法的参数,是字符在字符串中的位置(从0开始)。

let s = '????a';

s.codePointAt(0) // 134071
s.codePointAt(1) // 57271

s.codePointAt(2) // 97

codePointAt()方法返回的是码点的十进制值,如果想要十六进制的值,可以使用toString()方法转换一下。

let s = '????a';

s.codePointAt(0).toString(16) // "20bb7"
s.codePointAt(2).toString(16) // "61"