正則表達式的基本知識
2020-03-16 21:22:27
供稿:網友
正則表達式的基本知識:
元字符:
正則表達式的威力在于其能夠在模式中包含選擇和循環。它們通過使用
有兩組不同的元字符:一種是模式中除了方括號內都能被識別的,還有一種是在方括號內被識別的。方括號之外的元字符有這些:
有數種用途的通用轉義符
斷言目標的開頭(或在多行模式下行的開頭,即緊隨一換行符之后)
斷言目標的結尾(或在多行模式下行的結尾,即緊隨一換行符之前)
匹配除了換行符外的任意一個字符(默認情況下)
字符類定義開始
字符類定義結束
開始一個多選一的分支
子模式開始
子模式結束
擴展 ( 的含義,也是 0 或 1 數量限定符,以及數量限定符最小值
匹配 0 個或多個的數量限定符
匹配 1 個或多個的數量限定符
最少/最多數量限定開始
最少/最多數量限定結束
模式中方括號內的部分稱為“字符類”。字符類中可用的元字符為:
通用轉義字符
排除字符類,但僅當其為第一個字符時有效
指出字符范圍
結束字符類
更加詳細的每個元字符的使用方法請參照PHP手冊的:模式語法 一節。
看一些例子來講算法:(引自:http://php.mydict.com/ziliao/4/15/2006_06/PHPZhongDeZhengZeBiaoDaShi3539_1.html)
特殊字符"^"用來匹配以指定字符串開頭的字符串。例如:
"^hello" :這個模式與字符串"hello,PHP world!"匹配,但是與"Say hello to you"不匹配。
特殊字符"$"用來匹配以指定字符串結尾的字符串。例如:
"you$" :這個模式與"How are you"匹配,與"your"不匹配。
當特殊字符"^"和"$"同時使用時,表示精確匹配。例如:
"^hello$" :這個模式只匹配字符串"hello"。
如果一個模式不包括"^"和"$",那么它與任何包含該模式的字符串匹配。例如: "you" :與字符串"What is your name?"是匹配的。
在該模式中的字母只是普通的字符,數字也是一樣的。
如果要用到其他一些稍微復雜的字符,如標點符號和空白字符(比如空格、制表符等),就要要用到轉義序列。所有的轉義序列都用反斜杠("/")打頭,例如制表符的轉義序列是:"/t"。所以如果我們要檢測一個字符串是否以制表符開頭,可以用這個模式:
"^/t"
類似的,用"/n"表示換行,"/r"表示回車,反斜杠本身用"//"表示,句號"."用"/."表示,依此類推。
如何使用字符簇?
如果要判斷用戶輸入的電話號碼、地址、EMAIL地址、信用卡號碼等是否有效,用普通的基于字面的字符串比較是不夠的。所以要用一種更好的方法來描述我們想要的模式,這就是字符簇。
比如,要建立一個表示所有元音字符的字符簇,就可以這樣做:
"[AaEeIiOoUu]" :這個模式與任何元音字符匹配,但只能表示一個字符。
用特殊符號"-"可以表示一個字符的范圍,如:
"[a-z]" //匹配字母a-z,即所有的小寫字母
"[A-Z]" //匹配字母A-Z,即所有的大寫字母
"[a-zA-Z]" //匹配所有的字母
"[0-9]" //匹配所有的數字
"[0-9/./-]" //匹配所有的數字,以及句號和減號
"[ /f/r/t/n]" //匹配所有的白字符
同樣的,這些也只匹配一個字符。
如果要匹配由一個小寫字母和一位數字組成的字符串,比如"a4"、"b5"或"f1",但不是"aa4"、"b5a4" 或"f12"的話,用這個模式:
"^[a-z][0-9]$"
盡管[a-z]代表26個字母的范圍,但在這里它只能與第一個字符是小寫字母的字符串匹配。
我們已經知道"^"表示字符串的開頭,但是當在一組方括號里使用"^"時,它表示"非"或"排除"的意思,常常用來剔除某個字符。還用前面的例子,我們要求第一個字符不能是數字: "^[^0-9][0-9]$"
這個模式與"a4"、"b5"及"+2"是匹配的,但與"12"、"66"是不匹配的。下面是幾個排除特定字符的例子:
"[^a-z]" //除了小寫字母以外的所有字符
"[^/////^]" //除了(/)(/)(^)之外的所有字符
"[^/"/']" //除了雙引號(")和單引號(')之外的所有字符
特殊字符"." (點,英文句號)在正規表達式中用來匹配除了"換行"之外的所有字符。所以模式"^.5$"與任何兩個字符的、以數字5結尾和以其他非"換行"字符開頭的字符串匹配。模式"."可以匹配任何字符串,除了空串和只包括一個"換行"的字符串。
PHP的正規表達式有一些內置的通用字符簇,列表如下:
字符簇 含義
"[[:alpha:]]" 任何字母
"[[:digit:]]" 任何數字
"[[:alnum:]]" 任何字母和數字
"[[:space:]]" 任何白字符
"[[:upper:]]" 任何大寫字母
"[[:lower:]]" 任何小寫字母
"[[:punct:]]" 任何標點符號
"[[:xdigit:]]" 任何16進制的數字,相當于[0-9a-fA-F]
如何匹配重復出現的情況?
在很多的情況下,我們可能要匹配一個單詞或一組數字。一個單詞有若干個字母組成,一組數字有若干個單一的數字組成。我們用跟在字符或字符簇后面的花括?quot;{}"來確定前面的內容的重復出現的次數:假設x是一個數字,那么{x}表示"前面的字符或字符簇只出現x次";一個數字加逗號,{x,}的意思是"前面的內容出現x或更多的次數";兩個用逗號分隔的數字,{x,y}表示"前面的內容至少出現x次,但不超過y次"。
字符簇 含義
"^[a-zA-Z_]$" 所有的字母和下劃線
"^[[:alpha:]]{3}$" 所有的3個字母的單詞
"^a$" 字母a
"^a{4}$" 不是以字母a開頭并且有4個字母的單詞,比如Aaaa
^a{2,4}$" aa,aaa或aaaa
"^a{1,3}$" a,aa或aaa
"^a{2,}$" 包含多于兩個a的字符串,比如aaa,aaaa,aaaaa
"^a{2,}" 以兩個a開頭的單詞,如:aardvark和aaab,但apple不行
"a{2,}" 包含有兩個a的單詞,如:baad和aaa,但Nantucket不行
"/t{2}" 兩個制表符
".{2}" 所有的兩個字符
我們可以把模式擴展到更多的單詞或數字:
"^[a-zA-Z0-9_]{1,}$" 所有包含一個以上的字母、數字或下劃線的字符串
"^[0-9]{1,}$" 所有的正數
"^/-{0,1}[0-9]{1,}$" 所有的整數
"^/-{0,1}[0-9]{0,}/.{0,1}[0-9]{0,}$" 所有的整數
最后一個例子我們可以這樣考慮:所有以一個可選的負號(/-{0,1})開頭(^)、跟著0個或更多的數字([0-9]{0,})、和一個可選的小數點(/.{0,1})再跟上0個或多個數字([0-9]{0,}),并且沒有其他任何東西($)。
特殊字符"?"與"{0,1}"是相等的,它們都代表著:"0個或1個前面的內容"或"前面的內容是可選的"。所以:
"^/-{0,1}[0-9]{0,}/.{0,1}[0-9]{0,}$"
可以簡化為:
^/-?[0-9]{0,}/.?[0-9]{0,}$
特殊字符"*"與"{0,}"是相等的,它們都代表著"0個或多個前面的內容"。字符"+"與{1,}是相等的,表示"1個或多個前面的內容",所以上面的4個例子可以寫成:
"^[a-zA-Z0-9_]+$" 所有包含一個以上的字母、數字或下劃線的字符串
"^[0-9]+$" 所有的正數
"^/-?[0-9]+$" 所有的整數
"^/-?[0-9]*/.?[0-9]*$" 所有的小數