亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 編程設計 > 正文

在C/C++語言中使用正則表達式

2020-07-14 13:42:44
字體:
來源:轉載
供稿:網友

  本文所有的內容都來自網絡,加上一些個人的理解,不保證正確性,歡迎批評指正
  說起正則表達式(Regular Expression),也許有的朋友天天都在使用,比如grep、vim、sed、awk,只是可能對這個名詞不大熟悉。正則表達式一般簡寫為regex或者regexp,甚至是RE。關于正則表達式的介紹,有很多的文章,用搜索引擎查找就可以找到很不錯的使用說明。但是在C/C++語言中如何去使用,相應的介紹比較缺乏。大多數C標準庫自帶regex,可以通過/usr/include/regex.h去看,或者man regex看使用說明。perl,php等語言更是提供了功能強大的正則表達式,最著名的C語言正則表達式庫為PCRE(Perl Compatible Regular Expression)。本文主要對regex和pcre的使用做一點入門介紹。
  1、regex
  regex的使用非常簡單,只要看一下示例代碼1就能明白(示例代碼是從“GNU C 規則表達式入門”這篇文章里摘取出來的,是否為原始出處就
  不得而知了)。
  CODE:#include <stdio.h>
  #include <string.h>
  #include <regex.h>
  #define SUBSLEN 10              /* 匹配子串的數量 */
  #define EBUFLEN 128          /* 錯誤消息buffer長度 */
  #define BUFLEN 1024          /* 匹配到的字符串buffer長度 */
  int main()
  {
  size_t       len;
  regex_t       re;      /* 存儲編譯好的正則表達式,正則表達式在使用之前要經過編譯 */
  regmatch_t    subs [SUBSLEN]; /* 存儲匹配到的字符串位置 */
  char          matched [BUFLEN];     /* 存儲匹配到的字符串 */
  char          errbuf [EBUFLEN]; /* 存儲錯誤消息 */
  int          err, i;
  char          src    [] = "111 <title>Hello World</title> 222"; /* 源字符串 */
  char          pattern [] = "<title>(.*)</title>"; /* pattern字符串 */
  printf("String : %s/n", src);
  printf("Pattern: /"%s/"/n", pattern);
  /* 編譯正則表達式 */
  err = regcomp(&re, pattern, REG_EXTENDED);
  if (err) {
  len = regerror(err, &re, errbuf, sizeof(errbuf));
  printf("error: regcomp: %s/n", errbuf);
  return 1;
  }
  printf("Total has subexpression: %d/n", re.re_nsub);
  /* 執行模式匹配 */
  err = regexec(&re, src, (size_t) SUBSLEN, subs, 0);
  if (err == REG_NOMATCH) { /* 沒有匹配成功 */
  printf("Sorry, no match .../n");
  regfree(&re);
  return 0;
  } else if (err) {   /* 其它錯誤 */
  len = regerror(err, &re, errbuf, sizeof(errbuf));
  printf("error: regexec: %s/n", errbuf);
  return 1;
  }
  /* 如果不是REG_NOMATCH并且沒有其它錯誤,則模式匹配上 */
  printf("/nOK, has matched .../n/n");
  for (i = 0; i <= re.re_nsub; i++) {
  len = subs[i].rm_eo - subs[i].rm_so;
  if (i == 0) {
  printf ("begin: %d, len = %d   ", subs[i].rm_so, len); /* 注釋1 */
  } else {
  printf("subexpression %d begin: %d, len = %d   ", i, subs[i].rm_so, len);
  }
  memcpy (matched, src + subs[i].rm_so, len);
  matched[len] = '/0';
  printf("match: %s/n", matched);
  }
  regfree(&re); /* 用完了別忘了釋放 */
  return (0);
  }
  執行結果是
  CODE:String : 111 <title>Hello World</title> 222
  Pattern: "<title>(.*)</title>"
  Total has subexpression: 1
  OK, has matched ...
  begin: %, len = 4   match: <title>Hello World</title>
  subexpression 1 begin: 11, len = 11   match: Hello World
  從示例程序可以看出,使用之前先用regcomp()編譯一下,然后調用regexec()進行實際匹配。如果只是看有沒有匹配成功,掌握這2個函數的用法即可。有時候我們想要取得匹配后的子表達式,比如示例中想獲得title是什么,需要用小括號 "( )"把子表達式括起來"<title>(.*)</title>",表達式引擎會將小括號 "( )" 包含的表達式所匹配到的字符串記錄下來。在獲取匹配結果的時候,小括號包含的表達式所匹配到
  的字符串可以單獨獲取,示例程序就是我用來獲取http網頁的主題(title)的方式。
  regmatch_t subs[SUBSLEN]是用來存放匹配位置的,subs[0]里存放這個匹配的字符串位置,subs[1]里存放第一個子表達式的匹配位置,也就是例子中的title,通過結構里的rm_so和rm_eo可以取到,這一點很多人不太注意,應該強調一下。
  注釋1:開始調試代碼的時候是在FreeBSD 6.2上進行的,print出來的len總是0,但print出來的字符串又沒錯,很是迷惑,把它放到Linux上則完全正常,后來仔細檢查才發現rm_so在Linux上是32位,在FreeBSD上是64位,用%d的話實際取的是rm_so的高32位,而不是實際的len,把print rm_so的地方改為%llu就可以了。
  regex雖然簡單易用,但對正則表達式的支持不夠強大,中文處理也有問題,于是引出了下面要說的PCRE。
  2、PCRE   (http://www.pcre.org)
  PCRE的名字就說明了是Perl Compatible,熟悉Perl、PHP的人使用起來完全沒有問題。PCRE有非常豐富的使用說明和示例代碼(看看
  pcredemo.c就能明白基本的用法),下面的程序只是把上面regex改為pcre。
  CODE:/* Compile thuswise:
  * gcc -Wall pcre1.c -I/usr/local/include -L/usr/local/lib -R/usr/local/lib -lpcre
  *
  */
  #include <stdio.h>
  #include <string.h>
  #include <pcre.h>
  #define OVECCOUNT 30 /* should be a multiple of 3 */
  #define EBUFLEN 128
  #define BUFLEN 1024
  int main()
  {
  pcre          *re;
  const char    *error;
  int          erroffset;
  int          ovector[OVECCOUNT];
  int          rc, i;
  char          src [] = "111 <title>Hello World</title> 222";
  char          pattern [] = "<title>(.*)</title>";
  printf("String : %s/n", src);
  printf("Pattern: /"%s/"/n", pattern);
  re = pcre_compile(pattern, 0, &error, &erroffset, NULL);
  if (re == NULL) {
  printf("PCRE compilation failed at offset %d: %s/n", erroffset, error);
  return 1;
  }
  rc = pcre_exec(re, NULL, src, strlen(src), 0, 0, ovector, OVECCOUNT);
  if (rc < 0) {
  if (rc == PCRE_ERROR_NOMATCH) printf("Sorry, no match .../n");
  else printf("Matching error %d/n", rc);
  free(re);
  return 1;
  }
  printf("/nOK, has matched .../n/n");
  for (i = 0; i < rc; i++) {
  char *substring_start = src + ovector[2*i];
  int substring_length = ovector[2*i+1] - ovector[2*i];
  printf("%2d: %.*s/n", i, substring_length, substring_start);
  }
  free(re);
  return 0;
  }
  執行結果是:
  CODE:String : 111 <title>Hello World</title> 222
  Pattern: "<title>(.*)</title>"
  OK, has matched ...
  0: <title>Hello World</title>
  1: Hello World
  比較這2個例子可以看出,在regex用的是regcomp()、regexec(),pcre則使用pcre_compile()、pcre_exec(),用法幾乎完全一致。
  pcre_compile()有很多選項,詳細說明參見http://www.pcre.org/pcre.txt。如果是多行文本,可以設置PCRE_DOTALL的選項pcre_complie(re,
  PCRE_DOTALL,....),表示'.'也匹配回車換行"/r/n"。
  3、pcre++
  pcre++(http://www.daemon.de/PCRE)對pcre做了c++封裝,使用起來更加方便。
  CODE:/*
  * g++ pcre2.cpp -I/usr/local/include -L/usr/local/lib -R/usr/local/lib -lpcre++ -lpcre
  */
  #include <string>
  #include <iostream>
  #include <pcre++.h>
  using namespace std;
  using namespace pcrepp;
  int main()
  {
  string src("111 <title>Hello World</title> 222");
  string pattern("<title>(.*)</title>");
  cout << "String : " << src << endl;
  cout << "Pattern : " << pattern << endl;
  Pcre reg(pattern, PCRE_DOTALL);
  if (reg.search(src) == true) { //
  cout << "/nOK, has matched .../n/n";
  for(int pos = 0; pos < reg.matches(); pos++) {
  cout << pos << ": " << reg[pos] << endl;
  }
  } else {
  cout << "Sorry, no match .../n";
  return 1;
  }
  return 0;
  }
  執行結果是:
  CODE:String : 111 <title>Hello World</title> 222
  Pattern : <title>(.*)</title>
  OK, has matched ...
  0: Hello World
  4、oniguruma
  還有一個正則表達式的庫oniguruma(http://www.geocities.jp/kosako3/oniguruma/),對于東亞文字支持比較好,開始是用在ruby上,也可用于C++,是日本的開發人員編寫的。大多數人都不會用到,也就不做介紹了。如果有疑問可以通過email來討論它的用法。
  5、Regular Expression的內部實現
  關于Regular Expression的實現,用到了不少自動機理論(Automata Theory)的知識,有興趣的可以找這方面的資料來看,這本書“
  Introduction to Automata Theory, Languages, and Computation”寫的很好,編譯原理的書也有這方面的內容。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国内精品美女av在线播放| 国产精品欧美亚洲777777| 91精品久久久久久久久久久| 久久精品在线视频| 热久久这里只有精品| 亚洲女人天堂成人av在线| 人九九综合九九宗合| 日韩在线视频线视频免费网站| 日韩精品久久久久久福利| 麻豆成人在线看| 国产精品私拍pans大尺度在线| 日韩中文字幕网址| 国产精品成人v| 日韩中文字幕久久| 成人免费淫片视频软件| 亚洲一区制服诱惑| 国产精品美女久久久久av超清| 亚洲xxxxx电影| 欧美色欧美亚洲高清在线视频| 欧美激情第99页| 色久欧美在线视频观看| 欧美小视频在线| 精品国产老师黑色丝袜高跟鞋| 久久久精品免费| 91沈先生作品| zzjj国产精品一区二区| 日韩视频免费大全中文字幕| 精品久久久久久久久中文字幕| 久久成年人免费电影| 深夜福利一区二区| 91九色蝌蚪国产| 日韩成人激情视频| 欧美性猛交xxxx| 国产噜噜噜噜久久久久久久久| 久久青草精品视频免费观看| 欧美一区二区三区四区在线| 欧美日韩精品国产| 久久99热精品| 国产福利成人在线| 亚洲欧美国产制服动漫| 亚洲一区二区福利| 亚洲国产精品va在线观看黑人| 2019av中文字幕| 欧美激情影音先锋| 亚洲精品电影久久久| 欧洲美女7788成人免费视频| 色老头一区二区三区| 成人女保姆的销魂服务| 啊v视频在线一区二区三区| 欧美成人午夜激情| 国产精品女人网站| 欧美日韩在线影院| 欧美亚洲视频在线观看| 亚洲qvod图片区电影| 欧美激情视频免费观看| 欧美xxxx综合视频| 国产亚洲欧美日韩精品| 成人淫片在线看| 国产不卡视频在线| 欧亚精品在线观看| 亚洲第一福利视频| 久久夜色精品亚洲噜噜国产mv| 色综合久久中文字幕综合网小说| 欧美视频不卡中文| 亚洲欧洲中文天堂| 68精品久久久久久欧美| 日韩久久免费电影| 伊人青青综合网站| 国产精品午夜国产小视频| 亚洲综合第一页| 亚洲最新中文字幕| 亚洲成人xxx| 日韩av片电影专区| 18一19gay欧美视频网站| 亚洲xxx自由成熟| 欧美在线视频一区| 欧美在线免费观看| 日韩精品视频免费专区在线播放| 精品少妇v888av| 伦伦影院午夜日韩欧美限制| 久热爱精品视频线路一| 亚洲www永久成人夜色| 久久成年人免费电影| 久久久久久高潮国产精品视| 久久久久久中文字幕| 国内精品久久久久影院 日本资源| 亚洲国产精彩中文乱码av| 国产手机视频精品| 超碰日本道色综合久久综合| 中文字幕精品在线| 久久91亚洲精品中文字幕| 欧美最顶级丰满的aⅴ艳星| 国产亚洲一区精品| 久久成人精品电影| 亚洲免费福利视频| 国产精品∨欧美精品v日韩精品| 日本精品视频在线播放| 黑人极品videos精品欧美裸| 亚洲深夜福利网站| 亚洲一区二区三区四区在线播放| 国产精品三级久久久久久电影| 色综合伊人色综合网站| 日韩a**中文字幕| 国产视频福利一区| 亚洲人成网站色ww在线| 国产精品国语对白| 国产激情视频一区| 国产一区玩具在线观看| 欧美肥臀大乳一区二区免费视频| 国产精品夫妻激情| 亚洲最新在线视频| 秋霞成人午夜鲁丝一区二区三区| 亚洲国产另类 国产精品国产免费| 欧美激情在线观看视频| 欧美极品少妇xxxxⅹ免费视频| 亚洲欧美在线一区二区| 日本高清+成人网在线观看| 国产精品吴梦梦| 日韩av在线网页| 亚洲尤物视频网| 欧美专区第一页| 成人免费视频在线观看超级碰| 欧美丝袜美女中出在线| 久久久www成人免费精品张筱雨| 欧美超级免费视 在线| 国产激情综合五月久久| 中文字幕成人在线| 91夜夜揉人人捏人人添红杏| 欧美精品久久久久| 精品欧美激情精品一区| 亚洲国产成人精品久久| 国产成人小视频在线观看| 97av在线视频免费播放| 91国产视频在线| 久久久久国产精品www| 国产在线精品一区免费香蕉| 538国产精品一区二区在线| 成人亚洲激情网| 91亚洲国产成人久久精品网站| 日韩中文字幕在线视频播放| 亚洲欧美日韩国产成人| 日韩在线观看免费全集电视剧网站| 亚洲最大成人免费视频| 亚洲有声小说3d| 欧美在线播放视频| 成人黄色av网站| 国产成人精品免高潮费视频| 欧美日韩国产中文字幕| 日韩在线观看成人| 欧美又大又粗又长| 国产成人午夜视频网址| 欧美激情一级精品国产| 欧美裸体视频网站| 欧美国产激情18| 搡老女人一区二区三区视频tv| 亚洲另类图片色| 久久视频在线看| 午夜欧美不卡精品aaaaa| 91久久久久久| 欧美一级视频免费在线观看| 国产精品伦子伦免费视频| 精品性高朝久久久久久久| 欧美激情精品久久久久久黑人| 欧美野外猛男的大粗鳮|