
本文转载自微信公众号「微医大前端技术」,聊聊作者焦传锴 。极速转载本文请联系微医大前端技术公众号。通关
先推荐两个测试正则网站:
https://regex101.com/ https://jex.im/regulex/正则表达式,常用大家都很熟悉,正则说白了就是聊聊一堆约定俗成的匹配规则,包含模式和可选的极速修饰符。创建一个正则表达式对象有两种语法。通关较长一点的常用语法:
regexp = new RegExp("pattern", "flags");较短一点的语法,使用斜杠 "/":
regexp = /pattern/; // 没有修饰符 regexp = /pattern/gmi; // 伴随修饰符 g、正则m 和 i(后面会讲到)斜杠 "/" 会告诉 JavaScript 我们正在创建一个正则表达式。聊聊它的极速作用类似于字符串的引号。new RegExp 可以动态传入参数创建正则。通关
本文的常用主要目的就是为了方便快速查找并理解需要的正则规则。
常用的字符类:
\d 数字 \s 空格 space \w 单字符 word,包括字母数字下划线可以组合,比如:
let str = "test ES 6 AA" let reg = /e\w\s\d/i str.match(reg) // ["ES 6", index: 6, input: "test ES 6 AA", groups: undefined]每个字符类都有反向类,代表 非xx
\D 非数字 \S 非空格 \W 非单字符 let str = "+7(903)-123-45-67"; alert( str.replace(/\D/g, "") ); // 79031234567 . 匹配任意字符(换行符除外) "/ES./" \b 查找目标“词”是否在边界,IT技术网比如 /\bjava\b/ 可以匹配 !java! 但是不能匹配 javac二者结合可以用以完全匹配
const time = "12:02" let reg = /^\d\d:\d\d$/ // .test 可以测试是否匹配 reg.test(time) // true空字符串 ,可以用 /^$/ 匹配
a|b 相当于 [ab] ,我们可以这样使用:
gr(a|e)y 严格等同 gr[ae]y。 gra|ey 匹配 “gra” or “ey”。看一个例子
let str = `"hi" some word "ok" aa` let reg = /".+"/g str.match(reg) //["hi" some word "ok"]我们其实是想匹配出 ["hi","ok"] ,但是却匹配到了整句,这是因为 贪婪搜索 会先按顺序分别取匹配 " . +
当匹配 " 的时候,匹配到第一个引号,此时匹配字符串是 " 当匹配 . 的云南idc服务商时候,匹配字符串是 "h 当匹配 + 的时候,字符串变为了 "hi" some word "ok" aa !因为后面所有的字符都符合 .+ 的规则,即不包含换行的任意字符 此时匹配 " ,发现已经匹配多了,找不到 " ,于是开始 回溯 ,知道回溯成为 "hi" some word "ok"这就是 贪婪模式 。
再看一个例子:
let str = `123 456` let reg1 = /\d+ \d+?/ let reg2 = /\d+ \d+/ str.match(reg1) // 123 4 str.match(reg2) // 123 456在量词之后加上 ? ,即 .? +? ?? 等,会变为 懒惰模式 ,他不会一次性完全匹配,而是在匹配到满足条件的第一位时就停止匹配。
组
举个例子:
let str = "gogogoaa" let reg = /(go)+/ str.match(reg) // gogogo很好理解,就是将多个字符算成一个整体进行匹配
接下来看几个例子
域名匹配 /([\w-]+\.)+\w+/g 可以匹配的格式 aaa.aaa.aa aa-aa.aaa.aa email /[-.\w]+@([\w-]+\.)+[\w-]+/g(xx) 被称为 组(group) 的概念,括号内的内容不仅匹配时被作为一个整体,并且组内匹配的对象会被返回:
let str = <h1>Hello, world!</h1>; let tag = str.match(/<(.*?)>/); alert( tag[0] ); // <h1> alert( tag[1] ); // h1返回的结果数组, [0] 的源码下载位置是正常全匹配返回的值,而 [1] 的位置是括号内匹配到到的值。我们可以用这个方法做 嵌套组 :
let str = `<group1 group2>` let arr = str.match(/<((\w+)\s(\w+))>/) console.log(arr[0]) //<group1 group2> console.log(arr[1]) //group1 group2 console.log(arr[2]) //group1 console.log(arr[3]) //group2 let match = ac.match(/a(z)?(c)?/) alert( match.length ); // 3 alert( match[0] ); // ac(完全匹配) alert( match[1] ); // undefined,因为 (z)? 没匹配项 alert( match[2] ); // c上述都是在没有 g 标签时匹配的单个对象返回的数组,那么如果有 g 会返回多个对象的话,可以用 matchAll 来匹配:
let str = `<group1> <group2>` let arr = Array.from(str.matchAll(/<(group\d)>/g)) arr[0][0] // <group1> arr[0][1] // group1 arr[1][0] // <group2> arr[1][1] // group2注意, matchAll 返回的不是数组,而是一个可迭代的对象。
把上面的例子稍微修改
let str = `<group1 group2>` let arr = str.match(/<(?<g0>(?<g1>\w+)\s(?<g2>\w+))>/) let groups = arr.groups console.log(arr[0]) //<group1 group2> console.log(groups.g0) //group1 group2 console.log(groups.g1) //group1 console.log(groups.g2) //group2我们可以通过 在括号后立即加上 ? 的方式设置 group 名,通过返回数组的 groups 属性获取一个 group 对象
方法 str.replace(regexp, replacement) 用 replacement 替换 str 中匹配 regexp 的所有捕获组。这使用 $n 来完成,其中 n 是组号。例如,
let str = "John Bull"; let regexp = /(\w+) (\w+)/; alert( str.replace(regexp, $2, $1) ); // Bull, John对于命名括号,引用为 $。例如,让我们将日期格式从 “year-month-day” 更改为 “day.month.year”:
let regexp = /(?[0-9]{4})-(?[0-9]{2})-(?[0-9]{2})/g; let str = "2019-10-30, 2020-01-01"; alert( str.replace(regexp, $.$.$) ); // 30.10.2019, 01.01.2020我们需要找到带引号的字符串:单引号 ... 或双引号 "..."– 应匹配两种变体。然后我们有一句话 "Shes the one!" ,这时候如果我们用 /["](.*?)["]/g ,则会匹配到 "She ,显然不对
那么问题在于,我们怎么让正则记住我们某一个分组中捕获的内容 这时候可以使用 反向引用
let str = `He said: "Shes the one!".`; let regexp = /(["])(.*?)\1/g; alert( str.match(regexp) ); // "Shes the one!"这里的 \1 会找到第一个 group ,也就是 (["]) 匹配到的内容,也就是 " ,然后这个正则就相当于变成了 /(["])(.*?)"/g
我们还可以用 \k 的方式去引用:
let str = `He said: "Shes the one!".`; let regexp = /(?["])(.*?)\k/g; alert( str.match(regexp) ); // "Shes the one!"前瞻断言
用法:
x(?=y) 仅当 x 后面是 y 的时候匹配 let str = "1 turkey costs 30€"; alert( str.match(/\d+(?=€)/) ); // 30 (正确地跳过了单个的数字 1)x(?!y) 仅当 x 后面不是 y 的时候匹配
后瞻断言
(?<=y)x, 匹配 x, 仅在前面是 y 的情况。 (?<!y)x, 匹配 x, 仅在前面不是 y 的情况。断言仅仅是作为占位,不会匹配字符,比如 /q(?=u)i/ 匹配 quit 是会失败的,因为 /q(?=u)/ 只能匹配到 q 而不是 qu
捕获组
如果我们想要捕捉整个环视表达式或其中的一部分,那也是有可能的。只需要将其包裹在另加的括号中。例如,这里货币符号 (€|kr) 和金额一起被捕获了:
let str = "1 turkey costs 30€"; let reg = /\d+(?=(€|kr))/; // €|kr 两边有额外的括号 alert( str.match(reg) ); // 30, €参考
The Modern JavaScript Tutorial
别忘记对我素质三连,点赞、关注、评论^_^
前往微医互联网医院在线诊疗平台,快速问诊,3分钟为你找到三甲医生。(https://wy.guahao.com/?channel=influence)
(责任编辑:数据库)