javascript系列--正则表达式进修(二)位置婚配

文章首发于
sau交流学习社区

一、媒介

正则表达式是婚配情势,要么是婚配字符,要么婚配位置。

实在在开辟中很少用到婚配位置,本篇文章重要包含:

二、什么是位置

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

《javascript系列--正则表达式进修(二)位置婚配》

三、怎样婚配位置

在ES5中,共有6个锚:^, $, b, B, (?=p), (?!p)

可视化情势:

RegExp:/^$bB(?=a)(?!b)/g

《javascript系列--正则表达式进修(二)位置婚配》

3.1 ^和$

^(脱字符)婚配开首,在多行婚配中婚配行开首。

$(美圆符)婚配末端,在多行婚配中婚配行末端。

比方:我们把字符串的开首和末端用#替代(位置能够替代成字符的):

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

多行婚配情势(有修饰符m)时,两者是行的观点,我们须要注重:

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

3.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#"

起首,我们晓得w是字符组[0-9a-zA-Z]的简写,纵然字母数字或许下划线中任何一个字符。而W是字符组1的简写,即W是w之外的任何一个字符。

我们再来看#是怎样来的:

第 1 个,双方字符是 “[” 与 “J”,是 W 与 w 之间的位置。

第 2 个,双方字符是 “S” 与 “]”,也就是 w 与 W 之间的位置。

第 3 个,双方字符是空格与 “L”,也就是 W 与 w 之间的位置。

第 4 个,双方字符是 “1” 与 “.”,也就是 w 与 W 之间的位置。

第 5 个,双方字符是 “.” 与 “m”,也就是 W 与 w之间的位置。

第 6 个,位于末端,前面的字符 “4” 是 w,即 w 与 $ 之间的位置。

晓得了b观点后,那B就好邃晓了,b是单词边境,B黑白单词边境

var result = "[JS] Lesson_01.mp4".replace(/\B/g, '#');
console.log(result);
// "#[J#S]# L#e#s#s#o#n#_#0#1.m#p#4"

3.3 (?=p) 和 (?!p)

(?=p)个中p是一个子情势,即p前面的位置(该位置背面的字符要婚配p)

比方:(?=e),示意的是e字符前面的位置;

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#"

两者的学名分别是 positive lookahead 和 negative lookahead。

中文翻译分别是正向先行断言和负向先行断言。

ES5 以后的版本,会支撑 positive lookbehind 和 negative lookbehind。

详细是 (?<=p) 和 (?<!p)。

四、位置特征

关于位置的邃晓,我们能够累计额成空字符””。

比方”hello”字符串等价于以下情势:

“hello” == “” + “h” + “” + “e” + “” + “l” + “” + “l” + “” + “o” + “”;
也等价于

“hello” == “” + “” + “hello”
因而,把 /^hello$/ 写成 /^^hello$$$/,是没有任何题目的:

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

也就是说,字符之间的位置,能够写成多个。

注:把位置邃晓空字符,是对位置异常有用的邃晓体式格局

五、相干案例

5.1不婚配任何东西的正则

/.^/
正则请求只要一个字符,然则该字符背面是开首,而如许的字符串是不存在的。

5.2数字的千分位分隔符示意法

比方吧12345678,变成12,345,678。

剖析:那就是须要把响应的位置替代成”,”

5.2.1弄出来末了一个逗号

正则:/(?=d{3}$)/

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

个中(?=d{3}$)婚配\d{3}$前面的位置。而d{3}$婚配的是目的字符串末了那3为数字。

5.2.2弄出来多有逗号

由于逗号的涌现的位置,请求后边3个数字一组,也就是d{3}最少涌现1次。

能够运用量词 + :

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

5.2.3婚配其他案例

写完正则后,是须要举一般案例来考证的,就会发现题目:

var result = "123456789".replace(/(?=(\d{3})+$)/g, ',')
console.log(result);
// ,123,456,789

上面的正则,仅仅是示意把从末端向前数,一旦是3的倍数买酒吧前面的位置替代为逗号。

还须要请求:婚配的这个位置不能是开首。

我们晓得开首的婚配是运用^,然则不是开首怎样整?

运用(?!^)

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

5.2.4支撑其他情势

假如要把 “12345678 123456789” 替代成 “12,345,678 123,456,789″。

此时我们须要修正正则,把内里的开首 ^ 和末端 $,修正成 b:

var string = "12345678 123456789",
regex = /(?!\b)(?=(\d{3})+\b)/g;
var result = string.replace(regex, ',')
console.log(result);
// "12,345,678 123,456,789"

个中(?!b)怎样邃晓?

请求是当前的一个位置,但不是b前面的位置,实在(?!b)说的是B。

因而终究正则变成了:/B(?=(d{3})+b)/g。

可视化情势:

RegExp:/B(?=(d{3})+b)/g

《javascript系列--正则表达式进修(二)位置婚配》

5.2.5钱银格式化

千分符示意法一个罕见的运用就是钱银格式化。

把这个字符串:

1888
格式化:

$ 1888.00
完成:

function format (num) {
return num.toFixed(2).replace(/\B(?=(\d{3})+\b)/g, ",").replace(/^/, "$ ");
};
console.log( format(1888) );
// "$ 1,888.00"

5.3考证暗码的题目

暗码长度6-12,由数字,小写字母,大写字母构成,但必需最少包含2种字符。

写成多个正则来推断,比较轻易,然则要写成一个正则就比较难题。

来看看我们关于位置的邃晓是不是深入。

5.3.1简化

暂时不斟酌”必需最少包含2种字符”这个前提,能够轻易写出:

var regex = /^[0-9A-Za-z]{6,12}$/;
5.3.2推断是不是包含有某一种字符

假定,请求我们必需包含数字,怎样整?此时我们能够运用(?=.*[0-9])来完成。

正则变成:

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

5.3.3同时包含详细两种字符

比方同时包含数字和小写字母,能够运用(?=.[0-9])(?=.[a-z])来完成。

正则变成:

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

5.3.3详细完成

把原题变成以下几种状况:

1、同时包含数字和小写字母;

2、同时包含数字和大写字母;

3、同时包含小写字母和大写字母;

4、同时包含数字,小写字母和大写字母。

以上的4中状况是或的关联(实际上,第四条能够不必)。

终究答案:

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

可视化情势:

《javascript系列--正则表达式进修(二)位置婚配》

RegExp:/((?=.[0-9])(?=.[a-z])|(?=.[0-9])(?=.[A-Z])|(?=.[a-z])(?=.[AZ]))^[ 0-9A-Za-z]{6,12}$/

剖析:

上面正则看起来好庞杂,只须要邃晓第二步,/(?=.*[0-9])^[0-9A-Za-z]{6,12}$/;

关于这个正则,我们须要邃晓(?=.*[0-9])^这个;

离开来看就是(?=.*[0-9]) 和 ^。

示意开首前面另有个位置(固然也是开首,即同一个位置,想一想之前的空字符类比)。

(?=.*[0-9]) 示意该位置背面的字符婚配。

.*[0-9],即,有任何多个恣意字符,背面再跟个数字。

翻译成大白话,就是接下来的字符,必需包含个数字。

5.3.4别的一种解法

“最少包含两种字符”的意义就是说,不能全部都是数字,也不能全部都是小写字母,也不能全部都是大写字母。

那末请求“不能全部都是数字”,怎样做呢? (?!p) 出马!

对应的正则:

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

三种“都不能”呢?

终究答案是:

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

可视化情势:

RegExp:/(?!^[0-9]{6,12}$)(?!^[a-z]{6,12}$)(?!^[A-Z]{6,12}$)^[0-9A-Za-z]{6,12}$/;

《javascript系列--正则表达式进修(二)位置婚配》

  1. 0-9a-zA-Z
    原文作者:saucxs
    原文地址: https://segmentfault.com/a/1190000019191923
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞