乖,別哭的薄殼
          ~一份耕耘,一份收獲~
          posts - 23,comments - 260,trackbacks - 0
          一、簡單模式
          1.元字符
          這里是正則表達式用到的所有元字符
          (  [  {  \  ^  $  |  )  ?  *  +  .
          任何時候要在正則表達式中使用這些元字符,都必須對它們進行轉義。因此要匹配一個問號,正則表達式就該這樣表示:
          var veQMark =/\?/;
          或者這樣表示:
          var veQMark=new RegExp("\\?");
          注意當正則表達式以第二種形式表示時,所有的反斜杠都必須用兩個反斜杠來替換,因為javascript字符串解析器會按照\n的方式嘗試翻譯\?。為了保證不會出現這個問題,在元字符的前面加上兩個反斜杠(雙重轉義)。這個小小的gotcha就是多數開發者更偏好使用字面量語法的原因。

          2.使用特殊字符
          要使用ASCII來表示一個字符,則必須指定一個兩位的十六進制代碼,并有前面加上\x。例如字符b的ASCII碼為98,等于十六進制的62,因此,表示字符b可以使用\x62
          另外也可以使用八進制代替十六進制來指定字符代碼,直接在反斜杠之后跟上八進制數值,例如,b等于八進制的142,所以下面就該這樣
          /\142/
          如果要使用Unicode來表示字符,必須指定字符串的四位十六進制表示形式,因此b應該是\u0062
          另外還有其他一些預定義的特殊字符,如下表所列
          字符 描述
          \t 制表符
          \n 換行符
          \r 回車符
          \f 換頁符
          \a alert字符
          \e escape字符
          \cx 與X相對應的控制字符
          \b 回退字符
          \v 垂直制表符
          \0 空字符

          3.字符類
          3.1簡單類
          假設相匹配"bat","cat","fat"。使用簡單類可以很方便的解決這個問題
          /[bcf]at/gi
          后面的g表示全匹配,i表示不區分大小寫。
          3.2負向類
          要匹配除了a和b的所有字符,那么這個字符就是[^ab]
          3.3范圍類
          [a-z]
          [A-Z]
          [0-9]
          [^1-4]
          3.4組合類
          匹配所有從a~m的字母以及從1~4的數字,以及一個換行符
          [a-m1-4\n]
          3.5預定義類
          代碼 等同于 匹配
          . [^\n\r] 除了換行和回車之外的任意字符
          \d [0-9] 數字
          \D [^0-9] 非數字字符
          \s [\t\n\x0B\f\r] 空白字符
          \S [^\t\n\x0B\f\r] 非空白字符
          \w [a-zA-Z_0-9] 單詞字符(所有字母,所有的數字和下劃線)
          \W [^a-zA-Z_0-9] 非單詞字符

          4.量詞
          4.1簡單量詞
          代碼 描述
          ? 出現零次或一次
          * 出現零次或多次
          + 出現一次或多次(至少出現一次)
          {n} 一定出現n次
          {n,m} 至少出現n次但不超過m次
          {n,} 至少出現n次
          4.2貪婪的、惰性的和支配性的量詞
          貪婪量詞:先看整個的字符串是不是一個匹配,如果沒有發現匹配,它去掉字符串中的最后一個字符,并再次嘗試。直到發現一個匹配字符或者字符串不剩任何字符。
          惰性量詞:先看字符串中的第一個字線是不是一個匹配。如果單獨這個字符還不夠,就讀入下一個字符,組成兩個字符的字符串,如果還是沒有發現匹配,繼續添加字會直到發現一個或者整個字符都檢查過也沒有匹配。其工作方式恰好與貪婪相反。
          支配量詞:只嘗試匹配整個字符串。如果整個字符串不能產生匹配,不做進一步嘗試。
          表示方法如下表
          貪婪 惰性 支配 描述
          ?   ?? ?+ 零次或一次出現
          * *? *+ 零次或多次出現
          + +? ++ 一次或多次出現
          {n} {n}? {n}+ 恰好n次出現
          {n,m} {n,m}? {n,m}+ 至少n次至多m次出現
          {n,} {n,}? {n,}+ 至少n次出現


          瀏覽器對支配量詞的支持還很不完善。IE和Opera不支持,Mozilla將支配量詞看作是貪婪的。

          二、復雜模式
          1.分組
          分組是通過一系列括號包圍一系列字符、字符類以及量詞來使用的。
          /dogdog/可使用分組來重寫成/(dog){2}/
          /([bd]ad?)*/匹配零個和多個"ba","da","bad"或"dad"
          /(mom( and dad)?)/匹配"mom" 或 "mom and dad"

          2.反向引用
          在表達式計算完成后還可以怎樣利用分組?每個分組都被存放在一個特殊的地方以備將來使用。這些存儲在分組中的特殊值,我們稱之為反向引用。
          反向引用是按照從左到右遇到的左括號字符的順序進行創建和編號的。例如表達式(A?(B?(C?)))將產生編號從1-3的三個反向引用
          (1).(A?(B?(C?)))
          (2).(B?(C?))
          (3).(C?)
          反向引用可以有幾種不同的使用方法。
          首先,使用正則表達式對象的test(),match()或search()方法后,反向引用的值可以從RegExp構造函數中獲得。例
          var sToMatch = "#123456789";
          var reNumbers = /#(\d+)/;
          reNumbers.test(sToMatch);
          alert(RegExp.$1);      //outputs "123456789"
          然后,還可以直接在定義分組的表達式中包含反向引用,這可以通過使用特殊轉義序列如\1,\2等等實現
          例如
          var sToMatch = "dogdog";
          var reDogDog = /(dog)\1/;
          alert(reDogDog.test(sToMatch));      //outputs "true"
          正則表達式reDogDog首先創建單詞dog的組,然后又被特殊轉義序列\1引用,使得這個正則表達式等同于/dogdog/
          第三,反向引用可以用在String對象的replace()方法中,這通過使用特殊字符序列$1,$2等等來實現。例如
          var sToChange = "1234 5678";
          var reMatch = /(\d{4}) (\d{4})/
          var sNew  = sToChange.replace(reMatch,"$2 $1");
          alert(sNew);      //outputs "5678 1234"

          3.候選
          有時候要構建一個能夠正確匹配想得到所有可能性的模式是十分困難的。如果要對同一個表達式同時匹配"red"和"break"時要怎么做呢?這些單詞完全沒有相同的字符,這樣就要寫兩個不同的正則表達式,并分別對兩個字符串進行匹配,像這樣:
          var sToMatch1 = "red";
          var sToMatch2 = "black";
          var reRed = /red/;
          var reBlack = /black/;
          alert(reRed.test(sToMatch1)||reBlack.test(sToMatch1));      //outputs "true"
          alert(reRed.test(sToMatch2)||reBlack.test(sToMatch2));      //outputs "true"
          這完成了任務,但是十分冗長。還有另一種方式就是使用正則表達式的候選操作符。例:
          var sToMatch1 = "red";
          var sToMatch2 = "black";
          var reRedOrBlack = /(red|black)/;
          alert(reRedOrBlack.test(sToMatch1));      //outputs "true"
          alert(reRedOrBlack.test(sToMatch2));      //outputs "true"

          4.非捕獲性分組
          創建反向引用的分組,我們稱之為捕獲性分組。同時還有一種非捕獲性分組,它不會創建反向引用。在較長的正則表達式中,存儲反向引用會降低匹配速度。
          如果要創建一個非捕獲性分組,只要在左括號的后面加上一個問號和一個緊跟的冒號:
          var sToMatch = "#123456789";
          var reNumbers = /#(?:\d+)/;
          reNumbers.test(sToMatch);
          alert(RegExp.$1);      //outputs ""

          5.前瞻
          有時候,可能希望,當某個特定的字符分組出現在另一個字符串之前時,才去捕獲它。如果使用“前瞻”就可以讓這個過程變得十分簡單。
          前瞻,告訴正則表達式運算器向前看一些字符而不移動其位置。同樣存在正向前瞻和負向前瞻。正向前瞻檢查的是接下來出現的是不是某個特定字符集。而負向前瞻則是檢查接下來的不應該出現的特定字符集。
          創建正向前瞻要將模式放在(?=和)之間。注意這不是分組,雖然它也用到括號。
          var sToMatch1 = "bedroom";
          var sToMatch2 = "bedding";
          var reBed = /(bed(?=room)/;
          alert(reBed.test(sToMatch1));      //outputs "true"
          alert(RegExp.$1);      //outputs "bed"
          alert(reBed.test(sToMatch2));      //outputs "false"
          負向前瞻,則要將模式放在(?!和)之間。
          var sToMatch1 = "bedroom";
          var sToMatch2 = "bedding";
          var reBed = /(bed(?!room)/;
          alert(reBed.test(sToMatch1));      //outputs "false"
          alert(reBed.test(sToMatch2));      //outputs "true"
          alert(RegExp.$1);      //outputs "bed"

          6.邊界
          邊界 用于正則表達式中表示模式的位置。下表列出了幾種可能的邊界
          邊界 描述
          ^ 行開頭
          $ 行結尾
          \b 單詞的邊界
          \B 非單詞的邊界

          假設想查找一個單詞,但要它只出現在行尾,則可以使用美元符號$來表示它:
          var sToMatch = "Important word is the last one.";
          var reLastWord = /(\w+).$/;
          reLastWord.test(sToMatch);
          alert(RegExp.$1);      //outputs "one"

          7.多行模式
          要指定多行模式,只要在正則表達式后面添加一個m選項。這會讓$邊界匹配換行符(\n)以及字符串真正的結尾。
          var sToMatch = "First second\nthird fourth\nfifth sixth";
          var reLastWordOnLine = /(\w+)$/gm;
          var arrWords = sToMatch.match(reLastWordOnLine);
          表達式返回"second","fourth"和"sixth"。若不指定多行模式,表達式將只返回"sixth"。

          二、RegExp對象
          javascript中的每個正則表達式都是一個對象,同其他的對象一樣。
          1.實例屬性
          global--Boolean值,表示g(全局選項)是否已設置。
          ignoreCase--Boolean值,表示i(忽略大小寫選項)是否已設置。
          lastIndex--整數,代表下次匹配將會從哪個字符位置開始(只有當使用exec()或test()函數才會填入,否則為0)。
          multiline--Boolean值,表示m(多行模式選項)是否已設置。
          source--正則表達式的源字符串形式。例如表達式/[ba]*/的source將返回"[ba]*"。

          2.靜態屬性
          靜態的RegExp屬性對所有的正則表達式都有效。
          長名 短名 描述
          input $_ 最后用于匹配的字符串(傳遞給exec()或test()的字符串)
          lastMatch $& 最后匹配的字符
          lastParen $+ 最后匹配的分組
          leftContext $` 在上次匹配的前面的子串
          multiline $* 用于指定是否所有的表達式都使用多行模式的布爾值
          rightContext $' 在上次匹配之后的子串

          IE和Opera并不支持RegExp.multiline,所以最好單獨的對每個表達式設置m選項而不要直接設置這個標記。

          posted on 2007-04-17 00:45 小祝 閱讀(1621) 評論(7)  編輯  收藏 所屬分類: Javascript

          FeedBack:
          # re: javascript學習筆記(三)--正則表達式
          2007-04-17 14:17 | 穎穎
          you don't go asleep?hehe it's 12:00  回復  更多評論
            
          # re: javascript學習筆記(三)--正則表達式
          2007-04-17 15:37 | 小祝
          就是因為晚了點,所以還沒寫完撒,今天晚上回去接到寫。
          go asleep是不是用錯了,有點怪怪的~呵呵~  回復  更多評論
            
          # re: javascript學習筆記(三)--正則表達式
          2007-04-18 09:26 | cresposhi
          go to sleep go asleep都是對的。。。
          sleep是動詞所以用go to,而asleep是副詞用來修飾go這個動作。  回復  更多評論
            
          # re: javascript學習筆記(三)--正則表達式
          2007-04-18 11:07 | 小祝
          呵呵,稀爛,學的一點鳥語忘光了,本來就學的不好。。。
          慚愧啊~  回復  更多評論
            
          # re: javascript學習筆記(三)--正則表達式
          2007-04-19 18:47 | yangay
          語法好象不對@小祝
            回復  更多評論
            
          # re: javascript學習筆記(三)--正則表達式
          2007-04-19 18:48 | yangay
          what time so intristid in english?
            回復  更多評論
            
          # re: javascript學習筆記(三)--正則表達式 [未登錄]
          2007-08-07 08:33 | zhu
          博主學程序很認真,英語水平確實有問題  回復  更多評論
            
          主站蜘蛛池模板: 江门市| 如皋市| 蓝田县| 台东县| 安化县| 宝山区| 东方市| 清丰县| 铁岭市| 应城市| 思茅市| 本溪市| 和龙市| 扎囊县| 温泉县| 锡林郭勒盟| 陇西县| 曲麻莱县| 永清县| 依安县| 大化| 乐东| 镇雄县| 赣州市| 来凤县| 紫金县| 铁力市| 襄垣县| 曲水县| 龙岩市| 宽甸| 宾川县| 义马市| 陵水| 石城县| 梁平县| 天津市| 铅山县| 宕昌县| 临猗县| 安达市|