首页> 博客> 有必要了解的正则表达式
14 03 2020
1、简介

正则表达式(regular expression)描述了一种字符串匹配的模式(pattern),可以用来检查一个串是否含有某种子串、将匹配的子串替换或者从某个串中取出符合某个条件的子串等。学习正则表达式很大程度上就是学习正则表达式的语法规则。

文本的复杂处理 一种强大而灵活的文本处理工具: 大部分编程语言 、 数据库、文本编辑器、开发环境都支持正则表达式

2、基本语法
2.1、普通字符

字母、数字、汉字、下划线、以及没有特殊定义的标点符号,都是“普通字符”。表达式中的普通字符,在匹配一个字符串的时候,匹配与之相同的一个字符。 简单的转义字符

\n代表换行符
\t制表符
\代表\本身
^ ,$,.,(, ) , {, } , ? , + , * ,匹配这些字符本身标准字符集合
2.2、标准字符集合

能够与 ‘多种字符’ 匹配的表达式 注意区分大小写,大写是相反的意思

\d任意一个数字,0~9 中的任意一个
\D除了数字以外的任一字符
\w任意一个字母或数字或下划线,也就是 A~Z,a~z,0~9,_ 中任意一个
\W除了字母、数字和下划线外的任一一个字符
\s包括空格、制表符、换行符等空白字符的其中任意一个
\S除了空格,制表符和换行符等空白字符外的任一个字符
.小数点可以匹配任意一个字符(除了换行符);如果要匹配包括“\n”在内的所有字符,一般用[\s\S]
2.3、自定义字符集合

[ ]方括号匹配方式,能够匹配方括号中任意一个字符

[ab5@]匹配 "a" 或 "b" 或 "5" 或 "@"
[^ abc]匹配 "a","b","c" 之外的任意一个字符
[f-k]匹配 "f"~"k" 之间的任意一个字母
[^A-F0-3]匹配 "A"~"F","0"~"3" 之外的任意一个字符
  • 正则表达式的特殊符号,被包含到中括号中,则失去特殊意义,除了^,-之外。
  • 标准字符集合,除小数点外,如果被包含于中括号,自定义字符集合将包含该集合。比如: [\d.-+]将匹配:数字、小数点、+、-
2.4、量词(Quantifier)

修饰匹配次数的特殊符号:

{n}表达式重复n次
{m,n}表达式至少重复m次,最多重复n次
{m,}表达式至少重复m次
?匹配表达式0次或者1次,相当于 {0,1}
+表达式至少出现1次,相当于 {1,}
*表达式不出现或出现任意次,相当于 {0,}
  • 匹配次数中的贪婪模式(匹配字符越多越好,默认!)
  • 匹配次数中的非贪婪模式(匹配字符越少越好,修饰匹配次数的特殊符号后再加上一个 "?" 号)
    • 比如,这里匹配的是:至少两个数字,如果匹配到则不匹配 3、4等数字个数组合 \d{2,6}?
  • 字符边界: (本组标记匹配的不是字符而是位置,符合某种条件的位置)
^与字符串开始的地方匹配
$与字符串结束的地方匹配
\b匹配一个单词边界

\b 匹配这样一个位置:前面的字符和后面的字符不全是\w

正则表达式的匹配模式

  • IGNORECASE 忽略大小写模式
    • 匹配时忽略大小写。
    • 默认情况下,正则表达式是要区分大小写的。
  • SINGLELINE 单行模式
    • 整个文本看作一个字符串,只有一个开头,一个结尾。
    • 使小数点 "." 可以匹配包含换行符(\n)在内的任意字符。
  • MULTILINE 多行模式
    • 每行都是一个字符串,都有开头和结尾。
    • 在指定了 MULTILINE 之后,如果需要仅匹配字符串开始和结束位置,可以使用 \A 和 \Z
2.5、选择符和分组
表达式作用
竖线 分支结构左右两边表达式之间 "或" 关系,匹配左边或者右边
( )捕获组(1). 在被修饰匹配次数的时候,括号中的表达式可以作为整体被修饰 (2). 取匹配结果的时候,括号中的表达式匹配到的内容可以被单独得到 (3). 每一对括号会分配一个编号,使用()的捕获根据左括号的顺序从 1开始自动编号。捕获元素编号为零的第一个捕获是由整个正则表达式模式匹配的文本
(?:Expression)非捕获组一些表达式中,不得不使用( ),但又不需要保存( )中子表达式匹
配的内容,这时可以用非捕获组来抵消使用( )带来的副作用。
2.6、反向引用(\nnn)
  1. 每一对()会分配一个编号,使用 () 的捕获根据左括号的顺序从 1 开始自动编号。
  2. 通过反向引用,可以对分组已捕获的字符串进行引用。
2.7、预搜索(零宽断言)
  1. 只进行子表达式的匹配,匹配内容不计入最终的匹配结果,是零宽度;
  2. 这个位置应该符合某个条件。判断当前位置的前后字符,是否符合指定的条件,但不匹配前后的字符。是对位置的匹配;
  3. 正则表达式匹配过程中,如果子表达式匹配到的是字符内容,而非位置,并被保存到最终的匹配结果中,那么就认为这个子表达式是占有字符的;如果子表达式匹配的仅仅是位置,或者匹配的内容并不保存到最终的匹配结果中,那么就认为这个子表达式是零宽度的。占有字符还是零宽度,是针对匹配的内容是否保存到最终的匹配结果中而言的。
(?=exp)断言自身出现的位置的后面能匹配表达式exp
(?<=exp)断言自身出现的位置的前面能匹配表达式exp
(?!exp)断言此位置的后面不能匹配表达式exp
(?<!exp)断言此位置的前面不能匹配表达式exp
3、练习
3.1、电话号码验证
  1. 电话号码由数字和"-"构成
  2. 电话号码为7到8位
  3. 如果电话号码中包含有区号,那么区号为三位或四位, 首位是0.
  4. 区号用"-"和其他部分隔开
  5. 移动电话号码为11位
  6. 11位移动电话号码的第一位和第二位为"13“,”15”,”18”

(0\d{2,3}-\d{7,8})|(1[3456789]\d{9})

3.2、电子邮件地址验证
  1. 用户名:字母、数字、中划线、下划线组成。
  2. @
  3. 网址:字母、数字组成。
  4. 小数点:.
  5. 组织域名:2-4位字母组成。
  6. 不区分大小写

[\w-]+@[0-9a-zA-Z]+(.[a-zA-Z]{2,4}){1,2}

4、常用的正则式列表
匹配中文字符[\u4e00-\u9fa5]
匹配空白行\n\s*\r
匹配HTML标记<(\S*?)[^>]*>.*?</\1>
匹配首尾空白字符^\s*
匹配Email地址\w+([-+.]\w+)*@\w+([-.]\w+)*\.\w+([-.]\w+)*
匹配网址URL[a-zA-z]+://[^\s]*
匹配国内电话号码\d{3}-\d{8}
匹配腾讯QQ号[1-9][0-9]{4,}
匹配中国邮政编码[1-9]\d{5}(?!\d)
匹配身份证\d{15}
匹配ip地址\d+\.\d+\.\d+\.\d+
5、Java中使用正则表达式

相关类位于:java.util.regex包下面

  • 类 Pattern:
    • 正则表达式的编译表示形式。
    • Pattern p = Pattern.compile(r,int); //建立正则表达式,并启用相应模式
  • 类 Matcher:
    • 通过解释 Pattern 对 character sequence 执行匹配操作的引擎
    • Matcher m = p.matcher(str); //匹配str字符串
5.1、匹配字符
package com.codesofun.regex;

import java.util.regex.Matcher;
import java.util.regex.Pattern;

/**
 * @ClassName Demo1
 * @Description TODO
 * @Author 小莫
 * @Date 2019/12/27 10:38
 * @Version 1.0
 **/
public class Demo1 {
    public static void main(String[] args) {
        String str = "dsadasdasd&&21312321";
        //表达式对象
        Pattern p = Pattern.compile("\\w+");
        //创建matcher对象
        Matcher matcher = p.matcher(str);
        //find() 该方法扫描输入的序列,查找与该模式匹配的下一个子序列
        System.out.println("===============");
        while (matcher.find()){
            System.out.println(matcher.group());
        }
        System.out.println("===============");

        boolean yesorno = matcher.matches();
        System.out.println(yesorno);

    }
}
package com.codesofun.regex;

import java.util.regex.Matcher;
import java.util.regex.Pattern;

/**
 * @ClassName Demo1
 * @Description TODO:正则表达式匹配操作
 * @Author 小莫
 * @Date 2019/12/27 10:38
 * @Version 1.0
 **/
public class Demo2 {
    public static void main(String[] args) {
        String str = "aa32132**ssd312*sds122";
        //表达式对象
        Pattern p = Pattern.compile("([a-z]+)([0-9]+)");
        //创建matcher对象
        Matcher matcher = p.matcher(str);

        while (matcher.find()){
            //group() /group(0) 匹配整个表达式的子字符串
            System.out.println(matcher.group() + "===>" + matcher.group(1) + "+" + matcher.group(2));
        }
    }
}
5.2、替换字符
package com.codesofun.regex;

import java.util.regex.Matcher;
import java.util.regex.Pattern;

/**
 * @ClassName Demo1
 * @Description TODO:正则表达式 替换操作
 * @Author 小莫
 * @Date 2019/12/27 10:38
 * @Version 1.0
 **/
public class Demo3 {
    public static void main(String[] args) {
        String str = "aa32132**ssd312*sds122";
        //表达式对象
        Pattern p = Pattern.compile("[0-9]");
        //创建matcher对象
        Matcher matcher = p.matcher(str);

        //替换
        String newStr = matcher.replaceAll("#");
        System.out.println(newStr);
    }
}
5.3、分割字符
package com.codesofun.regex;

import java.util.Arrays;

/**
 * @ClassName Demo1
 * @Description TODO:正则表达式 分割操作
 * @Author 小莫
 * @Date 2019/12/27 10:38
 * @Version 1.0
 **/
public class Demo4 {
    public static void main(String[] args) {
        String str = "aa32132ssd312sds122huhu";
        String[] strs = str.split("[\\d]+");
        System.out.println(Arrays.toString(strs));
    }
}

类似文章

  1. Git常用操作总结
  2. JAVA 实现 FTP 文件上传、下载和删除

评论区

| 0 评论

还没有评论,快来抢沙发吧!