要点:字符串、常见字符串方法
先知:
一、串
1.1 定义
串(string) 是由零个或多个字符组成的有限序列,又名叫 字符串
1.2 串的逻辑结构
串 的逻辑结构和线性表很相似,不同之处在于串针对的是字符集,也就是串中的元素都是字符。
因此,对于串的基本操作与线性表是有很大差别的:
-
线性表更关注的是单个元素的操作,比如查找一个元素,插入或删除一个元素;
-
但串中更多的是查找子串位置,得到指定位置子串,替换子串等操作
1.3 串的存储结构
串 的存储结构与线性表相同,分为两种:
-
顺序存储结构
-
链式存储结构
串的顺序存储结构
串的顺序存储结构是用 一组地址连续的存储单元 来存储串中的字符序列。一般是用定长数组来定义
由于是定长数组,因此就会存在一个预定义的最大串长度;
一般可以将实际的串长度值保存在数组 0 下标位置,也可以放在数组最后一个下标位置;
也有些语言使用在串值后面加一个不计入串长度的结束标记符(比如
\0
)来表示串值得终结,这样就无需使用数字进行记录
串的链式存储结构
对于串的链式存储结构,与线性表是相似的;
但由于串结构的特殊性(结构中的每个元素数据都是一个字符),如果也简单地将每个链结点存储一个字符,就会存在很大的空间浪费;
因此,一个结点可以考虑存放多个字符,如果最后一个结点未被占满时,可以使用 “#” 或其他非串值字符补全
串的链式存储结构除了在链接串与串操作时有一定的方便之外
总的来说不如顺序存储灵活,性能也不如顺序存储结构好
二、JavaScript中的串
2.1 定义
字符串就是一个或多个排列在一起的字符,放在单引号或双引号之中。
‘abc’
“abc”
js里的字符串类似于数组,都是一个一个字符拼凑在一起组成的,因此可以用length属性
取得字符串的长度
1 | var str = "hello" |
2.2 字符串常用的一些方法
1.charAt()
1 | str.charAt(n) |
作用:返回字符串的第 n 个字符,如果不在 0~str.length-1之间,则返回一个空字符串。
1 | var str = "javascript"; |
2.indexOf()
1 | indexOf(substr[,start]) |
作用:返回 substr 在字符串 str 中首次出现的位置,从 start 位置开始查找,如果不存在,则返回 -1。
start可以是任意整数,默认值为 0。如果 start < 0 则查找整个字符串(如同传进了 0)。如果 start >= str.length,则该方法返回 -1,除非被查找的字符串是一个空字符串,此时返回 str.length.
1 | var str = "javascript" ; |
3.lastIndexOf()
1 | lastIndexOf(substr[,start]) |
作用:返回 substr 在字符串 str 中最后出现的位置,从 start 位置 向前开始查找,如果不存在,则返回 -1。
1 | 'lastindex'.lastIndexOf( 'a' ); // 1 |
4.substring()
1 | str.substring(start[, end]) |
作用:返回从 start 到 end(不包括)之间的字符,start、end均为 非负整数。若结束参数(end)省略,则表示从start位置一直截取到最后。
1 | var str = 'abcdefg' ; str.substring(1, 4); //"bcd" |
5.slice()
1 | str.slice(start[,end]) |
作用:返回从 start 到 end (不包括)之间的字符,可传负值
1 | var str = 'this is awesome' ; |
6. substr()
1 | str.slice(start[,end]) |
作用:返回 str 中从指定位置开始到指定长度的子字符串,start可为负值
1 | var str = "Just give me a reason" ; |
7. replace()
1 | str.replace(regexp|substr, newSubStr|function ) |
作用:替换 str 的子字符串
1 | var str = "do you love me" ; |
8. search()
1 | str.search(regexp) |
作用:查找 str 与一个正则表达式是否匹配。如果匹配成功,则返回正则表达式在字符串中首次匹配项的索引;否则,返回 -1
。如果参数传入的是一个非正则表达式对象,则会使用 new RegExp(obj) 隐式地将其转换为正则表达式对象
1 | var str = 'I love JavaScript!' ; |
9. match()
1 | str.match(regexp) |
作用:返回一个包含匹配结果的数组,如果没有匹配项,则返回 null。如果参数传入的是一个非正则表达式对象,则会使用 new RegExp(obj) 隐式地将其转换为正则表达式对象
1 | var str = 'Javascript java' ; str.match(/Java/); // ["Java"] |
10. split()
1 | str.split([separator][, limit]) |
= >返回一个数组,分隔符 separator 可以是一个字符串或正则表达式
1 | var str = "Hello?World!" ; str.split(); // ["Hello?World!"] |
11. trim()
1 | str.trim() |
作用:去除 str 开头和结尾处的空白字符,返回 str 的一个副本,不影响字符串本身的值
1 | var str = ' abc ' ; |
12. toLowerCase()
1 | str.toLowerCase() |
作用:将 str 转换为小写,并返回 str 的一个副本,不影响字符串本身的值
1 | var str = 'JavaScript' ; |
13. toUpperCase()
1 | str.toUpperCase() |
作用:将 str 转换为大写,并返回 str 的一个副本,不影响字符串本身的值
1 | var str = 'JavaScript' ; |
三、JavaScript正则表达式
关于正则匹配详见:MDN
3.1 元字符
( [ { \ ^ $ | ) ? * + .
3.2 预定义的特殊字符
字符 | 正则 | 描述 |
---|---|---|
\t | /\t/ | 制表符 |
\n | /\n/ | 换行符 |
\r | /\r/ | 回车符 |
\f | /\f/ | 换页符 |
\a | /\a/ | alert字符 |
\e | /\e/ | escape字符 |
\cX | /\cX/ | 与X相对应的控制字符 |
\b | /\b/ | 与回退字符 |
\v | /\v/ | 垂直制表符 |
\0 | /\0/ | 空字符 |
3.3 字符类
简单类
原则上正则的一个字符对应一个字符,我们可以用[]把它们括起来,让[]这个整体对应一个字符。如
1 | alert(/ruby/.test( "ruby" )); //true |
负向类
也是在那个括号里做文章,前面加个元字符进行取反,表示匹配不能为括号里面的字符。
1 | alert(/[^abc]/.test( "a" )); //false |
范围类
还是在那个中括号里面做文章。有时匹配的东西过多,而且类型又相同,全部输入太麻烦,我们可以用它。特征就是在中间加了个横线。
组合类
还是在那个中括号里面做文章。允许用中括号匹配不同类型的单个字符。
1 | alert(/[a-f]/.test( "b" )); //true |
预定义类
还是在那个中括号里面做文章,不过它好像已经走到尽头了。由于是中括号的马甲,因此它们还是对应一个字符。
字符 | 等同于 | 描述 |
---|---|---|
. | [^\n\r] | 除了换行和回车之外的任意字符 |
\d | [0-9] | 数字字符 |
\D | [^0-9] | 非数字字符 |
\s | [ \t\n\x0B\f\r] | 空白字符 |
\S | [^ \t\n\x0B\f\r] | 非空白字符 |
\w | [a-zA-Z_0-9] | 单词字符(所有的字母) |
\W | [^a-zA-Z_0-9] | 非单词字符 |
1 | alert(/\d/.test( "3" )) //true |
3.4 量词
由于元字符与特殊字符或字符类或者它们的组合(中括号)甚至它们的马甲(预定义类)都是一对一进行匹配。我们要匹配“司徒正美这个词”,最简单都要/…/,如果长到50多个字符岂不是要死人。因此我们逼切需要一个简单的操作,来处理这数量关系。
简单量词
代码 | 类型 | 描述 |
---|---|---|
? | 软性量词 | 出现零次或一次 |
* | 软性量词 | 出现零次或多次(任意次) |
+ | 软性量词 | 出现一次或多次(至道一次) |
{n} | 硬性量词 | 对应零次或者n次 |
{n,m} | 软性量词 | 至少出现n次但不超过m次 |
{n,} | 软性量词 | 至少出现n次(+的升级版) |
1 | alert(/..../.test( "司徒正美" )) //true |
/[\u4e00-\u9fa5]/用于匹配单个汉字。
贪婪量词,惰性量词与支配性量词
贪婪量词,上面提到的所有简单量词。就像成语中说的巴蛇吞象那样,一口吞下整个字符串,发现吞不下(匹配不了),再从后面一点点吐出来(去掉最后一个字符,再看这时这个整个字符串是否匹配,不断这样重复直到长度为零)
隋性量词,在简单量词后加问号。由于太懒了,先吃了前面第一个字符,如果不饱再捏起多添加一个(发现不匹配,就读下第二个,与最初的组成一个有两个字符串的字符串再尝试匹配,如果再不匹配,再吃一个组成拥有三个字符的字符串……)。其工作方式与贪婪量词相反。
支配性量词,在简单量词后加加号。上面两种都有个不断尝试的过程,而支配性量词却只尝试一次,不合口味就算了。就像一个出身高贵居支配地位的公主。但你也可以说它是最懒量词。由于javascript不支持,所以它连出场的机会也没有了。
1 | var re1 = /.*bbb/g; //贪婪 |
3.5 分组
到目前为止,我们只能一个字符到匹配,虽然量词的出现,能帮助我们处理一排密紧密相连的同类型字符。但这是不够的,下面该轮到小括号出场了,中括号表示范围内选择,大括号表示重复次数。小括号允许我们重复多个字符。
1 | //分组+量词 |
3.6 反向引用
反向引用标识由正则表达式中的匹配组捕获的子字符串。每个反向引用都由一个编号或名称来标识,并通过“\编号”表示法进行引用。
1 | var color = "#990000" ; |
3.7 候选
继续在分组上做文章。在分组中插入管道符(“|”),把它划分为两个或多个候多项。
1 | var reg = /(red|black|yellow)!!/; alert(reg.test( "red!!" )) //true |
3.8 非捕获性分组
并不是所有分组都能创建反向引用,有一种特别的分组称之为非捕获性分组,它是不会创建反向引用。反之,就是捕获性分组。要创建一个非捕获性分组,只要在分组的左括号的后面紧跟一个问号与冒号就行了。
1 | var color = "#990000" ; |
题目,移除所有标签,只留下innerText!
1 | var html = "<p><a href='http://www.cnblogs.com/rubylouvre/'>Ruby Louvre</a>by <em>司徒正美</em></p>" ; |
注意:javascript不存在命名分组
前瞻
继续在分组内做文章。前瞻与后瞻其实都属于零宽断言,但javascript不支持后瞻。
零宽断言 | ||
---|---|---|
正则 | 名称 | 描述 |
(?=exp) | 正向前瞻 | 匹配exp前面的位置 |
(?!exp) | 负向前瞻 | 匹配后面不是exp的位置 |
(?<=exp) | 正向后瞻 | 匹配exp后面的位置不支持 |
(?<!exp) | 负向后瞻 | 匹配前面不是exp的位置不支持 |
正向前瞻用来检查接下来的出现的是不是某个特定的字符集。而负向前瞻则是检查接下来的不应该出现的特定字符串集。零宽断言是不会被捕获的。
1 | var str1 = "bedroom" ; |
题目,移除hr以外的所有标签,只留下innerText!
1 | var html = "<p><a href='http://www.cnblogs.com/rubylouvre/'>Ruby Louvre</a></p><hr/><p>by <em>司徒正美</em></p>" ; |
边界
一个要与字符类合用的东西。
边界 | ||
---|---|---|
正则 | 名称 | 描述 |
^ | 开头 | 注意不能紧跟于左中括号的后面 |
$ | 结尾 | |
\b | 单词边界 | 指[a-zA-Z_0-9]之外的字符 |
\B | 非单词边界 |
题目,设计一个字符串原型方法,实现首字母大写!
1 | var a = "ruby" ; |
单词边界举例。要匹配的东西的前端或未端不能为英文字母阿拉伯字数字或下横线。
1 | var str = "12w-eefd&efrew" ; |
实例属性 | 描述 |
---|---|
global | 是当前表达式模式首次匹配内容的开始位置,从0开始计数。其初始值为-1,每次成功匹配时,index属性都会随之改变。 |
ignoreCase | 返回创建RegExp对象实例时指定的ignoreCase标志(i)的状态。如果创建RegExp对象实例时设置了i标志,该属性返回True,否则返回False,默认值为False。 |
lastIndex | 是当前表达式模式首次匹配内容中最后一个字符的下一个位置,从0开始计数,常被作为继续搜索时的起始位置,初始值为-1, 表示从起始位置开始搜索,每次成功匹配时,lastIndex属性值都会随之改变。(只有使用exec()或test()方法才会填入,否则为0) |
multiLine | 返回创建RegExp对象实例时指定的multiLine标志(m)的状态。如果创建RegExp对象实例时设置了m标志,该属性返回True,否则返回False,默认值为False。 |
source | 返回创建RegExp对象实例时指定的表达式文本字符串。 |
1 | var str = "JS's Louvre" ; |
四、JavaScript字符串应用
4.1 回数
对称数即一个数倒序时,和原数值一样。例如:11,22,111等。
现在利用所学js知识写一个取某范围内所有对称数的函数。
1 | function reverseToNum(num){ //将该数值反转,取其反转后的值 |
4.2 首字母转化为大写
1 | function toUpCase(str){ |
Tips: Please indicate the source and original author when reprinting or quoting this article.