Java正则表达式之应用篇


本文将介绍如何在Java中使用正则表达式来处理文本数据。正则表达式就是一个字符串,但和普通的字符串不同的是,正则表达式是对一组相似字符串的抽象,如下面的几个字符串:

 
a98b   c0912d   c10b   a12345678d   ab
 
    我们仔细分析上面五个字符串,可以看出它们有一个共同特征,就是第一个字符必须是'a'或'c',最后一个字符必须是'b'或'd',而中间的字符是任意多个数字组成(包括0个数字)。因此,我们可以将这五个字符串的共同特点抽象出来,这就产生了一个正则表达式:[ac]//d*[bd]。而根据这个正则表达式,我们可以写出无穷多个满足条件的字符串。
 
在Java中使用正则表达式的方法非常多,最简单的就是和字符串一起使用。在String中有四个方法可以使用正则表达式,它们是matches、split、replaceAll和replaceFirst。
 
一、matches方法
 
matches方法可以判断当前的字符串是否匹配给定的正则表达式。如果匹配,返回true,否则,返回false。matches方法的定义如下:

public boolean matches(String regex)
  
  如上面给出的正则表达式我们可以用如下程序验证。
 

 

 

String[] ss = new String[]{"a98b", "c0912d",  "c10b",  "a12345678d",  "ab"};
for(String s: ss)
    System.out.println(s.matches("[ac]//d*[bd]"));

 


输出结果:
 
true
true
true
true
true
 
   下面简单解释一下这个正则表达式的含义。如果我们学过编译原理的词法分析,就会很容易理解上面的正则表达式(因为正则表达式的表示方法和词法分析中的表达式类似)。如在 [...]中的相当于或"|",如[abcd]相当于a|b|c|d,也就是a或b或c或d。如上面的正则表达式的开头部分是[ac],就代表着字符串的开头只能是a或c。[bd]表达字符串结尾只能是b或d。而中间的/d表达0-9的数字,由于/在正则表达式中有特殊含义,所以用//来表示/。而*表示有0或无穷多个(这在词法分析中叫*闭包),由于*跟在/d后面,因此表达有0或无穷多个数字。
 
二、split方法
 
split方法使用正则表达式来分割字符串,并以String数组的形式返回分割结果。split有两种重载形式,它们定义如下:
 

 

public String[] split(String regex)
public String[] split(String regex, int limit)

 


    如下面的代码将使用split的第一种重载形式来分割HTTP请求头的第一行,代码如下:
 

 

String s = "GET /index.html HTTP/1.1";
String ss[] = s.split(" +");
for(String str: ss)
System.out.println(str);

 


输出结果:
GET
/index.html
HTTP/1.1
 
    在使用split的第一种重载形式时应注意,如果分割后的字符串最后有空串,将被忽略。如使用正则表达式/d来分割字符串a0b1c3456时,得到的数组的长度为3,而不是7。
在split的第二种重载形式中有一个limit参数,要分三种情况讨论:
 
1. 大于0: 如limit的值为n,那么将对正则表达式使用n-1次,下面的代码:

 

String s = "a0b1c3456";
String ss[] = s.split("//d", 3);
for(String str: ss)
    System.out.println(str);

 



输出结果:
 
a
b
c3456
 
从输出结果可以看出,程序只对" a0b1c3456"使用了两次正则表达式,也就是在少扫描完字符'1'后,不管后面有没有满足条件的字符串,都将后面的字符串作为一个整体来作为返回数组的最后一个值。
 
2. 小于0: 不忽略结尾的空串。也就是上面的例子返回数组的长度应该是7,而不是3。
3. 等于0:这是默认值,相当于split的第一种重载形式。
 
 
三、replaceAll 和 replaceFirst方法
 
为两个方法的定义如下:
public String replaceAll(String regex, String replacement)
public String replaceFirst(String regex, String replacement)
 
    这两个方法用replacement替换当前字符串中和regex匹配的字符串。使用方法很简单,这里不再详述,感兴趣的读者可以参考相关的文档。
 
在Java中,我们为了查找某个给定字符串中是否有需要查找的某个字符或者子字串、或者对字符串进行分割、或者对字符串一些字符进行替换/删除,一般会通过if-else、for 的配合使用来实现这些功能 。如下所示:

 

Java代码
  1. public class Test{   
  2.   public static void main(String args[]) {   
  3.          String str="@Shang Hai Hong Qiao Fei Ji Chang";   
  4.          boolean rs = false;   
  5.          for(int i=0;i<str.length();i++){   
  6.                 char z=str.charAt(i);     
  7.                 if('a' == z || 'F' == z) {   
  8.                        rs  = true;   
  9.                        break;   
  10.                 }else{   
  11.                        rs= false;   
  12.                 }   
  13.          }      
  14.          System.out.println(rs);   
  15.        }   
  16. }  

      这种方法使用简单直观,但是 难以解决复杂的工作,而且代码量也会增加很多,不利于维护。

 

      这时,我们可以使用正则表达式来实现这些功能,而且代码简单易维护。下面就来介绍了Java中对字符串的正则表达式的几个常用的功能,具体情况如下所示(其中用到了java.util.regex包):

 

1.Java中在某个字符串中查询某个字符或者某个子字串

Java代码
  1. String s = "@Shang Hai Hong Qiao Fei Ji Chang";      
  2. String regEx = "a|F"; //表示a或F   
  3. Pattern pat = Pattern.compile(regEx);   
  4. Matcher mat = pat.matcher(s);   
  5. boolean rs = mat.find();   

    如果s中有regEx,那么rs为true,否则为flase。

    如果想在查找时忽略大小写,则可以写成Pattern pat=Pattern.compile(regEx,Pattern.CASE_INSENSITIVE);

 

2.在某个文件中获取一段字符串

Java代码 
  1. String regEx = ".+/(.+)$";   
  2. String s = "c:/test.txt";   
  3. Pattern pat = Pattern.compile(regEx);   
  4. Matcher mat = pat.matcher(s);   
  5. boolean rs = mat.find();   
  6. for(int i=1;i<=mat.groupCount();i++){   
  7.   System.out.println(mat.group(i));   
  8. }   

  以上的执行结果为test.txt,提取的字符串储存在mat.group(i)中,其中i最大值为mat.groupCount();

 

3.对字符串的分割

Java代码 
  1. String regEx=":";   
  2. Pattern pat = Pattern.compile(regEx);   
  3. String[] rs = pat.split("aa:bb:cc");   

  执行后,r就是{"aa","bb","cc"}

    如果用正则表达式分割就如上所示,一般我们都会使用下面更简单的方法:

Java代码 
  1. String s = "aa:bb:cc";   
  2. String[] rs=s.split(":");   

 

4.字符串的替换/删除

Java代码 
  1. String regEx="@+"; //表示一个或多个@   
  2. Pattern pat=Pattern.compile(regEx);   
  3. Matcher mat=pat.matcher("@@aa@b cc@@");   
  4. String s=mat.replaceAll("#");   

 结果为"##aa#b cc##"
  
 如果要把字符串中的@都给删除,只用要空字符串替换就可以了:

Java代码 
  1. String s=mat.replaceAll("");  

  结果为"aab cc"

 

 

注:对Pattern类的说明: 
      1.public final class java.util.regex.Pattern是正则表达式编译后的表达法。

      下面的语句将创建一个Pattern对象并赋值给句柄pat:Pattern pat = Pattern.compile(regEx);
      有趣的是,Pattern类是final类,而且它的构造器是private。也许有人告诉你一些设计模式的东西,或者你自己查有关资料。这里的结论是:Pattern类不能被继承,我们不能通过new创建Pattern类的对象。
       因此在Pattern类中,提供了2个重载的静态方法,其返回值是Pattern对象(的引用)。如:

Java代码 
  1. public static Pattern compile(String regex) {   
  2.         return new Pattern(regex, 0);   
  3. }  

       当然,我们可以声明Pattern类的句柄,如Pattern pat = null;

 

    2.pat.matcher(str)表示以用Pattern去生成一个字符串str的匹配器,它的返回值是一个Matcher类的引用。
       我们可以简单的使用如下方法:boolean rs = Pattern.compile(regEx).matcher(str).find();

 

 

附 : 常用的正则表达式:

匹配特定数字:
^[1-9]d*$    //匹配正整数
^-[1-9]d*$   //匹配负整数
^-?[1-9]d*$   //匹配整数
^[1-9]d*|0$  //匹配非负整数(正整数 + 0)
^-[1-9]d*|0$   //匹配非正整数(负整数 + 0)
^[1-9]d*.d*|0.d*[1-9]d*$   //匹配正浮点数
^-([1-9]d*.d*|0.d*[1-9]d*)$  //匹配负浮点数
^-?([1-9]d*.d*|0.d*[1-9]d*|0?.0+|0)$  //匹配浮点数
^[1-9]d*.d*|0.d*[1-9]d*|0?.0+|0$   //匹配非负浮点数(正浮点数 + 0)
^(-([1-9]d*.d*|0.d*[1-9]d*))|0?.0+|0$  //匹配非正浮点数(负浮点数 + 0)
评注:处理大量数据时有用,具体应用时注意修正

匹配特定字符串:
^[A-Za-z]+$  //匹配由26个英文字母组成的字符串
^[A-Z]+$  //匹配由26个英文字母的大写组成的字符串
^[a-z]+$  //匹配由26个英文字母的小写组成的字符串
^[A-Za-z0-9]+$  //匹配由数字和26个英文字母组成的字符串
^w+$  //匹配由数字、26个英文字母或者下划线组成的字符串

在使用RegularExpressionValidator验证控件时的验证功能及其验证表达式介绍如下:

只能输入数字:“^[0-9]*$”
只能输入n位的数字:“^d{n}$”
只能输入至少n位数字:“^d{n,}$”
只能输入m-n位的数字:“^d{m,n}$”
只能输入零和非零开头的数字:“^(0|[1-9][0-9]*)$”
只能输入有两位小数的正实数:“^[0-9]+(.[0-9]{2})?$”
只能输入有1-3位小数的正实数:“^[0-9]+(.[0-9]{1,3})?$”
只能输入非零的正整数:“^+?[1-9][0-9]*$”
只能输入非零的负整数:“^-[1-9][0-9]*$”
只能输入长度为3的字符:“^.{3}$”
只能输入由26个英文字母组成的字符串:“^[A-Za-z]+$”
只能输入由26个大写英文字母组成的字符串:“^[A-Z]+$”
只能输入由26个小写英文字母组成的字符串:“^[a-z]+$”
只能输入由数字和26个英文字母组成的字符串:“^[A-Za-z0-9]+$”
只能输入由数字、26个英文字母或者下划线组成的字符串:“^w+$”
验证用户密码:“^[a-zA-Z]w{5,17}$”正确格式为:以字母开头,长度在6-18之间,

只能包含字符、数字和下划线。
验证是否含有^%&’,;=?$”等字符:“[^%&’,;=?$x22]+”
只能输入汉字:“^[u4e00-u9fa5],{0,}$”
验证Email地址:“^w+[-+.]w+)*@w+([-.]w+)*.w+([-.]w+)*$”
验证InternetURL:“^http://([w-]+.)+[w-]+(/[w-./?%&=]*)?$”
验证电话号码:“^((d{3,4})|d{3,4}-)?d{7,8}$”

正确格式为:“XXXX-XXXXXXX”,“XXXX-XXXXXXXX”,“XXX-XXXXXXX”,

“XXX-XXXXXXXX”,“XXXXXXX”,“XXXXXXXX”。
验证身份证号(15位或18位数字):“^d{15}|d{}18$”
验证一年的12个月:“^(0?[1-9]|1[0-2])$”正确格式为:“01”-“09”和“1”“12”
验证一个月的31天:“^((0?[1-9])|((1|2)[0-9])|30|31)$”

正确格式为:“01”“09”和“1”“31”。

匹配中文字符的正则表达式: [u4e00-u9fa5]
匹配双字节字符(包括汉字在内):[^x00-xff]
匹配空行的正则表达式:n[s| ]*r
匹配HTML标记的正则表达式:/< (.*)>.*|< (.*) />/
匹配首尾空格的正则表达式:(^s*)|(s*$)
匹配Email地址的正则表达式:w+([-+.]w+)*@w+([-.]w+)*.w+([-.]w+)*
匹配网址URL的正则表达式:http://([w-]+.)+[w-]+(/[w- ./?%&=]*)?

对于Java中正则表达式的详细内容,请参考JDK文档

你可能感兴趣的:(Java正则表达式之应用篇)