正则
行定位符(^与$)
行定位符是用来描述字符串的边界。“ ” 表 示 行 结 尾 , 表 示 行 开 始 e g : h e " , 表 示 以 h e 开 头 的 字 符 串 " l l o ”表示行结尾,^表示行开始 eg: ^ he",表示以he开头的字符串 "llo ”表示行结尾,表示行开始eg:he",表示以he开头的字符串"llo",表示以llo结尾的字符串。
单词定界符
定位符用来描述字符串或单词的边界,^和$分别指字符串的开始与结束,\b描述单词的前或后边界,\B表示非单词边界
我们在查找的一个单词的时候,如an是否在一个字符串”gril and boy”中存在,很明显如果匹配的话,an肯定是可以匹配字符串“gril and boy”匹配到,怎样才能让其匹配单词,而不是单词的一部分呢?这时候,我们可以是哟个单词定界符\b。
\ban\b 去匹配”gril and body”的话,就会提示匹配不到。
当然还有一个大写的\B,它的意思,和\b正好相反,它匹配的字符串不能使一个完整的单词,而是其他单词或字符串中的一部分。如\Ban\B。
选择字符(|) ,表示或
选择字符表示或的意思。如Aa|aA,表示Aa或者是aA的意思。注意使用”[]”与”|”的区别,在于”[]”只能匹配单个字符,而”|”可以匹配任意长度的字符串。在使用”[]”的时候,往往配合连接字符”-“一起使用,如[a-d],代表a或b或c或d。
排除字符,排除操作
正则表达式提供了””来表示排除不符合的字符,一般放在[]中。如[^1-5],该字符不是1~5之间的数字。
限定符(?*+{n,m})
限定符主要是用来限定每个字符串出现的次数。
限定字符
限定符 | 含义 |
---|---|
? | 0/1 次 |
* | 0/多次 |
+ | 一次或多次 |
{n} | n次 |
{n,} | 至少n次 |
{n,m} | n到m次 |
点号操作符
匹配任意一个字符(不包含换行符)
表达式中的反斜杠()
表达式中的反斜杠有多重意义,如转义、指定预定义的字符集、定义断言、显示不打印的字符。
- 常用特殊字符有”.”,”?”、”\”等
- 指定预定义的字符集
字符 | 字符集 |
---|---|
\d | 任意一个十进制数字[0-9] |
\D | 任意一个非十进制数字 |
\s | 任意一个空白字符(空格、换行符、换页符、回车符、字表符) |
\S | 任意一个非空白字符 |
\w | 任意一个单词字符 |
\W | 任意个非单词字符 |
- 显示不可打印的字符
字符 | 含义 |
---|---|
\a | 报警 |
\b | 退格 |
\f | 换页 |
\n | 换行 |
\r | 回车 |
\t | 字表符 |
括号字符()
在正则表达式中小括号的作用主要有:
改变限定符如(|、* 、^)的作用范围
如(my|your)baby,如果没有”()”,|将匹配的是要么是my,要么是yourbaby,有了小括号,匹配的就是mybaby或yourbaby。
进行分组,便于反向引用
反向引用 (接下来,我们在匹配重复的数字里有用到)
反向引用,就是依靠子表达式的”记忆”功能,匹配连续出现的字串或是字符。如(dqs)(pps)\1\2,表示匹配字符串dqsppsdqspps。在下面php应用中,我将详细展开学习反向引用。
代码示例
我们来常识模拟QQ排除靓号的规则来写个代码
<?php
class QQNumbers {
//创建12个为一个批次列表
protected $listCount = 12;
public function gNumCheck($str)
{
//5个及以上重复的数字 eg AAAAA
$pattern='/([\d])\1{4,}/';
preg_match($pattern,$str,$arr);
if ($arr[0]??false){
return false;
}
//两组及以上四个相同数字 AAAABBBB
$pattern='/([\d])\1{3,}([\d])\2{3,}/';
preg_match($pattern,$str,$arr);
if ($arr[0]??false){
return false;
}
//三组及以上三个相同数字 AAACCCFFF
$pattern='/([\d])\1{2,}([\d])\2{2,}([\d])\3{2,}/';
preg_match($pattern,$str,$arr);
if ($arr[0]??false){
return false;
}
//三位连对 AABBCC
$pattern='/(?:22(?=33)|33(?=44)|44(?=55)|55(?=66)|66(?=77)|77(?=88)|88(?=99)){3}\d/';
preg_match($pattern,$str,$arr);
if ($arr[0]??false){
return false;
}
//两组以上AACC的对子 AACCAACC
$pattern='/(([\d]){1,}([\d]){1,})\1{1,}/';
preg_match($pattern,$str,$arr);
if ($arr[0]??false){
return false;
}
//五位以上顺子 ABCDE
$pattern='/(?:0(?=1)|1(?=2)|2(?=3)|3(?=4)|4(?=5)|5(?=6)|6(?=7)|7(?=8)|8(?=9)){4,}\d/';
preg_match($pattern,$str,$arr);
if ($arr[0]??false){
return false;
}
//五位以上逆顺 EDCBA
$pattern = '/(?:9(?=8)|8(?=7)|7(?=6)|6(?=5)|5(?=4)|4(?=3)|3(?=2)|2(?=1)|1(?=0)){4,}\d/';
preg_match($pattern,$str,$arr);
if ($arr[0]??false){
return false;
}
//两组以上三位顺子ABCABC
$pattern = '/((?:9(?=8)|8(?=7)|7(?=6)|6(?=5)|5(?=4)|4(?=3)|3(?=2)|2(?=1)|1(?=0)){2,}\d)\1/';
preg_match($pattern,$str,$arr);
if ($arr[0]??false){
return false;
}
//两组以上三位逆顺子CBACBA
$pattern = '/((?:0(?=1)|1(?=2)|2(?=3)|3(?=4)|4(?=5)|5(?=6)|6(?=7)|7(?=8)|8(?=9)){2,}\d)\1/';
preg_match($pattern,$str,$arr);
if ($arr[0]??false){
return false;
}
//对称性 ABC(D)CBA
$pattern = '/(\d)(\d)(\d)(\d)\4?\3\2\1/';
preg_match($pattern,$str,$arr);
if ($arr[0]??false){
return false;
}
//爱情号
$pattern = '/^5201314/';
preg_match($pattern,$str,$arr);
if ($arr[0]??false){
return false;
}
return true;
}
//递归批量创建qq号
public function creatBatchNumbers(array $gNumbers)
{
while (count($gNumbers) < $this->listCount) {
$gNumber = mt_rand(10000000, 99999999);
if ($this->gNumCheck($gNumber)) {
$gNumbers[] = $gNumber;
}
}
//TODO 去除已占用QQ号码
//TODO 去除预占用列表
return count($gNumbers) == $this->listCount ? $gNumbers : $this->creatGNumbers($gNumbers);
}
//获取可选的QQ号码
public function getBatchQQNums()
{
//todo 限定获取次数
//生成可用QQ号码列表
$numArr = [];
$numArr = $this->creatBatchNumbers($numArr);
//TODO 失效该用户预占用qq号码
//TODO记录可用QQ号码列表 作为该用户的预占用
return ['qq_nums' => $numArr];
}
}
$qqNums = (new QQNumbers())->getBatchQQNums();
var_dump($qqNums);
// array(1) {
// ["qq_nums"]=>
// array(12) {
// [0]=>
// int(71843795)
// [1]=>
// int(69223632)
// [2]=>
// int(84505498)
// [3]=>
// int(16698823)
// [4]=>
// int(37498613)
// [5]=>
// int(82087251)
// [6]=>
// int(80695814)
// [7]=>
// int(22823504)
// [8]=>
// int(79333762)
// [9]=>
// int(99749277)
// [10]=>
// int(18317311)
// [11]=>
// int(57864377)
// }
// }
?>
以上代码在github 排除qq靓号
总结
- 以上代码用的比较多的就是反向引用,反向引用给正则添加了无线可能但是也更复杂了,在上面的例子中可能也存在优化的余地,希望大家能指正