问题
新写一个正则,比如/^[A-Za-z]+[A-Za-z0-9\_]*$/g
,在调用exec和test方法时交替返回true和false,乍一看还以为正则出了什么问题。
const reg = /^[A-Za-z]+[A-Za-z0-9\_]*$/g;
const str = 'a_1';
reg.test(str);
// 返回 true
reg.test(str);
// 返回 false
思路
- 先了解下
exec
和test
方法
方法 | 描述 |
---|---|
exec | 一个在字符串中执行查找匹配的RegExp方法,它返回一个数组(未匹配到则返回null)。 |
test | 一个在字符串中测试是否匹配的RegExp方法,它返回true或false。 |
- 了解下修饰符
g
和lastIndex
修饰符
标志 | 描述 |
---|---|
g | 全局搜索。 |
属性
属性或索引 | 描述 |
---|---|
lastIndex | 下一个匹配的索引值。(这个属性只有在使用g参数时可用) |
重点
lastIndex 属性用于规定下次匹配的起始位置。
注意: 该属性只有设置标志 g 才能使用。 上次匹配的结果是由方法 RegExp.exec() 和 RegExp.test() 找到的,它们都以 lastIndex 属性所指的位置作为下次检索的起始点。这样,就可以通过反复调用这两个方法来遍历一个字符串中的所有匹配文本。
注意:该属性是可读可写的。 只要目标字符串的下一次搜索开始,就可以对它进行设置。当方法 exec() 或 test() 再也找不到可以匹配的文本时,它们会自动把 lastIndex 属性重置为 0。
解决方案
exec
和test
使用正则带有修饰符g
时,每一次执行都会把 lastIndex 属性所指的位置作为下次检索的起始点,那么我们执行时手动重置该属性位置即可正常返回查询boolean值。
const reg = /^[A-Za-z]+[A-Za-z0-9\_]*$/g;
function test(reg, str) {
const result = reg.test(str);
reg.lastIndex = 0; // 重置该位置
return result;
}
test(reg, 's_1');
// true
test(reg, 's_1');
// true