0

某一个字符串为'a_b_2_d_1'
我想匹配到_后面的字母与数字,但是不匹配_
于是我写了:

let str = "a_b_2_d_1"
let result = str.match(/(?:_)[a-z0-9]+/gi);
console.log(result); //["_b", "_2", "_d", "_1"]

结果将_也匹配了进去,match是与(?:)不生效呢,还是为什么。
如果不用这种方式还能怎么写呢?

查看全部 5 个回答

1

我觉得你是想用零宽断方。JS 的正则表达式支持正向零宽断言,不支持负向零宽断言(正向是向右),而你的需求是负向断言 _ 号,所以用零宽断言实现不了你的需求。

.split() 是一种比较取巧的解决办法,如果需求不扩展的话,够用了

如果实在需要通过正则表达式来取 _ 后的字母/数字,可以用 /_[a-z0-9]+?/g,得到 match 数组之后,再把前面的 _ 号去掉

s.match(/_[a-z0-9]+?/g).map(s => s.substr(1))
// ["b", "2", "d", "1"]

突然觉得我理解错了,你应该不是想用零宽断言,而是搞错了 (?:) 的意思。

在正则表达式不带 g 参数的时候,括号内的匹配内容会保存下来,比如

m = s.match(/(_)[a-z0-9]+?/)
// ["_b", "_"]

数组第一个元素是正则表达式匹配到的整个字符串,第二个元素是括号中的匹配到的内容。如果有多个括号,会依次保存在结果数组中

但是如果在括号里面前缀 ?:,表示不保存匹配内容

m = s.match(/(?:_)[a-z0-9]+?/)
// ["_b"]

结果只有整个

推荐答案

3
  1. 你可以去查下match的用法,match根据正则表达式里面有没有全局匹配g,会返回不同的结果,如果不是全局匹配,那么返回的结果是数组,第一个元素是匹配的完整字符串,从第二个元素开始是捕获型括号的内容;如果是全局匹配,返回的结果是一个数组,数组的每个元素都是匹配的完整字符串,没有捕获型括号的内容。

  2. 使用exec,代码如下:

    let str = "a_b_2_d_1"
    let re = /(?:_)([a-z0-9]+)/gi
    let match
    let result = []
    
    while (match = re.exec(str)) {
        result.push(match[1])
    }
    console.log(result)