转自:https://segmentfault.com/q/1010000009780103/a-1020000009781723
直接上例子:每三个数字中间加逗号
"123456789".replace(/(\d{3})(?:[^$])/g, "$1,");
//"123,567,9"
"123456789".replace(/(\d{3})(?=[^$])/g, "$1,");
//"123,456,789"
再上一个之前论坛里出现过的例子,也是每三个数字中间加逗号
先看看 (?=pattern) 的使用,下面这个是正确的:
function groupByCommas(n) {
return n.toString().replace(/\B(?=(\d{3})+(?!\d))/g, ",");
}
console.log(groupByCommas(1234567)); //1,234,567
如果我们把 ?= 换成 ?: 的话:
function groupByCommas(n) {
return n.toString().replace(/\B(?:(\d{3})+(?!\d))/g, ",");
}
console.log(groupByCommas(1234567)); //1,
两者的概念不用多说,查到有回答说:区别在于 ?= 是正向肯定 断言,进行的匹配是不占查询长度的;而 ?: 是非获取匹配,进行的匹配是占据查询长度的。
但是还是不是很理解这里的查询占据长度的说法,对着例子解释,难道是说第一个例子(?=[^$])匹配的是非结尾,所以123之后的非结尾的长度最小是1个字符,所以把4给一起替代了?那怎么不直接替代到结尾呢? 第二个例子(?=(\d{3})+(?!\d))匹配的是3或3的倍数个数字,直接匹配到了结尾,所以把234567也直接替代了?所以我的理解肯定是不对的
理解的不是很透彻,欢迎各位对着例子来解答一下我的困惑,不胜感激!
这个问题已被关闭,原因:问题已解决 - 问题已解决,且对他人无借鉴意义
- 关注 | 10
- 收藏 | 3
- 评论 · 5
- 编辑
山外de楼 1.1k
2017-06-14 提问
查看全部 5 个回答
答案对人有帮助,有参考价值1答案没帮助,是错误的答案,答非所问
(?=456)
匹配一个位置,这个位置后面跟了456
。
比如123(?=456)
会匹配123456
中的123
,而不会匹配123457
中的123
,不占用的意思是,至匹配123
后面的456
并不会被占用掉。123456
匹配的是123456
, 而123(?=456)456
同样匹配123456
后面加了(?=456)
其实是没有什么意义的。
正则中可以用括号改变优先级等,另外,对于加括号的部分,会从左到右分配递增的分配一个编号,在后面可以用编号引用这一部分匹配到的文本。在JS replace
里,替换的部分可以用$1
之类的引用这一部分的匹配。
比如(a)\1
会匹配两个连续的a,([A-Z])\1
匹配两个连续相同的大小字母,(A-Z)\1([a-z])\2
匹配两个连续的大小字母,后面跟两个连续的小写字母(大小写字母可以不同)。
有时候,我们只想改变优先级,不想分配编号(很少用到),就用(?:)
比如(a)(?:b)(c)\1\2
匹配abcac
,但是(a)(b)(c)\1\2
匹配abcab
.