准確來說,不論在C++或C中,只要在Linux系統中都可以使用本文講訴的正則表達式使用方式。
一、Linux中正則表達式的使用步驟:
- 編譯正則表達式 regcomp()
- 匹配正則表達式 regexec()
- 釋放正則表達式 regfree()
- 錯誤處理 regerror()
二、使用舉例:

1 /***************************************************************************** 2 * Copyright (C) 2017-2018 Hanson Yu All rights reserved. 3 ------------------------------------------------------------------------------ 4 * File Module : Regex.c 5 * Description : 6 7 [yweifeng@localhost test]$ gcc -o Regex Regex.c 8 [yweifeng@localhost test]$ ./Regex 9 Match 10 a=fmtp:96 packetization-mode=1;profile-level-id=640020;sprop-parameter-sets=Z2QAIKzZQEgG2wEQAAADABAAAAMB6PGDGWA=,aOvjyyLA 11 1 12 640020 13 Z2QAIKzZQEgG2wEQAAADABAAAAMB6PGDGWA= 14 aOvjyyLA 15 [yweifeng@localhost test]$ 16 17 18 * Created : 2017.11.01. 19 * Author : Yu Weifeng 20 * Function List : 21 * Last Modified : 22 * History : 23 ******************************************************************************/ 24 #include <stdio.h> 25 #include <stdlib.h> 26 27 #include <regex.h> 28 29 30 31 /***************************************************************************** 32 -Fuction : main 33 -Description : main 34 .點 匹配除“\r\n”之外的任何單個字符 35 * 匹配前面的子表達式任意次。例如,zo*能匹配“z”,也能匹配“zo”以及“zoo”。*等價於o{0,} 36 (pattern) 匹配模式串pattern並獲取這一匹配。所獲取的匹配可以從產生的Matches集合得到 37 [xyz] 字符集合。匹配所包含的任意一個字符。例如,“[abc]”可以匹配“plain”中的“a”。 38 + 匹配前面的子表達式一次或多次(大於等於1次)。例如,“zo+”能匹配“zo”以及“zoo”,但不能匹配“z”。+等價於{1,}。 39 //如下例子中不用+,默認是一次,即只能匹配到一個數字6 40 41 [A-Za-z0-9] 26個大寫字母、26個小寫字母和0至9數字 42 43 [A-Za-z0-9+/=] 26個大寫字母、26個小寫字母0至9數字以及+/= 三個字符 44 45 46 -Input : 47 -Output : 48 -Return : 49 * Modify Date Version Author Modification 50 * ----------------------------------------------- 51 * 2017/11/01 V1.0.0 Yu Weifeng Created 52 ******************************************************************************/ 53 int main(int argc,char **argv) 54 { 55 int i,j=0; 56 char acErrBuf[256]; 57 int iRet=-1; 58 regex_t tReg; //定義一個正則實例 59 const char *strPattern = "a=.*packetization-mode=([0-9]+);profile-level-id=([0-9]+);sprop-parameter-sets=([A-Za-z0-9+/=]+),([A-Za-z0-9]+)";//定義模式串 60 //const char *strPattern = "a=.*packetization-mode=([0-9]+);profile-level-id=([0-9]+);sprop-parameter-sets=([A-Za-z0-9]+)";//定義模式串 61 char *pStrBuf = "a=fmtp:96 packetization-mode=1;profile-level-id=640020;sprop-parameter-sets=Z2QAIKzZQEgG2wEQAAADABAAAAMB6PGDGWA=,aOvjyyLA;"; //定義待匹配串 62 const size_t dwMatch = 5; //定義匹配結果最大允許數 //表示允許幾個匹配 63 regmatch_t atMatch[5]; //定義匹配結果在待匹配串中的下標范圍 //存儲匹配串位置的數組 64 //數組0單元存放主正則表達式位置,后邊的單元依次存放子正則表達式位置 65 66 67 68 //REG_ICASE 匹配字母時忽略大小寫。 69 iRet =regcomp(&tReg, strPattern, REG_EXTENDED); //編譯正則模式串 70 if(iRet != 0) 71 { 72 regerror(iRet, &tReg, acErrBuf, sizeof(acErrBuf)); 73 printf("Regex Error: %s\n", acErrBuf); 74 } 75 else 76 { 77 iRet = regexec(&tReg, pStrBuf, dwMatch, atMatch, 0); //匹配他 78 if (iRet == REG_NOMATCH) 79 { //如果沒匹配上 80 printf("No Match\n"); 81 } 82 else if (iRet == REG_NOERROR) 83 { //如果匹配上了 84 printf("Match\n"); 85 for(j=0;j<dwMatch;j++) 86 { 87 for (i= atMatch[j].rm_so; i < atMatch[j].rm_eo; i++) 88 { //遍歷輸出匹配范圍的字符串 89 printf("%c", pStrBuf[i]); 90 } 91 printf("\n"); 92 } 93 } 94 else 95 { 96 printf("Unknow err:%d",iRet); 97 } 98 99 regfree(&tReg); //釋放正則表達式 100 } 101 102 return 0; 103 }

1 /***************************************************************************** 2 * Copyright (C) 2017-2018 Hanson Yu All rights reserved. 3 ------------------------------------------------------------------------------ 4 * File Module : Regex.c 5 * Description : 6 7 [yweifeng@localhost test]$ gcc -o Regex Regex.c 8 [yweifeng@localhost test]$ ./Regex 9 Match 10 a=fmtp:96 packetization-mode=1;profile-level-id=640020;sprop-parameter-sets=Z2QAIKzZQEgG2wEQAAADABAAAAMB6PGDGWA=,aOvjyyLA 11 1 12 640020 13 Z2QAIKzZQEgG2wEQAAADABAAAAMB6PGDGWA= 14 aOvjyyLA 15 [yweifeng@localhost test]$ 16 17 18 * Created : 2017.11.01. 19 * Author : Yu Weifeng 20 * Function List : 21 * Last Modified : 22 * History : 23 ******************************************************************************/ 24 #include <stdio.h> 25 #include <stdlib.h> 26 27 #include <regex.h> 28 29 30 31 /***************************************************************************** 32 -Fuction : main 33 -Description : main 34 .點 匹配除“\r\n”之外的任何單個字符 35 * 匹配前面的子表達式任意次。例如,zo*能匹配“z”,也能匹配“zo”以及“zoo”。*等價於o{0,} 36 (pattern) 匹配模式串pattern並獲取這一匹配。所獲取的匹配可以從產生的Matches集合得到 37 [xyz] 字符集合。匹配所包含的任意一個字符。例如,“[abc]”可以匹配“plain”中的“a”。 38 + 匹配前面的子表達式一次或多次(大於等於1次)。例如,“zo+”能匹配“zo”以及“zoo”,但不能匹配“z”。+等價於{1,}。 39 //如下例子中不用+,默認是一次,即只能匹配到一個數字6 40 41 [A-Za-z0-9] 26個大寫字母、26個小寫字母和0至9數字 42 43 [A-Za-z0-9+/=] 26個大寫字母、26個小寫字母0至9數字以及+/= 三個字符 44 45 46 -Input : 47 -Output : 48 -Return : 49 * Modify Date Version Author Modification 50 * ----------------------------------------------- 51 * 2017/11/01 V1.0.0 Yu Weifeng Created 52 ******************************************************************************/ 53 int main(int argc,char **argv) 54 { 55 int i,j=0; 56 char acErrBuf[256]; 57 int iRet=-1; 58 regex_t tReg; //定義一個正則實例 59 const char *strPattern = "a=.*packetization-mode=([0-9]+);profile-level-id=([0-9]+)";//定義模式串 60 //profile-level-id=640020;profile-level-id=640021; 相同的,只能匹配前一個 61 char *pStrBuf = "a=fmtp:96 packetization-mode=1;profile-level-id=640020;profile-level-id=640021"; //定義待匹配串 62 const size_t dwMatch = 5; //定義匹配結果最大允許數 //表示允許幾個匹配 63 regmatch_t atMatch[5]; //定義匹配結果在待匹配串中的下標范圍 //存儲匹配串位置的數組 64 //數組0單元存放主正則表達式位置,后邊的單元依次存放子正則表達式位置 65 66 67 68 //REG_ICASE 匹配字母時忽略大小寫。 69 iRet =regcomp(&tReg, strPattern, REG_EXTENDED); //編譯正則模式串 70 if(iRet != 0) 71 { 72 regerror(iRet, &tReg, acErrBuf, sizeof(acErrBuf)); 73 printf("Regex Error: %s\n", acErrBuf); 74 } 75 else 76 { 77 iRet = regexec(&tReg, pStrBuf, dwMatch, atMatch, 0); //匹配他 78 if (iRet == REG_NOMATCH) 79 { //如果沒匹配上 80 printf("No Match\n"); 81 } 82 else if (iRet == REG_NOERROR) 83 { //如果匹配上了 84 printf("Match\n"); 85 for(j=0;j<dwMatch;j++) 86 { 87 for (i= atMatch[j].rm_so; i < atMatch[j].rm_eo; i++) 88 { //遍歷輸出匹配范圍的字符串 89 printf("%c", pStrBuf[i]); 90 } 91 printf("\n"); 92 } 93 } 94 else 95 { 96 printf("Unknow err:%d",iRet); 97 } 98 99 regfree(&tReg); //釋放正則表達式 100 } 101 102 return 0; 103 }
三、使用說明:
- int regcomp (regex_t *compiled, const char *pattern, int cflags)
- regex_t 是一個結構體數據類型,用來存放編譯后的正則表達式
- regex_t 的成員re_nsub 用來存儲正則表達式中的子正則表達式的個數,子正則表達式就是用圓括號包起來的部分表達式
- pattern 是指向我們寫好的正則表達式的指針
- cflags 有如下4個值或者是它們或運算(|)后的值:
- REG_EXTENDED 以功能更加強大的擴展正則表達式的方式進行匹配。
- REG_ICASE 匹配字母時忽略大小寫。
- REG_NOSUB 不用存儲匹配后的結果。
- REG_NEWLINE 識別換行符,這樣'$'就可以從行尾開始匹配,'^'就可以從行的開頭開始匹配
- int regexec (regex_t *compiled, char *string, size_t nmatch, regmatch_t matchptr [], int eflags)
- 如果在編譯正則表達式的時候沒有指定cflags的參數為REG_NEWLINE,則默認情況下是忽略換行符的,也就是把整個文本串當作一個字符串處理。執行成功返回0
- regmatch_t 是一個結構體數據類型,在regex.h中定義:
typedef struct
{
regoff_t rm_so;
regoff_t rm_eo;
} regmatch_t; - 成員rm_so 存放匹配文本串在目標串中的開始位置,rm_eo存放結束位置
- 通常我們以數組的形式定義一組這樣的結構。因為往往我們的正則表達式中還包含子正則表達式。數組0單元存放主正則表達式位置,后邊的單元依次存放子正則表達式位置
- compiled 是已經用regcomp函數編譯好的正則表達式
- string 是目標文本串
- nmatch 是regmatch_t結構體數組的長度
- matchptr regmatch_t類型的結構體數組,存放匹配文本串的位置信息
- eflags 有兩個值
- REG_NOTBOL 按我的理解是如果指定了這個值,那么’^’就不會從我們的目標串開始匹配。總之我到現在還不是很明白這個參數的意義
- REG_NOTEOL 和上邊那個作用差不多,不過這個指定結束end of line
- void regfree (regex_t *compiled)
- 清空compiled指向的regex_t結構體的內容
- 如果是重新編譯的話,一定要先清空regex_t結構體c
- size_t regerror (int errcode, regex_t *compiled, char *buffer, size_t length)
- 當執行regcomp 或者regexec 產生錯誤的時候,就可以調用這個函數而返回一個包含錯誤信息的字符串
- errcode 是由regcomp 和 regexec 函數返回的錯誤代號
- compiled 是已經用regcomp函數編譯好的正則表達式,這個值可以為NULL
- buffer 指向用來存放錯誤信息的字符串的內存空間
- length 指明buffer的長度,如果這個錯誤信息的長度大於這個值,則regerror 函數會自動截斷超出的字符串,但他仍然會返回完整的字符串的長度。所以我們可以用如下的方法先得到錯誤字符串的長度。
四、正則表達式模式串的介紹
1.符號
(摘自《正則表達式之道》)
正則表達式由一些普通字符和一些元字符(metacharacters)組成。普通字符包括大小寫的字母和數字,而元字符則具有特殊的含義,我們下面會給予解釋。
在最簡單的情況下,一個正則表達式看上去就是一個普通的查找串。例如,正則表達式"testing"中沒有包含任何元字符,它可以匹配"testing"和"testing123"等字符串,但是不能匹配"Testing"。
要想真正的用好正則表達式,正確的理解元字符是最重要的事情。下表列出了所有的元字符和對它們的一個簡短的描述。
元字符 |
描述 |
\ |
將下一個字符標記符、或一個向后引用、或一個八進制轉義符。例如,“\\n”匹配\n。“\n”匹配換行符。序列“\\”匹配“\”而“\(”則匹配“(”。即相當於多種編程語言中都有的“轉義字符”的概念。 |
^ |
匹配輸入字符串的開始位置。如果設置了RegExp對象的Multiline屬性,^也匹配“\n”或“\r”之后的位置。 |
$ |
匹配輸入字符串的結束位置。如果設置了RegExp對象的Multiline屬性,$也匹配“\n”或“\r”之前的位置。 |
* |
匹配前面的子表達式任意次。例如,zo*能匹配“z”,也能匹配“zo”以及“zoo”。*等價於o{0,} |
+ |
匹配前面的子表達式一次或多次(大於等於1次)。例如,“zo+”能匹配“zo”以及“zoo”,但不能匹配“z”。+等價於{1,}。 |
? |
匹配前面的子表達式零次或一次。例如,“do(es)?”可以匹配“do”或“does”中的“do”。?等價於{0,1}。 |
{n} |
n是一個非負整數。匹配確定的n次。例如,“o{2}”不能匹配“Bob”中的“o”,但是能匹配“food”中的兩個o。 |
{n,} |
n是一個非負整數。至少匹配n次。例如,“o{2,}”不能匹配“Bob”中的“o”,但能匹配“foooood”中的所有o。“o{1,}”等價於“o+”。“o{0,}”則等價於“o*”。 |
{n,m} |
m和n均為非負整數,其中n<=m。最少匹配n次且最多匹配m次。例如,“o{1,3}”將匹配“fooooood”中的前三個o為一組,后三個o為一組。“o{0,1}”等價於“o?”。請注意在逗號和兩個數之間不能有空格。 |
? |
當該字符緊跟在任何一個其他限制符(*,+,?,{n},{n,},{n,m})后面時,匹配模式是非貪婪的。非貪婪模式盡可能少的匹配所搜索的字符串,而默認的貪婪模式則盡可能多的匹配所搜索的字符串。例如,對於字符串“oooo”,“o+”將盡可能多的匹配“o”,得到結果[“oooo”],而“o+?”將盡可能少的匹配“o”,得到結果 ['o', 'o', 'o', 'o'] |
.點 |
匹配除“\r\n”之外的任何單個字符。要匹配包括“\r\n”在內的任何字符,請使用像“[\s\S]”的模式。 |
(pattern) |
匹配pattern並獲取這一匹配。所獲取的匹配可以從產生的Matches集合得到,在VBScript中使用SubMatches集合,在JScript中則使用$0…$9屬性。要匹配圓括號字符,請使用“\(”或“\)”。 |
(?:pattern) |
非獲取匹配,匹配pattern但不獲取匹配結果,不進行存儲供以后使用。這在使用或字符“(|)”來組合一個模式的各個部分時很有用。例如“industr(?:y|ies)”就是一個比“industry|industries”更簡略的表達式。 |
(?=pattern) |
非獲取匹配,正向肯定預查,在任何匹配pattern的字符串開始處匹配查找字符串,該匹配不需要獲取供以后使用。例如,“Windows(?=95|98|NT|2000)”能匹配“Windows2000”中的“Windows”,但不能匹配“Windows3.1”中的“Windows”。預查不消耗字符,也就是說,在一個匹配發生后,在最后一次匹配之后立即開始下一次匹配的搜索,而不是從包含預查的字符之后開始。 |
(?!pattern) |
非獲取匹配,正向否定預查,在任何不匹配pattern的字符串開始處匹配查找字符串,該匹配不需要獲取供以后使用。例如“Windows(?!95|98|NT|2000)”能匹配“Windows3.1”中的“Windows”,但不能匹配“Windows2000”中的“Windows”。 |
(?<=pattern) |
非獲取匹配,反向肯定預查,與正向肯定預查類似,只是方向相反。例如,“(?<=95|98|NT|2000)Windows”能匹配“2000Windows”中的“Windows”,但不能匹配“3.1Windows”中的“Windows”。 |
(?<!pattern) |
非獲取匹配,反向否定預查,與正向否定預查類似,只是方向相反。例如“(?<!95|98|NT|2000)Windows”能匹配“3.1Windows”中的“Windows”,但不能匹配“2000Windows”中的“Windows”。這個地方不正確,有問題 此處用或任意一項都不能超過2位,如“(?<!95|98|NT|20)Windows正確,“(?<!95|980|NT|20)Windows 報錯,若是單獨使用則無限制,如(?<!2000)Windows 正確匹配 |
x|y |
匹配x或y。例如,“z|food”能匹配“z”或“food”(此處請謹慎)。“[zf]ood”則匹配“zood”或“food”。 |
[xyz] |
字符集合。匹配所包含的任意一個字符。例如,“[abc]”可以匹配“plain”中的“a”。 |
[^xyz] |
負值字符集合。匹配未包含的任意字符。例如,“[^abc]”可以匹配“plain”中的“plin”。 |
[a-z] |
字符范圍。匹配指定范圍內的任意字符。例如,“[a-z]”可以匹配“a”到“z”范圍內的任意小寫字母字符。 注意:只有連字符在字符組內部時,並且出現在兩個字符之間時,才能表示字符的范圍; 如果出字符組的開頭,則只能表示連字符本身. |
[^a-z] |
負值字符范圍。匹配任何不在指定范圍內的任意字符。例如,“[^a-z]”可以匹配任何不在“a”到“z”范圍內的任意字符。 |
\b |
匹配一個單詞邊界,也就是指單詞和空格間的位置(即正則表達式的“匹配”有兩種概念,一種是匹配字符,一種是匹配位置,這里的\b就是匹配位置的)。例如,“er\b”可以匹配“never”中的“er”,但不能匹配“verb”中的“er”。 |
\B |
匹配非單詞邊界。“er\B”能匹配“verb”中的“er”,但不能匹配“never”中的“er”。 |
\cx |
匹配由x指明的控制字符。例如,\cM匹配一個Control-M或回車符。x的值必須為A-Z或a-z之一。否則,將c視為一個原義的“c”字符。 |
\d |
匹配一個數字字符。等價於[0-9]。grep 要加上-P,perl正則支持 |
\D |
匹配一個非數字字符。等價於[^0-9]。grep要加上-P,perl正則支持 |
\f |
匹配一個換頁符。等價於\x0c和\cL。 |
\n |
匹配一個換行符。等價於\x0a和\cJ。 |
\r |
匹配一個回車符。等價於\x0d和\cM。 |
\s |
匹配任何不可見字符,包括空格、制表符、換頁符等等。等價於[ \f\n\r\t\v]。 |
\S |
匹配任何可見字符。等價於[^ \f\n\r\t\v]。 |
\t |
匹配一個制表符。等價於\x09和\cI。 |
\v |
匹配一個垂直制表符。等價於\x0b和\cK。 |
\w |
匹配包括下划線的任何單詞字符。類似但不等價於“[A-Za-z0-9_]”,這里的"單詞"字符使用Unicode字符集。 |
\W |
匹配任何非單詞字符。等價於“[^A-Za-z0-9_]”。 |
\xn |
匹配n,其中n為十六進制轉義值。十六進制轉義值必須為確定的兩個數字長。例如,“\x41”匹配“A”。“\x041”則等價於“\x04&1”。正則表達式中可以使用ASCII編碼。 |
\num |
匹配num,其中num是一個正整數。對所獲取的匹配的引用。例如,“(.)\1”匹配兩個連續的相同字符。 |
\n |
標識一個八進制轉義值或一個向后引用。如果\n之前至少n個獲取的子表達式,則n為向后引用。否則,如果n為八進制數字(0-7),則n為一個八進制轉義值。 |
\nm |
標識一個八進制轉義值或一個向后引用。如果\nm之前至少有nm個獲得子表達式,則nm為向后引用。如果\nm之前至少有n個獲取,則n為一個后跟文字m的向后引用。如果前面的條件都不滿足,若n和m均為八進制數字(0-7),則\nm將匹配八進制轉義值nm。 |
\nml |
如果n為八進制數字(0-7),且m和l均為八進制數字(0-7),則匹配八進制轉義值nml。 |
\un |
匹配n,其中n是一個用四個十六進制數字表示的Unicode字符。例如,\u00A9匹配版權符號(©)。 |
\p{P} |
小寫 p 是 property 的意思,表示 Unicode 屬性,用於 Unicode 正表達式的前綴。中括號內的“P”表示Unicode 字符集七個字符屬性之一:標點字符。 其他六個屬性: L:字母; M:標記符號(一般不會單獨出現); Z:分隔符(比如空格、換行等); S:符號(比如數學符號、貨幣符號等); N:數字(比如阿拉伯數字、羅馬數字等); C:其他字符。 *注:此語法部分語言不支持,例:javascript。 |
\< \> |
匹配詞(word)的開始(\<)和結束(\>)。例如正則表達式\<the\>能夠匹配字符串"for the wise"中的"the",但是不能匹配字符串"otherwise"中的"the"。注意:這個元字符不是所有的軟件都支持的。 |
( ) |
將( 和 ) 之間的表達式定義為“組”(group),並且將匹配這個表達式的字符保存到一個臨時區域(一個正則表達式中最多可以保存9個),它們可以用 \1 到\9 的符號來引用。 |
| |
將兩個匹配條件進行邏輯“或”(Or)運算。例如正則表達式(him|her) 匹配"it belongs to him"和"it belongs to her",但是不能匹配"it belongs to them."。注意:這個元字符不是所有的軟件都支持的。 |
最簡單的元字符是點,它能夠匹配任何單個字符(注意不包括換行符)。
2.速記理解
. [ ] ^ $四個字符是所有語言都支持的正則表達式,所以這四個是基礎的正則表達式。正則難理解因為里面有一個等價的概念,這個概念大大增加了理解難度,讓很多初學者看起來會懵,如果把等價都恢復成原始寫法,自己書寫正則就超級簡單了,就像說話一樣去寫你的正則了:
等價:
等價是等同於的意思,表示同樣的功能,用不同符號來書寫。
?,*,+,\d,\w 都是等價字符
?等價於匹配長度{0,1}
*等價於匹配長度{0,}
+等價於匹配長度{1,}
\d等價於[0-9]
\D等價於[^0-9]
\w等價於[A-Za-z_0-9]
\W等價於[^A-Za-z_0-9]。
常用運算符與表達式:
^ 開始
() 域段
[] 包含,默認是一個字符長度
[^] 不包含,默認是一個字符長度
{n,m} 匹配長度
. 任何單個字符(\. 字符點)
| 或
\ 轉義
$ 結尾
[A-Z] 26個大寫字母
[a-z] 26個小寫字母
[0-9] 0至9數字
[A-Za-z0-9] 26個大寫字母、26個小寫字母和0至9數字
, 分割
分割語法:
[A,H,T,W] 包含A或H或T或W字母
[a,h,t,w] 包含a或h或t或w字母
[0,3,6,8] 包含0或3或6或8數字
語法與釋義:
基礎語法 "^([]{})([]{})([]{})$"
正則字符串 = "開始([包含內容]{長度})([包含內容]{長度})([包含內容]{長度})結束"
?,*,+,\d,\w 這些都是簡寫的,完全可以用[]和{}代替,在(?:)(?=)(?!)(?<=)(?<!)(?i)(*?)(+?)這種特殊組合情況下除外。
初學者可以忽略?,*,+,\d,\w一些簡寫標示符,學會了基礎使用再按表自己去等價替換
實例:
字符串;tel:086-0666-88810009999
原始正則:"^tel:[0-9]{1,3}-[0][0-9]{2,3}-[0-9]{8,11}$"
速記理解:開始 "tel:普通文本"[0-9數字]{1至3位}"-普通文本"[0數字][0-9數字]{2至3位}"-普通文本"[0-9數字]{8至11位} 結束"
等價簡寫后正則寫法:"^tel:\d{1,3}-[0]\d{2,3}-\d{8,11}$" ,簡寫語法不是所有語言都支持。
五、參考原文
https://baike.baidu.com/item/%E6%AD%A3%E5%88%99%E8%A1%A8%E8%BE%BE%E5%BC%8F/1700215?fr=aladdin
http://pubs.opengroup.org/onlinepubs/009695399/functions/regcomp.html
http://www.yingzinanfei.com/2017/02/06/c-regex-yongfa/
http://blog.csdn.net/yingzinanfei/article/details/54893394
http://www.cnblogs.com/pmars/archive/2012/10/24/2736831.html