PHP之正則表達(dá)式捕獲組與非捕獲組(詳解)
在項(xiàng)目開發(fā)過程中正則表示經(jīng)常會(huì)用到,可以說會(huì)正則表達(dá)式是每個(gè)程序員最基本的要求,初學(xué)者在剛接觸正則表達(dá)式都感到很吃力。最近看到一位朋友的博客寫的《PHP正則表達(dá)式》獲益頗多,在章節(jié)對(duì)通配符以及捕獲數(shù)據(jù)非常感興趣。這兩章節(jié)剛好也涉及到了正則表達(dá)式的捕獲組和非捕獲組的內(nèi)容,以此來分析這方面的內(nèi)容
我們知道,在正則表達(dá)式下(x) 表示匹配'x'并記錄匹配的值。這只是比較通俗的說法,甚至說這是不嚴(yán)謹(jǐn)?shù)恼f法,只有()捕獲組形式才會(huì)記錄匹配的值。非捕獲組則只匹配,不記錄。
捕獲組:
(pattern)
這種形式是我們見到最多的一種形式,匹配并返回捕獲結(jié)果,可以嵌套,組號(hào)順序從左到右依次排列‘。
$regex = '/(ab(c)+)+d(e)?/';
$str = 'abccde';
$matches = array();
if(preg_match($regex, $str, $matches)){
print_r($matches);
}
匹配結(jié)果:
Array ( [0] => abccde [1] => abcc [2] => c [3] => e )
(?P<name>pattern)
這種方式雖然看起來在構(gòu)造正則表達(dá)式的時(shí)候略微復(fù)雜一點(diǎn),但實(shí)質(zhì)上與(pattern)一樣。最大的優(yōu)勢(shì)體現(xiàn)在對(duì)結(jié)果處理上,程序員可以直接根據(jù)自己設(shè)置的<name>直接快速調(diào)用結(jié)果,而不用再去數(shù)需要的結(jié)果在第幾個(gè)子組了。
$regex = '/(?P<group1>\w(?P<group2>\w))abc(?P<group3>\w)45/';
$str = 'fsabcd45';
$matches = array();
if(preg_match($regex, $str, $matches)){
print_r($matches);
}
匹配結(jié)果:
Array ( [0] => fsabcd45 [group1] => fs [1] => fs [group2] => s [2] => s [group3] => d [3] => d )
\num
num是一個(gè)整數(shù),是對(duì)捕獲組的反向引用。 例如\2表示第二個(gè)子組匹配值,\表示第一個(gè)子組匹配值
$regex = '/(\w)(\w)\2\1/';
$str = 'abba';
$matches = array();
if(preg_match($regex, $str, $matches)){
print_r($matches);
}
匹配結(jié)果:
Array ( [0] => abba [1] => a [2] => b )
注意,這里我疏忽了一個(gè)小細(xì)節(jié),一開始我第一樣代碼是 $regex = “/(\w)(\w)\2\1/”; 結(jié)果返回?zé)o匹配結(jié)果,經(jīng)過調(diào)試后,發(fā)現(xiàn)這里只能用' '。'與" 用法差別大家還是需要注意下。
\k< name >
了解了(?P<name>pattern)與\num,這個(gè)就不難理解了。\k< name >是對(duì)命名捕獲組的反向引用。其中 name 是捕獲組名。
$regex='/(?P<name>\w)abc\k<name>/';
$str="fabcf";
echo preg_match_all($regex, $str,$matches);
print_r($matches);
匹配結(jié)果:
Array ( [0] => Array ( [0] => fabcf ) [name] => Array ( [0] => f ) [1] => Array ( [0] => f ) )
非捕獲組:
(?:pattern)
與(pattern)的唯一區(qū)別是,匹配pattern但不捕獲匹配結(jié)果。這里便不再舉例。
還有四種方式實(shí)際上講的是一個(gè)事情:預(yù)查。
預(yù)查分為正向預(yù)查與反向預(yù)查。根據(jù)字面理解,正向預(yù)查是判斷匹配字符串后面某些字符存在與否,而反向預(yù)查則是判斷匹配字符串前面某些字符存在與否。
正向預(yù)查判斷存在使用(?=pattern),判斷不存在使用(?!pattern)。
反向預(yù)查判斷存在使用(?<=pattern),判斷不存在使用(?<!pattern)。
$regx='/(?<=a)bc(?=d)/';
$str="abcd ebcd abce ebca";
if(preg_match_all($regx, $str, $matches)){
print_r($matches);
}
匹配結(jié)果:
Array ( [0] => Array ( [0] => bc) )
這四種形式使用的是否只要注意好相對(duì)匹配字符串的位置和斷言肯定還是否定,就會(huì)很快掌握。
另外,預(yù)查的四種形式是零寬度的,匹配的時(shí)候只做一個(gè)判斷,本身是不占位置的。/HE(?=L)LLO/ 與HELLO匹配,而/HE(?=L)LO/與HELLO是不匹配的。畢竟但從字節(jié)數(shù)上兩者就是不匹配的,前者只有4個(gè),而后者有5個(gè)。
以上就是PHP之正則表達(dá)式捕獲組與非捕獲組詳解的全部內(nèi)容,希望對(duì)大家有所啟迪。
相關(guān)文章
PHP經(jīng)典面試題之設(shè)計(jì)模式(經(jīng)常遇到)
php中設(shè)計(jì)模式非常多,但是設(shè)計(jì)模式在php面試題經(jīng)常會(huì)提到,本文主要給大家介紹php經(jīng)典面試題之設(shè)計(jì)模式,需要的朋友一起看看吧2015-10-10php無法連接mysql數(shù)據(jù)庫的正確解決方法
這篇文章主要為大家詳細(xì)介紹了php無法連接mysql數(shù)據(jù)庫的正確解決方法,感興趣的小伙伴們可以參考一下2016-07-07php設(shè)計(jì)模式之狀態(tài)模式實(shí)例分析【星際爭霸游戲案例】
這篇文章主要介紹了php設(shè)計(jì)模式之狀態(tài)模式,結(jié)合星際爭霸游戲案例形式分析了php狀態(tài)模式相關(guān)原理、使用技巧與注意事項(xiàng),需要的朋友可以參考下2020-03-03php使用workman框架實(shí)現(xiàn)socket服務(wù)以及連接客戶端
這篇文章主要介紹了php使用workman框架實(shí)現(xiàn)socket服務(wù)以及連接客戶端,本文列舉了詳細(xì)的過程和代碼展示,能夠幫助你學(xué)習(xí),需要的朋友可以參考下2021-06-06