1. 程式人生 > >解析posix與perl標準的正則表示式區別

解析posix與perl標準的正則表示式區別

正則表示式(Regular Expression,縮寫為regexp,regex或regxp),又稱正規表示式、正規表示式或常規表示式或正規化表示法或正規表示法,是指一個用 來描述或者匹配一系列符合某個句法規則的字串的單個字串。在很多文字編輯器或其他工具裡,正則表示式通常被用來檢索和/或替換那些符合某個模式的文字 內容。許多程式設計語言都支援利用正則表示式進行字串操作。例如,在Perl中就內建了一個功能強大的在正則表示式引擎。正則表示式這個概念最初是由 Unix中的工具軟體(例如sed和grep)普及開的。(摘自維基百科)

PHP同時使用兩套正則表示式規則,一套是由電氣和電子工程師 協會(IEEE)制定的POSIX Extended 1003.2相容正則(事實上PHP對此標準的支援並不完善),另一套來自PCRE(Perl Compatible Regular Expression)庫提供PERL相容正則,這是個開放原始碼的軟體,作者為 Philip Hazel。

使用POSIX相容規則的函式有:
ereg_replace()
ereg()
eregi()
eregi_replace()
split()
spliti()
sql_regcase()
mb_ereg_match()
mb_ereg_replace()
mb_ereg_search_getpos()
mb_ereg_search_getregs()
mb_ereg_search_init()
mb_ereg_search_pos()
mb_ereg_search_regs()
mb_ereg_search_setpos()
mb_ereg_search()
mb_ereg()
mb_eregi_replace()
mb_eregi()
mb_regex_encoding()
mb_regex_set_options()
mb_split()

使用PERL相容規則的函式有:
preg_grep()
preg_replace_callback()
preg_match_all()
preg_match()
preg_quote()
preg_split()
preg_replace()

定界符:

POSIX相容正則沒有定界符,函式的相應引數會被認為是正則。

PERL相容正則可以使用任何不是字母、數字或反斜線()的字元作為定界符,如果作為定界符的字元必須被用在表示式本身中,則需要用反斜線轉義。也可以使用(),{},[] 和 <> 作為定界符

修正符:

POSIX相容正則沒有修正符。

PERL相容正則中可能使用的修正符(修正符中的空格和換行被忽略,其它字元會導致錯誤):

i (PCRE_CASELESS):
匹配時忽略大小寫。

m(PCRE_MULTILINE):
當設定了此修正符,行起始(^)和行結束($)除了匹配整個字串開頭和結束外,還分別匹配其中的換行符(\n)的之後和之前。

s(PCRE_DOTALL):
如果設定了此修正符,模式中的圓點元字元(.)匹配所有的字元,包括換行符。沒有此設定的話,則不包括換行符。

x(PCRE_EXTENDED):
如果設定了此修正符,模式中的空白字元除了被轉義的或在字元類中的以外完全被忽略。

e:
如果設定了此修正符,preg_replace() 在替換字串中對逆向引用作正常的替換,將其作為 PHP 程式碼求值,並用其結果來替換所搜尋的字串。 只有 preg_replace() 使用此修正符,其它 PCRE 函式將忽略之。

A(PCRE_ANCHORED):
如果設定了此修正符,模式被強制為“anchored”,即強制僅從目標字串的開頭開始匹配。

D(PCRE_DOLLAR_ENDONLY):
如果設定了此修正符,模式中的行結束($)僅匹配目標字串的結尾。沒有此選項時,如果最後一個字元是換行符的話,也會被匹配在裡面。如果設定了 m 修正符則忽略此選項。

S:
當一個模式將被使用若干次時,為加速匹配起見值得先對其進行分析。如果設定了此修正符則會進行額外的分析。目前,分析一個模式僅對沒有單一固定起始字元的 non-anchored 模式有用。

U(PCRE_UNGREEDY):
使“?”的預設匹配成為貪婪狀態的。

X(PCRE_EXTRA):
模式中的任何反斜線後面跟上一個沒有特殊意義的字母導致一個錯誤,從而保留此組合以備將來擴充。預設情況下,一個反斜線後面跟一個沒有特殊意義的字母被當成該字母本身。

u(PCRE_UTF8):
模式字串被當成UTF-8。

邏輯區隔:
POSIX相容正則和PERL相容正則的邏輯區隔符號作用和使用方法完全一致:
[]:包含任選一操作的相關資訊。
{}:包含匹配次數的相關資訊。
():包含一個邏輯區間的相關資訊,可被用來進行引用操作。
|:表示“或”,[ab]和a|b是等價的。

元字元與“[]”相關:

有兩組不同的元字元:一種是模式中除了方括號內都能被識別的,還有一種是在方括號“[]”內被識別的。

POSIX相容正則和PERL相容正則“[]之外”“一致”的元字元:
 有數種用途的通用轉義符
^ 匹配字串的開頭
$ 匹配字串的結尾
? 匹配0或者1

  • 匹配 0 個或多個前面指定型別的字元
  • 匹配 1 個或多個前面指定型別的字元

POSIX相容正則和PERL相容正則“[]之外”“不一致”的元字元:
. PERL相容正則匹配除了換行符外的任意一個字元
. POSIX相容正則匹配任意一個字元

POSIX相容正則和PERL相容正則“[]之內”“一致”的元字元:
 有數種用途的通用轉義符
^ 取反字元,但僅當其為第一個字元時有效

  • 指定字元ASCII範圍,仔細研究ASCII碼,你會發現[W-c]等價於[WXYZ\^_`abc]

POSIX相容正則和PERL相容正則“[]之內”“不一致”的元字元:

  • POSIX相容正則中[a-c-e]的指定會丟擲錯誤。
  • PERL相容正則中[a-c-e]的指定等價於[a-e]。

匹配次數與“{}”相關:
POSIX相容正則和PERL相容正則在匹配次數方面完全一致:
{2}:表示匹配前面的字元2次
{2,}:表示匹配前面的字元2次或多次,預設都是貪婪(儘可能多)的匹配
{2,4}:表示匹配前面的字元2次或4次

邏輯區間與“()”相關:
使用()包含起來的區域是一個邏輯區間,邏輯區間的主要作用是體現出一些字元出現的邏輯次序,另一個用處就是可以用來引用(可以將此區間內的值引用給一個變數)。後一個作用比較奇特:
<?php
$str = "http://www.163.com/";
// POSIX相容正則:
echo ereg_replace("(.+)","\1",$str);
// PERL相容正則:
echo preg_replace("/(.+)/","$1",$str);
// 顯示兩個連結
?>

在引用的時候,括號是可以巢狀的,邏輯次序是按照“(”出現的次序來標定的。

型別匹配:
POSIX相容正則:
[:upper:]:匹配所有的大寫字母
[:lower:]:匹配所有的小寫字母
[:alpha:]:匹配所有的字母
[:alnum:]:匹配所有的字母和數字
[:digit:]:匹配所有的數字
[:xdigit:]:匹配所有的十六進位制字元,等價於[0-9A-Fa-f]
[:punct:]:匹配所有的標點符號,等價於 [.,"'?!;:]
[:blank:]:匹配空格和TAB,等價於[ \t]
[:space:]:匹配所有的空白字元,等價於[ \t\n\r\f\v]
[:cntrl:]:匹配所有ASCII 0到31之間的控制符。
[:graph:]:匹配所有的可列印字元,等價於:[^ \t\n\r\f\v]
[:print:]:匹配所有的可列印字元和空格,等價於:[^\t\n\r\f\v]
[.c.]:功能不明
[=c=]:功能不明
[:<:]:匹配單詞的開始
[:>:]:匹配單詞的結尾

PERL相容正則(這裡可以看出PERL正則的強大):
\a alarm,即 BEL 字元('0)
\cx "control-x",其中 x 是任意字元
\e escape('0B)
\f 換頁符 formfeed('0C)
\n 換行符 newline('0A)
\r 回車符 carriage return('0D)
\t 製表符 tab('0)
\xhh 十六進位制程式碼為 hh 的字元
\ddd 八進位制程式碼為 ddd 的字元,或 backreference
\d 任一十進位制數字
\D 任一非十進位制數的字元
\s 任一空白字元
\S 任一非空白字元
\w 任一“字”的字元
\W 任一“非字”的字元
\b 字分界線
\B 非字分界線
\A 目標的開頭(獨立於多行模式)
\Z 目標的結尾或位於結尾的換行符前(獨立於多行模式)
\z 目標的結尾(獨立於多行模式)
\G 目標中的第一個匹配位置