参考文章:字符串
1. 定义
字符串就是零个或多个排在一起的字符,放在单引号或双引号之中。
1.1 单引号和双引号
单引号字符串的内部,可以使用双引号。双引号字符串的内部,可以使用单引号。
如果要在单引号字符串的内部,使用单引号,就必须在内部的单引号前面加上反斜杠,用来转义。双引号字符串内部使用双引号,也是如此转义。
1.2 多行书写,输出单行
字符串默认只能写在一行内,分成多行将会报错。
'a
b
c' //Uncaught SyntaxError: Invalid or unexpected token
如果长字符串必须写成多行,可以在每一行的尾部使用反斜杠(续行符)。
注意,反斜杠的后面不能有其他字符(比如空格),否则会报错。
var longString = 'Long \
long \
long \
string';
longString
// "Long long long string"
连接运算符+
可以连接多个单行字符串,将长字符串拆成多行书写,输出的时候也是单行。
var longString = 'Long '
+ 'long '
+ 'long '
+ 'string';
longString
//"Long long long string"
1.3 多行字符串
由于多行字符串用\n
,然后.replce((/\n/g,"
写起来比较费事,所以最新的
"))ES6
标准新增了一种多行字符串的表示方法,用反引号表示:
`这是一个
多行
字符串`;
1.4 模板字符串
要把多个字符串连接起来,可以用+
号连接:
var name = '小明';
var age = 20;
var message = '你好, ' + name + ', 你今年' + age + '岁了!';
alert(message);
如果有很多变量需要连接,用+
号就比较麻烦。ES6
新增了一种模板字符串,它会自动替换字符串中的变量:
var name = '小明';
var age = 20;
var message = `你好, ${name},
你今年${age}岁了!`;
alert(message);
2. 转义
反斜杠(\)
在字符串内有特殊含义,用来表示一些特殊字符,所以又称为转义符。
需要用反斜杠转义的特殊字符,主要有下面这些:
-
\0
:null
(\u0000
) -
\b
:后退键(\u0008
) -
\f
:换页符(\u000C
) -
\n
:换行符(\u000A
) -
\r
:回车键(\u000D
) -
\t
:制表符(\u0009
) -
\v
:垂直制表符(\u000B
) -
\'
:单引号(\u0027
) -
\"
:双引号(\u0022
) -
\\
:反斜杠(\u005C
)
上面这些字符前面加上反斜杠,都表示特殊含义。
- 下面代码中,
\n
表示换行,输出的时候就分成了两行。
console.log('1\n2')
// 1
// 2
- 反斜杠还有三种特殊用法(都代表一个字符)。
(1)\HHH
反斜杠后面紧跟三个八进制数(000
到377
),代表一个字符。HHH
对应该字符的 Unicode
码点,比如\251
表示版权符号。显然,这种方法只能输出256
种字符。
(2)\xHH
\x
后面紧跟两个十六进制数(00
到FF
),代表一个字符。HH
对应该字符的 Unicode
码点,比如\xA9
表示版权符号。这种方法也只能输出256
种字符。
(3)\uXXXX
(重点)
\u
后面紧跟四个十六进制数(0000
到FFFF
),代表一个字符。XXXX
对应该字符的 Unicode
码点,比如\u00A9
表示版权符号。
下面是这三种字符特殊写法的例子:
'\251' // "©"
'\xA9' // "©"
'\u00A9' // "©"
'\172' === 'z' // true
'\x7A' === 'z' // true
'\u007A' === 'z' // true
- 如果在非特殊字符前面使用反斜杠,则反斜杠会被省略。
'\a'
// "a"
上面代码中,a
是一个正常字符,前面加反斜杠没有特殊含义,反斜杠会被自动省略。
- 如果字符串的正常内容之中,需要包含反斜杠,则反斜杠前面需要再加一个反斜杠,用来对自身转义。
"Prev \\ Next"
// "Prev \ Next"
3. 字符串与数组
字符串可以被视为字符数组,因此可以使用数组的方括号运算符,用来返回某个位置的字符(位置编号从0
开始)。
var s = 'hello';
s[0] // "h"
s[1] // "e"
s[4] // "o"
// 直接对字符串使用方括号运算符
'hello'[1] // "e"
如果方括号中的数字超过字符串的长度,或者方括号中根本不是数字,不会报错,会返回undefined
。
'abc'[3] // undefined
'abc'[-1] // undefined
'abc'['x'] // undefined
- 但是,字符串与数组的相似性仅此而已。实际上,无法改变字符串之中的单个字符。
var s = 'hello';
delete s[0];
s // "hello"
s[1] = 'a';
s // "hello"
s[5] = '!';
s // "hello"
上面代码表示,字符串内部的单个字符无法改变和增删,这些操作会默默地失败。
4. length 属性
length属性返回字符串的长度,该属性也是无法改变的。
var s = 'hello';
s.length // 5
s.length = 3;
s.length // 5
5. 字符集
JavaScript
引擎内部,所有字符都用 Unicode
表示。
JavaScript
不仅以 Unicode
储存字符,还允许直接在程序中使用 Unicode
码点表示字符,即将字符写成\uxxxx
的形式,其中xxxx
代表该字符的 Unicode
码点。
比如,\u00A9
代表版权符号。
var s = '\u00A9';
s // "©"
解析代码的时候,JavaScript
会自动识别一个字符是字面形式表示,还是 Unicode
形式表示。输出给用户的时候,所有字符都会转成字面形式。
- 下面代码中,第一行的变量名
foo
是Unicode
形式表示,第二行是字面形式表示。JavaScript
会自动识别。
var f\u006F\u006F = 'abc';
foo // "abc"
-
Unicode
共包含1,112,064个有效码点(code points
),Unicode
标准定义了UTF-8
,UTF-16
,UTF-32
等编码方式(UTF:unicode transformation format
) - 我们还需要知道,每个字符在
JavaScript
内部都是以16
位(即2
个字节)的UTF-16
格式储存。也就是说,JavaScript
的单位字符长度固定为16
位长度,即2个字节。
6. Base64 转码
有时,文本里面包含一些不可打印的符号,比如 ASCII
码0
到31
的符号都无法打印出来,这时可以使用 Base64
编码,将它们转成可以打印的字符。另一个场景是,有时需要以文本格式传递二进制数据,那么也可以使用 Base64
编码。
所谓 Base64
就是一种编码方法,可以将任意值转成 0~9、A~Z、a-z、+
和/
这64
个字符组成的可打印字符。
使用它的主要目的,不是为了加密,而是为了不出现特殊字符,简化程序的处理。
JavaScript
原生提供两个 Base64
相关的方法:
-
btoa()
:任意值转为Base64
编码 -
atob()
:Base64
编码转为原来的值
var string = 'Hello World!';
btoa(string) // "SGVsbG8gV29ybGQh"
atob('SGVsbG8gV29ybGQh') // "Hello World!"
注意,这两个方法不适合非 ASCII
码的字符,会报错。
btoa('你好') // 报错
要将非 ASCII
码字符转为 Base64
编码,必须中间插入一个转码环节,再使用这两个方法。
function b64Encode(str) {
return btoa(encodeURIComponent(str));
}
function b64Decode(str) {
return decodeURIComponent(atob(str));
}
b64Encode('你好') // "JUU0JUJEJUEwJUU1JUE1JUJE"
b64Decode('JUU0JUJEJUEwJUU1JUE1JUJE') // "你好"