正規表示法 文章列表

正規表示法

程式語言 進階篇

程式語言 進階篇

正規語言表示法

Regular Expression (正規表達式 or REGEXP) 在本質上是一種小型的特用語言。

這種 語言主要是用來做字串的比對、刪除、替換或拷貝重製等工作。當資料量龐大且形式規律時,Regular Expressions特別顯得有效益,因為,幾十幾百 甚至上千上萬的替換工作,要用人工來作業或巨集(marcro)指令,都幾乎是一個不可能 的任務。

其實 ~ 正規表示式、在各種程式語言裡,大同小異,僅供參考不負責任 ...

哪些符號是必須要被脫逸掉的(Escape)特殊符號呢?

許多控制字元族、白字元族是看不到的,所以必須透過 ”\” 這個跳脫字元才能表達。

其中 REGEXP所選用的特用字元(metacharacter)族 . ^ $ * + ? { [ ] \ | ( ) 這些則各有其
作用,如果要回歸到普通字元(跳脫特用字元功能)就必須透過 ”\” 這個跳脫字元才能
回歸普通字元。瞭解和操作這些不可見的字元,正是進階搜尋必要的一種新的「透視力」。

白字 ── whitespace
這是在操作資料時,一個隱藏在資料中的重要字族(Character Class),因為
空白、換行、跳格(tab)這類的控制字元都是看不見的,所以稱為「白字
(whitespapce) 也就是下列字元:

\t \n \r \f \v
因為這些字元都沒有可見圖像字形,所以必須要用 ”\” 加上一個字母來標示這
個白字元。白字元的集合就是在 REGULAR EXPRESSION中的白字族可以以
[ \t\n\r\f\v] 來表示 (注意[….]中第一個字元為空白字元),分別說明如下:


\t Tab ascii 碼 08 , 水平跳格
\n new line feed ascii 碼 12 ($0C) , 跳下一行
\r carrige return ascii碼13 ($0D), 歸位,回到一行的開始
\f form feed ascii 碼 14 ($0E), 換到下一頁
\s ascii碼 32, 空白字元

 

通用式

說明及範例

比對不成立之字串

/a/

含字母 "a" 的字串,例如 "ab", "bac", "cba"

"xyz"

/a./

含字母 "a" 以及其後任一個字元的字串,例如 "ab", "bac"(若要比對.,請使用 \.)

"a", "ba"

/^xy/

以 "xy" 開始的字串,例如 "xyz", "xyab"(若要比對 ^,請使用 \^)

"axy", "bxy"

/xy$/

以 "xy" 結尾的字串,例如 "axy", "abxy"以 "xy" 結尾的字串,例如 "axy", "abxy" (若要比對 $,請使用 \$)

"xya", "xyb"

[13579]

包含 "1" 或 "3" 或 "5" 或 "7" 或 "9" 的字串,例如:"a3b", "1xy"

"y2k"

[0-9]

含數字之字串

不含數字之字串

[a-z0-9]

含數字或小寫字母之字串

不含數字及小寫字母之字串

[a-zA-Z0-9]

含數字或字母之字串

不含數字及字母之字串

b[aeiou]t

"bat", "bet", "bit", "bot", "but"

"bxt", "bzt"

[^0-9]

不含數字之字串(若要比對 ^,請使用 \^)

含數字之字串

[^aeiouAEIOU]

不含母音之字串(若要比對 ^,請使用 \^)

含母音之字串

[^\^]

不含 "^" 之字串,例如 "xyz", "abc"

"xy^", "a^bc"

 

字元

說明

簡單範例

\

 跳脫字元

/A\*/ 可用於比對 "A*",其中 * 是一個特殊字元,為避開其特殊意義,所以必須加上 "\"

^

 比對輸入列的啟始位置

/^A/ 可比對 "Abcd" 中的 "A",但不可比對 "aAb"

$

 比對輸入列的結束位置

/A$/ 可比對 "bcdA" 中的 "A",但不可比對 "aAb"

*

 比對前一個字元零次或更多次

/bo*/ 可比對 "Good boook" 中的 "booo",亦可比對 "Good bk" 中的 "b"

+

 比對前一個字元一次或更多次,等效於 {1,}

/a+/ 可比對 "caaandy" 中的 "aaa",但不可比對 "cndy"

?

 比對前一個字元零次或一次

/e?l/ 可比對 "angel" 中的 "el",也可以比對 "angle" 中的 "l"

.

 比對任何一個字元(但換行符號不算)

/.n/ 可比對 "nay, an apple is on the tree" 中的 "an" 和 "on",但不可比對 "nay"

(x)

 比對 x 並將符合的部分存入一個變數

/(a*) and (b*)/ 可比對 "aaa and bb" 中的 "aaa" 和 "bb",並將這兩個比對得到的字串設定至變數 RegExp.id="mce_marker" 和 RegExp.$2。

xy

 比對 x 或 y

/a*b*/g 可比對 "aaa and bb" 中的 "aaa" 和 "bb"

{n}

 比對前一個字元 n 次,n 為一個正整數

/a{3}/ 可比對 "lllaaalaa" 其中的 "aaa",但不可比對 "aa"

{n,}

 比對前一個字元至少 n 次,n 為一個正整數

/a{3,}/ 可比對 "aa aaa aaaa" 其中的 "aaa" 及 "aaaa",但不可比對 "aa"

{n,m}

 比對前一個字元至少 n 次,至多 m 次,m、n 均為正整數

/a{3,4}/ 可比對 "aa aaa aaaa aaaaa" 其中的 "aaa" 及 "aaaa",但不可比對 "aa" 及 "aaaaa"

[xyz]

 比對中括弧內的任一個字元

/[ecm]/ 可比對 "welcome" 中的 "e" 或 "c" 或 "m"

[^xyz]

 比對不在中括弧內出現的任一個字元

/[^ecm]/ 可比對 "welcome" 中的 "w"、"l"、"o",可見出其與 [xyz] 功能相反。(同時請同學也注意 /^/ 與 [^] 之間功能的不同。)

[\b]

 比對退位字元(Backspace character)

可以比對一個 backspace ,也請注意 [\b] 與 \b 之間的差別

\b

 比對英文字的邊界,例如空格

例如 /\bn\w/ 可以比對 "noonday" 中的 ‘no’ ;
/\wy\b/ 可比對 "possibly yesterday." 中的 ‘ly’

\B

 比對非「英文字的邊界」

例如, /\w\Bn/ 可以比對 "noonday" 中的 ‘on’ , 
另外 /y\B\w/ 可以比對 "possibly yesterday." 中的 ‘ye’

\cX

 比對控制字元(Control character),其中 X 是一個控制字元

/\cM/ 可以比對一個字串中的 control-M

\d

 比對任一個數字,等效於 [0-9]

/[\d]/ 可比對由 "0" 至 "9" 的任一數字但其餘如字母等就不可比對

\D

 比對任一個非數字,等效於 [^0-9]

/[\D]/ 可比對 "w" "a"… 但不可比對如 "7" "1" 等數字

\f

 比對 form-feed

若是在文字中有發生 "換頁" 的行為則可以比對成功

\n

 比對換行符號

若是在文字中有發生 "換行" 的行為則可以比對成功

\r

 比對 carriage return

 

\s

 比對任一個空白字元(White space character),等效於 [ \f\n\r\t\v]

/\s\w*/ 可比對 "A b" 中的 "b"

\S

 比對任一個非空白字元,等效於 [^ \f\n\r\t\v]

/\S/\w* 可比對 "A b" 中的 "A"

\t

 比對定位字元(Tab)

 

\v

 比對垂直定位字元(Vertical tab)

 

\w

 比對數字字母字元(Alphanumerical characters)或底線字母("_"),等效於 [A-Za-z0-9_]

/\w/ 可比對 ".A _!9" 中的 "A"、"_"、"9"。

\W

 比對非「數字字母字元或底線字母」,等效於 [^A-Za-z0-9_]

/\W/ 可比對 ".A _!9" 中的 "."、" "、"!",可見其功能與 /\w/ 恰好相反。

\ooctal

 比對八進位,其中octal是八進位數目

/\oocetal123/ 可比對與八進位的ASCII中 "123" 所相對應的字元值。

\xhex

 比對十六進位,其中hex是十六進位數目

/\xhex38/ 可比對與 16進位的ASCII中 "38" 所相對應的字元。



RegularExpresson(正規語言的語法)基礎教學
 
^ 匹配字元串的開始
$ 匹配字元串的結束
ex
"^The": 開頭一定要有"The"字元串; 
"of despair$": 結尾一定要有"of despair" 的字元串; 
那麼
"^abc$": 就是要求以abc開頭和以abc結尾的字元串,實際上是只有abc匹配 
"notice": 匹配「包含」notice的字元串 
你可以看見如果你沒有用我們提到的兩個字符(最後一個例子),可以出現下被檢驗字元串的任何地方,你沒有把他鎖定到兩邊 

* 相當於{0,} zero or more
+ 相當於{1,} one or more
? 相當於{0,1} zero or one 
要點, '*', '+',和 '?'只管它前面那個字符.
ex
"ab*": 和ab{0,}同義,匹配以a開頭,後面可以接0個或者N個b組成的字元串("a", "ab", "abbb", 等); 
"ab+": 和ab{1,}同義,同上條一樣,但最少要有一個b存在 ("ab", "abbb", 等.); 
"ab?":和ab{0,1}同義,可以沒有或者只有一個b; 
"a?b+$": 匹配以一個或者0個a再加上一個以上的b結尾的字元串. 


{ } 限制字符出現的個數
ex
"ab{2}": 要求a後面一定要跟兩個b(一個也不能少)("abb"); 
"ab{2,}": 要求a後面一定要有兩個或者兩個以上b(如"abb", "abbbb", 等.); 
"ab{3,5}": 要求a後面可以有3-5個b("abbb", "abbbb", or "abbbbb"). 
( ) 幾個字符放到小括號裡
ex
"a(bc)*": 匹配 a 後面跟0個或者一個"bc"; 
"a(bc){1,5}": 一個到5個 "bc." 

還有一個字符 '│', 相當於OR 操作: 

"hi│hello": 匹配含有"hi" 或者 "hello" 的 字元串; 
"(b│cd)ef": 匹配含有 "bef" 或者 "cdef"的字元串; 
"(a│b)*c": 匹配含有這樣多個(包括0個)a或b,後面跟一個c 
的字元串; 

一個點('.')可以代表所有的單一字符,不包括"\n"
如果,要匹配包括"\n"在內的所有單個字符,怎么辦?
對了,用'[\n.]'這種模式.

"a.[0-9]": 一個a加一個字符再加一個0到9的數字 
"^.{3}$": 三個任意字符結尾 . 


中括號括住的內容只匹配一個單一的字符 

"[ab]": 匹配單個的 a 或者 b ( 和 "a│b" 一樣); 
"[a-d]": 匹配'a' 到'd'的單個字符 (和"a│b│c│d" 還有 "[abcd]"效果一樣); 一般我們都用[a-zA-Z]來指定字符為一個大小寫英文
"^[a-zA-Z]": 匹配以大小寫字母開頭的字元串 
"[0-9]%": 匹配含有 形如 x% 的字元串 
",[a-zA-Z0-9]$": 匹配以逗號再加一個數字或字母結尾的字元串 

你也可以把你不想要得字符列在中括號裡,你只需要在總括號裡面使用'^' 作為開頭 "%[^a-zA-Z]%" 匹配含有兩個百分號裡面有一個非字母的字元串. 
要點:^用在中括號開頭的時候,就表示排除括號裡的字符 
為了PHP能夠解釋,你必須在這些字符面前後加'',並且將一些字符轉義. 
不要忘記在中括號裡面的字符是這條規路的例外─在中括號裡面, 所有的特殊字符,包括(''), 都將失去他們的特殊性質 "[*\+?{}.]"匹配含有這些字符的字元串.
還有,正如regx的手冊告訴我們: "如果清單裡含有 ']', 最好把它作為清單裡的第一個字符(可能跟在'^'後面). 如果含有'-', 最好把它放在最前面或者最後面, or 或者一個範圍的第二個結束點[a-d-0-9]中間的‘-’將有效. 
看了上面的例子,你對{n,m}應該理解了吧.要注意的是,n和m都不能為負整數,而且n總是小於m. 這樣,才能 最少匹配n次且最多匹配m次. 如"p{1,5}"將匹配 "pvpppppp"中的前五個p

好,我們來做個應用:
如何構建一個模式來匹配 貨幣數量 的輸入 
構建一個匹配模式去檢查輸入的訊息是否為一個表示money的數字。
我們認為一個表示money的數量有四種模式︰ "10000.00" 和 "10,000.00",或者沒有小數部分, "10000" and "10,000". 
現下讓我們開始構建這個匹配模式: 
^[1-9][0-9]*$ 
這是所變量必須以非0的數字開頭.但這也意味著 單一的 "0" 也不能透過測試. 以下是解決的方法: 
^(0│[1-9][0-9]*)$ 
"只有0和不以0開頭的數字與之匹配",我們也可以允許一個負號在數字之前: 
^(0│-?[1-9][0-9]*)$ 
這就是: "0 或者 一個以0開頭 且可能 有一個負號在前面的數字." 好了,現下讓我們別那麼嚴謹,
允許以0開頭.現下讓我們放棄 負號 , 因為我們在表示錢幣的時候並不需要用到. 我們現下指定 模式 用來匹配小數部分: 
^[0-9]+(\.[0-9]+)?$ 
這暗示匹配的字元串必須最少以一個阿拉伯數字開頭. 但是注意,在上面模式中 "10." 是不匹配的, 只有 "10" 和 "10.2" 才可以. (你知道為什麼嗎) 
^[0-9]+(\.[0-9]{2})?$ 
我們上面指定小數點後面必須有兩位小數.如果你認為這樣太苛刻,你可以改成: 
^[0-9]+(\.[0-9]{1,2})?$ 
這將允許小數點後面有一到兩個字符. 現下我們加上用來增加可讀性的逗號(每隔三位), 我們可以這樣表示: 
^[0-9]{1,3}(,[0-9]{3})*(\.[0-9]{1,2})?$ 
不要忘記 '+' 可以被 '*' 替代 如果你想允許空白字元串被輸入話 (為什麼?). 也不要忘記反斜杆 ’\’ 在php字元串中可能會出現錯誤 (很普遍的錯誤). 
現下,我們已經可以確認字元串了, 我們現下把所有逗號都去掉 str_replace(",", "", $money) 然後在把類型看成 double然後我們就可以透過他做數學計算了. 

再來一個:
構造檢查email的正規語言表示法
在一個完整的email位址中有三個部分: 
1. 用戶名 (在 [email]'@'[/email] 左邊的一切), 
[email=2.]2.'@'[/email],
3. 伺服器名(就是剩下那部分). 
用戶名可以含有大小寫字母阿拉伯數字,句號 ('.'), 減號('-'), and 下劃線 ('_'). 伺服器名字也是符合這個規則,當然下劃線除外. 
現下, 用戶名的開始和結束都不能是句點. 伺服器也是這樣. 還有你不能有兩個連續的句點他們之間至少存在一個字符,好現下我們來看一下怎么為用戶名寫一個匹配模式: 
^[_a-zA-Z0-9-]+$ 
現下還不能允許句號的存在. 我們把它加上: 
^[_a-zA-Z0-9-]+(\.[_a-zA-Z0-9-]+)*$ 
上面的意思就是說: "以至少一個規範字符(除了.)開頭,後面跟著0個或者多個以點開始的字元串." 
簡單化一點, 我們可以用 eregi()取代 ereg().eregi()對大小寫不敏感, 我們就不需要指定兩個範圍 "a-z" 和 "A-Z" 只需要指定一個就可以了: 
^[_a-z0-9-]+(\.[_a-z0-9-]+)*$ 
後面的伺服器名字也是一樣,但要去掉下劃線: 
^[a-z0-9-]+(\.[a-z0-9-]+)*$ 
好. 現下只需要用”@”把兩部分連接: 
^[_a-z0-9-]+(\.[_a-z0-9-]+)*@[a-z0-9-]+(\.[a-z0-9-]+)*$ 

這就是完整的email認証匹配模式了,只需要調用 
eregi(‘^[_a-z0-9-]+(\.[_a-z0-9-]+)*@[a-z0-9-]+(\.[a-z0-9-]+)*$ ’,$eamil) 
就可以得到是否為email了 
正規語言表示法的其他用法 

 

JavaScript 程式語言 :

var reg = new RegExp('ABC','i');

傳回比對搜尋的結果 = "123456A000B000被比對的字符串".match(reg);

PHP 程式語言:

邏輯值= preg_match(正規表示式 ,要比對的字串 [,結果] );

$str = "xa111b222a333a20112233";

$pattern = "/a[3]{3}/i";

preg_match($pattern, $str, $X);

echo $X[0]; //a333

---------------------------------------

$text = "xa111b222a3330000022222223333";

$pattern = "/.*?(a[3]{3}).*/i";

echo preg_replace($pattern,"\\1", $text); //a333
 
VB.NET 程式語言:

Dim  regExp As Regex = New Regex("[A-Z] {1} [0-9] {9}")


'是否符合身份証字號

If  regExp.IsMatch(code.Text) Then

Lcode.Text ="身份証字號 : " & code.Text

Else

Lcode.Text ="不符合身份証字號格式 !"

End If


C# 程式語言:

using System;
using System.Text.RegularExpressions;
namespace MyApplication
{
class Program
{
static void Main(string[] args)
{
string allScore;
allScore = "88,62,(100),86,(99),77,(93)";
Console.WriteLine(Regex.Replace(allScore, @"\(\d+\),?",""));
 //  88,62,86,77,
}
}


Regex 規則運算式選項

 Regex 引擎依預設,輸入字串與規則運算式模式中任何常值字元的比較會區分大小寫,規則運算式模式中的空白字元會解譯成常值空白字元,而規則運算式中的擷取群組會隱含也會明確命名。

您可以藉由指定規則運算式選項來修改這些預設規則運算式行為,選項可以是複數 /規則運算/,"選項"

var reg = new RegExp('ABC','ig');

  選項 flag   意義   說明
 i ignore case  使用不區分大小寫的比對方式
 s singleline  使用單行模式,使句點 . 會比對每個字元,而不是換行符號 \n 以外的每個字元
 m multiline  使用多行模式,使 ^ 和 $ 會比對每一行的開頭與結尾,而不是輸入字串的開頭和結尾
 g global search  使用全局匹配模式。有時候你會希望重複匹配目標字串多次,使用 g flag 就可以保留 lastIndex 的狀態,讓下一次再匹配時,可以從 lastIndex 的位置開始找起,你就可以遍歷整個文本中的所有可匹配字串,而不是每一次再匹配時都得到一樣的結果 (文本中從頭開始第一個找到的字串)。另外預設上,如果沒有開啟 g flag,Regex 引擎只會返回「第一個」匹配到的字串結果


        搜尋比對時的邊界對齊

搜尋獨立的完整字串
\b ─── 要在字詞(word)的邊界上,”\bthe\b” 意思就是搜尋單獨的”the”。
例:”… the theater is other …”  (標紅色的獨立字 the 會被搜到,不是獨立字的綠色the則不會被搜出。)

搜尋靠邊的字串
而 “\bthe” 就是只在左邊有限制:
例:”…the theater is other …,then dog,there... " 共四個 the 會被搜到在,other裡的the就不會。
注意:她們所臨接的 . ,,(句號、空白、英文逗號,中文逗號都是合法的邊)。

搜尋不限制位置的字串
而搜尋 “the” 沒有加上限制出現的位置時就對任何字串都有效:
例:”… the theater is other …” 三個 the 字串都會被搜到。

搜尋不再邊邊上的字串
\B ─── 與 \b 剛好相反互補,就是不可在字詞(word)的邊界上,\Bt\B 意思就是尋找不在兩端的t。

例:”… the theater is other …” 只有不在兩邊的紅t會被搜到,靠邊的綠色t則不會被搜出。

字串樣本群組
小括弧( )的使用時機。
一般的Regular Expression(正則表達式)系統會提供數個的字串樣本變數來容納由 “( … )”內的字串。
在字串樣本中,也可以放上各種特用字元(metacharacter)就像數學式(sub-expression)一般,
而成為一種字串樣本的一部份,而可以在稍後以 \1  \3 的方式加以引用,
 “\1” 就是叫用先前建立的第一個 (…) 內的字串「樣本1」; ”\3” 則表示引用(先前建立的)「樣本3」。

※在 Javascript 與某些程式語言,樣本變數是使用錢字號表示$,$1~$9。

補充說明

REGEXP 預設是(貪婪的,greedy)

例: A.*B 會搜出: xxxA xxxB xxxxB zzzzB   當有好幾種可能時,REGEXP基本原則上是「盡可
能搜出」,在這裡就是到最右邊的B。( xxxB xxxxB zzzz這一段就是滿足 * 的內容 )
A.*?B 會搜出:xxxA xxxB xxxxB zzzzB  ? 就是指定採取保守模式(不貪婪,non-greedy),也就是
只搜到A之後第一個遇到的B就停下來。
注意: “?”若是用在 (*, +, ?, {n}, 之後時,則用來指定保守的「非重複擴張」模式 (不貪婪,non
greedy),也就是若是可多可少的重複次數時,「以搜尋比對最少次數為準」。

雖然不是很滿意,但還可以接受

 檢查網址URL = /(https?:\/\/[\w\-_]+(\.[\w\-_]+)+([\w\-\.,@?^=%&:\/~\+#]*[\w\-\@?^=%&\/~\+#-])?)|/ig

延伸閱讀 :  在這個程式裡 Blogger 留言小工具 我使用了正規表示式,自動加上超連結、顯示圖像。

2021/02/5 最近寫了2支程式,都需要用到 RegexpBlogger小工具 最新文章 不會使用還真的很難玩 ~

Regex Tester on line   

 



           

預 覽 視 窗

新舊排序 :

    © Copyright 2021 Design By 灰鴿   技術堤供 : Blogger