Java里正则表达式(包括一些高级用法)
有些内容只有在 Java 里面才能使用。 ------原文作者:逗号 (他不高兴发博客)
## 字符种类
部分使用频率几乎为 0 的这里不会提及(比如 \x 之流)
### 普通
| | 含义 | 助记 |
| ---- | ------------------------------------- | ----- |
| \s | 非空白 | space |
| \s | 空白 | |
| \d | 数字 | digit |
| \D | 非数字 | |
| [] | 自定义字符种类可使用 ^ \| && 随意组合 | |
### 特别
| | 含义 | 助记 |
| ------- | ------------------------------------------------------------ | ---------- |
| \h | 横向字符 | horizontal |
| \h | 非横向字符 | |
| \v | 纵向字符(比如说换行符) | vertical |
| \V | 非纵向字符 | |
| \w | 单词字符(包括 - _) | word |
| \W | 非单词字符 | |
| \p{XXX} | POSIX,这个并不常用,在有些时候会方便,Character 有很多 is开头的方法,将方法名中的`is`替换为`Java`可直接使用进行字符匹配:\p{javaLowerCase} | posix |
| \Q | 字符转义开始符 | |
| \E | 字符转义结束符,这两个符号之间的内容全部按照字面量处理,里面是所有字符都不会有什么含义,比如 . 就是 .,\s 必须匹配\s,而不是匹配空白。 | |
## 量词
| | 含义 | 助记 |
| ----- | ------------- | ---- |
| ? | 0个或者1个 | |
| * | 0个或多个 | |
| + | 1个或多个 | |
| {m} | m个 | |
| {m,n} | 介于m~n个之间 | |
`?`修饰在量词之后,表尽可能少的匹配!
## 边界
### 普通
| | 含义 | 助记 |
| ---- | ------------ | ---- |
| ^ | 字符开始位置 | |
| $ | 字符结束位置 | |
### 特别
| | 含义 | 助记 |
| ---- | ------------------------------ | ---- |
| \b | 单词边界,单词的最前面和最后面 | |
| \B | 非字符串边界 | |
下面的内容普通人基本上就已经不管了!
## 分组捕获
正则表达式中,每出现一对括号就是新建了个新的组。比如:`(\\w+)|(\\d+)`
正则表达式的组可以用使用索引进行引用,形式为:`$index`
`$0` 默认指向原来的全部字符
然后索引递增规律如下:
`((a)|(bc))`
`$1` 指向 `((a)|(bc))` 匹配的内容
`$2` 指向 `(a)` 匹配的内容
`$3` 指向 `(bc)` 匹配的内容
然而当正则复杂之后,改动将会对索引有较大影响,也不方便计算索引,此时,你可以为组制定名称:
比如匹配字符串:`a=123,b=456`中间的键值,使用索引,需要写成:`(\\w+)=(\\S+)`,然后使用`$1`指向 key,使用 `$2` 指向 value。
我们可以将正则改为:`(?<key>\\w+)=(?<value>\\S+)`,上述表达式使用 `?<name>` 的形式在正则中为组定义名称。
## 引用分组
在 Java 中,你可以在两种地方使用引用:
* 正则表达式中
匹配 `xxx-xxx` 字符 `-` 两边的内容相同,使用索引引用前面的组(`\\index`):
```regexp
(\\w+)-\\1
```
使用名字引用前面的组:
```regexp
(?<content>\\w+)-\\k<content>
```
* 在 Java 的字符串替换中
Markdown 中的字符串转为斜线的html:
```java
"*name*".replaceAll("\\*(?<content>\\w+)\\*", "<em>$1</em>")
"*name*".replaceAll("\\*(?<content>\\w+)\\*", "<em>${content}</em>")
```
## 模式
你可以在构造 `Pattern`时设定模式:
```Java
Pattern.compile("abc", Pattern.CASE_INSENSITIVE)
```
模式可以自行到 `Pattern`下查看,这里只说使用方式,在组开头使用 `?flags:`的方式声明模式,比如:
匹配单词,不区分大小写
```regex
(?i:\\w+)
```
记住`i`(忽略大小写)、`s`(. 匹配全部字符)即可,其他基本不用。
## 前瞻后顾
前瞻后顾,是正则表达式中的高级技巧,主要有5种:
> 注意:虽然这货也有括号,但是这货只算个条件,并不会捕获任何内容,也不会影响索引
### 前瞻
前瞻中不带`<`,并且只影响前面的表达式,意思就是偷偷往前看一下(这看一下没有限制的,可以看到结尾)。
* `exp1(?=exp2)`
寻找后面是 exp2 的 exp1
* `exp1(?!exp2)`
### 后顾
* `(?<=exp1)exp2`
匹配 exp1 后面的 exp2,前面没有exp1的内容将被忽略,比如,替换`abcba`替换`c`后面的`b`为 `xxx`,需要写为:
```java
"abcba".replaceAll("(?<=c)b", "xxx")
```
`a`后面的`b`则不受影响。
> 注意:正则表达式`(?<=c)b`不会匹配字符串`cb`,意思就是:
>
> ```java
> assertFalse("cb".matches("(?<=c)b"));
> assertFalse("b".matches("(?<=c)b"));
> assertTrue("cb".matches("c(?<=c)b"));
> ```
* `(?<!exp1)exp2`
后顾中带有 `<` ,并且只影响后面的表达式,也就是说,你如果写成这样:`exp2(?<=exp1)`,是没什么卵用的。
## 牛刀小试
以下所有内容只能使用`String#repalceAll`或者`String#matches`完成。
### 手机号遮罩
将`11`位手机号除前4位之外,其他的使用`*`替换。
### 格式化数字
例如:`1234567890` 使用正则替换之后,变为 `1,234,567,890`。
### Trim 字符串
使用正则去除字符串开始和结束的空白字符。
### 去除数字结尾的`0`
`1234.4500`应该处理为`1234.45`,`1234.00`应该处理为`1234`。
### 处理文档
将字符串的`* 测试 * ** 测试 **`的使用单个`*`包围的内容两边分别加上`<h1>`、`</h1>`,不能影响两个`**`包围的内容。
### 测试某个字符串是不是合法的Java标识符
### 测试`IPv4`
已知`IPv4`中每一段的数字需要介于0~255,测试某个字符串是否是合法的`IPv4`表达式。
### 去除`//`注释
已知`Java`中可以使用`//` 进行单行注释,请去除一段代码中的这种注释。
### 去除`/* 内容 */`注释
已知`Java`中可以使用`/* */` 进行多行注释,请去除代码中的这种注释。
posted on 2020-07-23 09:51 tomcat and jerry 阅读(1511) 评论(0) 编辑 收藏 举报