正则总结二(正则表达式位置匹配攻略附代码详解)

前言

正则表达式是匹配模式,要么匹配字符,要么匹配位置。

2.1什么是位置

位置是相邻字符之间的位置。

2.2如何匹配位置

在ES5中,共有6个锚字符

^ $ \b \B (?=p) (?!p)

2.2.1 $和^

  • ^(脱字符)匹配开头,在多行匹配中匹配行开头
  • $(美元符号)匹配结尾,在多行匹配中匹配行结尾

比如我们把字符串的开头和结尾用"#"替换(位置可以替换成字符的!

var result = "hello".replace(/^|$/g, '#');
console.log(result); 
// => "#hello#"

多行匹配模式时,二者是行的概念,这个需要我们的注意

var result = "I\nlove\njavascript".replace(/^|$/gm, '#');
console.log(result);
/*
#I#
#love#
#javascript#
*

2.2.2 \b和\B

\b是单词边界,具体就是\w和\W之间的位置,也包括\w和^之间的位置,也包括\w和$之间的位置

  • 比如一个文件名是"[JS] Lesson_01.mp4"中的\b,如下
var result = "[JS] Lesson_01.mp4".replace(/\b/g, '#');
console.log(result); 
// => "[#JS#] #Lesson_01#.#mp4#"

2.2.3 (?=p)和(?!p)

  • (?=p),其中p是一个子模式,即p前面的位置
  • 比如(?=l),表示’l’字符前面的位置
var result = "hello".replace(/(?=l)/g, '#');
console.log(result); 
// => "he#l#lo"

而(?!p)就是(?=p)的反面意思,比如

var result = "hello".replace(/(?!l)/g, '#');

console.log(result); 
// => "#h#ell#o#"

分别是正向先行断言和负向先行断言,具体是(?<=p)和(?<!p)
(?=p)就与^一样好理解,就是p前面的那个位置

2.3位置的特性

  • 对于位置的理解,我们可以理解成空字符""
  • 比如"hello"字符串等价于如下的形式
"hello" == "" + "h" + "" + "e" + "" + "l" + "" + "l" + "o" + ""

等价于

"hello" == "" + "" + "hello"

因此,把/^helloKaTeX parse error: Expected group after '^' at position 5: /写成/^̲^hello$$/,是没有任何问题的

var result = /^^hello$$$/.test("hello");
console.log(result); 
// => true

2.4 相关案例

2.4.1 不匹配任何东西的正则

/.^/

2.4.2 数字的千位分隔符表示法

比如把"12345678",变成"12,345,678"

弄出最后一个逗号

使用(?=\d{3}$)就可以做到

var result = "12345678".replace(/(?=\d{3}$)/g, ',')
console.log(result); 
// => "12345,678"

弄出所有的逗号

因为逗号出现的位置,要求后面3个数字一组,也就是\d{3}至少出现一次

  • 此时可以使用量词+
var result = "12345678".replace(/(?=(\d{3})+$)/g, ',')
console.log(result); 
// => "12,345,678"

匹配其余案例

写完正则后,要多验证几个案例,此时我们会发现问题

var result = "123456789".replace(/(?=(\d{3})+$)/g, ',')
console.log(result); 
// => ",123,456,789"
  • 因为上面的正则,仅仅表示把从结尾向前数,一但是3的倍数,就把其前面的位置替换成逗号。因此才会出现这个问题

  • 我们知道匹配开头可以使用^,但要求这个位置不是开头怎么办

  • (?!^),你想到了吗?测试如下

    var string1 = "12345678",
    string2 = "123456789";
    reg = /(?!^)(?=(\d{3})+$)/g;
    
    var result = string1.replace(reg, ',')
    console.log(result); 
    // => "12,345,678"
    
    result = string2.replace(reg, ',');
    console.log(result); 
    // => "123,456,789"
    

    支持其他形式

如果要把"12345678 123456789"替换成"12,345,678 123,456,789"。

  • 此时我们需要修改正则,把里面的开头^和结尾$,替换成\b
var string = "12345678 123456789",
reg = /(?!\b)(?=(\d{3})+\b)/g;

var result = string.replace(reg, ',')
console.log(result); 
// => "12,345,678 123,456,789"
  • 其中(?!\b)怎么理解呢?
  • 要求当前是一个位置,但不是\b前面的位置,其实(?!\b)说的就是\B
  • 因此最终正则变成了:/\B(?=(\d{3})+\b)/g

2.4.3 验证密码问题

  • 密码长度6-12位,由数字、小写字符和大写字母组成,但必须至少包括2种字符。
  • 此题,如果写成多个正则来判断,比较容易。但要写成一个正则就比较困难。
  • 那么,我们就来挑战一下。看看我们对位置的理解是否深刻

简化

不考虑“但必须至少包括2种字符”这一条件。我们可以容易写出

var reg = /^[0-9A-Za-z]{6,12}$/;

判断是否包含有某一种字符

假设,要求的必须包含数字,怎么办?此时我们可以使用(?=.*[0-9])来做

var reg = /(?=.*[0-9])^[0-9A-Za-z]{6,12}$/;

同时包含具体两种字符

  • 比如同时包含数字和小写字母,可以用(?=.*[0-9])(?=.*[a-z])来做
var reg = /(?=.*[0-9])(?=.*[a-z])^[0-9A-Za-z]{6,12}$/;

我们可以把原题变成下列几种情况之一

  • 同时包含数字和小写字母
  • 同时包含数字和大写字母
  • 同时包含小写字母和大写字母
  • 同时包含数字、小写字母和大写字母
var reg = /((?=.*[0-9])(?=.*[a-z])|(?=.*[0-9])(?=.*[A-Z])|(?=.*[a-z])(?=.*[A-Z]))^[0-9A-Za-z]{6,12}$/;
console.log( reg.test("1234567") ); // false 全是数字
console.log( reg.test("abcdef") ); // false 全是小写字母
console.log( reg.test("ABCDEFGH") ); // false 全是大写字母
console.log( reg.test("ab23C") ); // false 不足6位
console.log( reg.test("ABCDEF234") ); // true 大写字母和数字
console.log( reg.test("abcdEF234") ); // true 三者都有

解惑

  • /(?=.*[0-9])^[0-9A-Za-z]{6,12}$/ 对于这个正则,我们只需要弄明白(?=.*[0-9])^即可
  • 分开来看就是(?=.*[0-9])和^
  • 表示开头前面还有个位置
  • (?=.*[0-9])表示该位置后面的字符匹配.*[0-9],即,有任何多个任意字符,后面再跟个数字

猜你喜欢

转载自blog.csdn.net/qq_52151772/article/details/115504556