看到这个标题是不是虎躯一震,啥玩意啊?别着急,别看这个名词生僻,下面的…更加生僻哈哈哈。
言归正传,不管它叫啥,我们会用就行了。零宽嘛,就是没有宽度,只匹配位置,不匹配内容,是不是想起来了,对的,就是类似于^、$、\b这种。
零宽度正预测先行断言
(?=exp)
,断言自身出现的位置的后面能匹配表达式exp。
例如,我需要匹配以ing结尾的每个单词。
"hello sing dancing".match(/\b[a-z]+(?=ing\b)/g) //["s", "danc"]
零宽度正回顾后发断言
(?<=exp)
,它断言自身出现的位置的前面能匹配表达式exp。
例如,匹配以wo开头的单词后半部分。
import java.util.regex.Matcher;
import java.util.regex.Pattern;
public class regularTest {
public static void main(String[] args) {
// TODO Auto-generated method stub
Pattern pattern2 = Pattern.compile("(?<=\\bbo)\\w+\\b");
Matcher m2 = pattern2.matcher("hello world java");
while(m2.find()){
System.out.println(m2.group()); //rld
}
}
}
注意:这里为什么我要用Java写呢,因为JavaScript不支持这个语法。
"hello world java".match(/(?<=\bwo)\w+\b/g) //Uncaught SyntaxError: Invalid regular expression: /(?<=\bwo)\w+\b/: Invalid group
不过现在字符串既会在前台验证,也会在后台验证,只要可以验证就行了对吧。
零宽度负预测先行断言
(?!exp)
,断言此位置的后面不能匹配表达式exp。
如果我们想要匹配一个单词,这个单词中有一个字母a,a后面不是字母b,就像apple,hallo,但不能是absolute。
/\b[a-z]*a(?!b)[a-z]*\b/.test("absolute") //false
/\b[a-z]*a(?!b)[a-z]*\b/.test("apple") //true
这个东西还可以用在判断密码上。密码要求:同时含有数字和字母,且长度要在3-5位之间。
/(?![0-9]+$)(?![a-zA-Z]+$)[0-9A-Za-z]{3,5}$/.test("123") //false
/(?![0-9]+$)(?![a-zA-Z]+$)[0-9A-Za-z]{3,5}$/.test("abc") //false
/(?![0-9]+$)(?![a-zA-Z]+$)[0-9A-Za-z]{3,5}$/.test("abc12") //true
零宽度负回顾后发断言
(?<!exp)
,断言此位置的后面不能匹配表达式exp。
例如,我们要匹配一个七位数字,该七位数字之前不能有字母。
import java.util.regex.Matcher;
import java.util.regex.Pattern;
public class regularTest {
public static void main(String[] args) {
// TODO Auto-generated method stub
Pattern pattern = Pattern.compile("(?<![a-z])\\d{7}");
Matcher m = pattern.matcher("1a234567890");
while(m.find()){
System.out.println(m.group()); //3456789
}
}
}
JavaScript也不支持这个。
最后
- 对于正则表达式,不同的语言有不同的实现方式。不过Java的还是比JavaScript强大一点。
- JavaScript中,利用正则对象函数的构造调用和直接使用字面量写法有区别。
var re = new RegExp("\\w+");
var re = /\w+/;
- 以上代码都经过测试,水平有限,如有错误和疑问,请微博私信。