內(nèi)容概覽
有時候我們要處理的是非結(jié)構(gòu)化的數(shù)據(jù),例如網(wǎng)頁或是電郵資料,那么就需要用R來抓取所需的字符串,整理為進(jìn)一步處理的數(shù)據(jù)形式。R語言中有一整套可以用來處理字符的函數(shù),在之前的 博文 中已經(jīng)有所涉及。但真正的要用好字符處理函數(shù),則不得不用到正則表達(dá)式。 正則表達(dá)式(Regular Expression、regexp) 是指一種用來描述一定數(shù)量文本的模式。熟練掌握正則表達(dá)式能使你隨心所欲的操作文本來達(dá)成目標(biāo)。其實(shí)學(xué)習(xí)正則表達(dá)式并沒有想像中的那么困難。最好方法是從例子開始,然后多練習(xí),多使用。網(wǎng)絡(luò)上已經(jīng)有許多不錯的參考資料,例如 這篇 或 那篇 。本文假設(shè)你對正則表達(dá)式有了基本的了解,下面我們來看看如何在R里面來使用它。
R語言處理文本的能力雖然不強(qiáng),但適當(dāng)用用還是可以大幅提高工作效率的,而且有些文本操作還不得不用。高效處理文本少不了正則表達(dá)式(regular expression),雖然R在這方面先天不高效,但它處理字符串的絕大多數(shù)函數(shù)都使用正則表達(dá)式。
正則表達(dá)式簡介:
正則表達(dá)式不是R的專屬內(nèi)容,這里也只簡單介紹,更詳細(xì)的內(nèi)容請查閱其他文章。正則表達(dá)式是用于描述/匹配一個文本集合的表達(dá)式。
1. 所有英文字母、數(shù)字和很多可顯示的字符本身就是正則表達(dá)式,用于匹配它們自己。比如 'a' 就是匹配字母 'a' 的正則表達(dá)式
2. 一些特殊的字符在正則表達(dá)式中不在用來描述它自身,它們在正則表達(dá)式中已經(jīng)被“轉(zhuǎn)義”,這些字符稱為“元字符”。perl類型的正則表達(dá)式中被轉(zhuǎn)義的字符有:. \ | ( ) [ ] { } ^ $ * + ?。被轉(zhuǎn)義的字符已經(jīng)有特殊的意義,如點(diǎn)號 . 表示任意字符;方括號表示選擇方括號中的任意一個(如[a-z] 表示任意一個小寫字符);^ 放在表達(dá)式開始出表示匹配文本開始位置,放在方括號內(nèi)開始處表示非方括號內(nèi)的任一字符;大括號表示前面的字符或表達(dá)式的重復(fù)次數(shù);| 表示可選項(xiàng),即 | 前后的表達(dá)式任選一個。
3. 如果要在正則表達(dá)式中表示元字符本身,比如我就要在文本中查找問號‘?’, 那么就要使用引用符號(或稱換碼符號),一般是反斜杠 '\'。需要注意的是,在R語言中得用兩個反斜杠即 ‘\\’,如要匹配括號就要寫成 ’\\(\\)‘
4. 不同語言或應(yīng)用程序(事實(shí)上很多規(guī)則都通用)定義了一些特殊的元字符用于表示某類字符,如 \d 表示數(shù)字0-9, \D 表示非數(shù)字,\s 表示空白字符(包括空格、制表符、換行符等),\S 表示非空白字符,\w 表示字(字母和數(shù)字),\W 表示非字,\< 和 \> 分別表示以空白字符開始和結(jié)束的文本。
5. 正則表達(dá)式符號運(yùn)算順序:圓括號括起來的表達(dá)式最優(yōu)先,然后是表示重復(fù)次數(shù)的操作(即:* + {} ),接下來是連接運(yùn)算(其實(shí)就是幾個字符放在一起,如abc),最后是表示可選項(xiàng)的運(yùn)算(|)。所以 'foot|bar' 可以匹配’foot‘或者’bar‘,但是 'foot|ba{2}r'匹配的是’foot‘或者’baar‘。
關(guān)于正則表達(dá)式
正則表達(dá)式是編程語言的特色,也是一大難點(diǎn),幾乎各類編程語言都有數(shù)據(jù)自己的正則表達(dá)式,但方法都大同小異,R里面有關(guān)正則表達(dá)式里面選擇采用具有特色的perl語言風(fēng)格的正則表達(dá)式。掌握好這些正則表達(dá)式的使用方法,就可以輕易地完成字符串處理任務(wù)。正則表達(dá)式,又稱正規(guī)表示法、常規(guī)表示法,它使用單個字符串來描述、匹配一系列符合某個句法規(guī)則的字符串。在很多文本編輯器里,正則表達(dá)式通常被用來檢索、替換那些符合某個模式的文本。下面列出正則表達(dá)式常用模式:
[0-9] 匹配所有數(shù)字字符
[^0-9] 匹配所有非數(shù)字字符
[^a-z] 匹配所有非小寫字母字符
^ 匹配字符開頭的字符
$ 匹配字符結(jié)尾的字符
\d 匹配一個數(shù)字的字符
\d+ 匹配多個數(shù)字字符串
\D 非數(shù)字,其他同 \d
\w 英文字母或數(shù)字的字符串,和 [a-zA-Z0-9] 語法一樣
\W 非英文字母或數(shù)字的字符串
\s 空格
\S 非空格
\b 匹配以英文字母,數(shù)字為邊界的字符串
\B 匹配不以英文字母,數(shù)值為邊界的字符串
. 匹配除換行符以外的所有單個字符
[…] 字符組 匹配單個列出的字符
x? 匹配 0 次或一次 x 字符串
x* 匹配 0 次或多次 x 字符串,但匹配可能的最少次數(shù)
x+ 匹配 1 次或多次 x 字符串,但匹配可能的最少次數(shù)
.* 匹配 0 次或一次的任何字符
.+ 匹配 1 次或多次的任何字符
{m} 匹配剛好是 m 個 的指定字符串
{m,n} 匹配在 m個 以上 n個 以下 的指定字符串
{m,} 匹配 m個 以上 的指定字符串
[] 匹配符合 [] 內(nèi)的字符
[^] 匹配不符合 [] 內(nèi)的字符
常用表達(dá)式語法結(jié)構(gòu)
grep, grepl, regexpr, gregexpr 函數(shù)在字符串向量中尋找特定的匹配模式pattern,具體區(qū)別在參數(shù)的選擇。sub, gsub 分別用于替換單個或者全部的匹配模式,這里g意味著global。
詳細(xì)語法:
grep(pattern, x, ignore.case = FALSE, perl = FALSE, value = FALSE,fixed = FALSE, useBytes = FALSE, invert = FALSE)
grepl(pattern, x, ignore.case = FALSE, perl = FALSE,fixed = FALSE, useBytes = FALSE)
sub(pattern, replacement, x, ignore.case = FALSE, perl = FALSE,fixed = FALSE, useBytes = FALSE)
gsub(pattern, replacement, x, ignore.case = FALSE, perl = FALSE,fixed = FALSE, useBytes = FALSE)
regexpr(pattern, text, ignore.case = FALSE, perl = FALSE,fixed = FALSE, useBytes = FALSE)
gregexpr(pattern, text, ignore.case = FALSE, perl = FALSE,fixed = FALSE, useBytes = FALSE)
regexec(pattern, text, ignore.case = FALSE,fixed = FALSE, useBytes = FALSE)
參數(shù)說明:
pattern : 用于匹配的正則表達(dá)式。只接受一個元素。
x, text : 被匹配的字符串(向量)。
ignore.case : 是否大小寫敏感,默認(rèn)為FALSE
perl : 是否兼容Perl語言的正則表達(dá)式,默認(rèn)FALSE
value : 是否返回匹配的值。默認(rèn)FALSE,那么將會返回向量的索引indice; 如果為TRUE,則返回被匹配的字符串
fixed : 如果TRUE,則匹配整個元素。默認(rèn)FALSE
useBytes : 是否使用byte-by-byte還是character-by-character,默認(rèn)FALSE
invert : 是否取反,如果TRUE,則返回未匹配的索引indices或值values
replacement : 適用于sub和gsub。只接受一個元素。
grep(value = FALSE) 返回匹配向量的索引(若invert = TRUE,情況相反)
grep(value = TRUE) 返回匹配向量的原始值
grepl 返回布爾向量,包含是否匹配的信息
sub 和 gsub 返回和原先向量同樣長度的新向量.
regexpr 返回一個和初始向量text長度保持一致的向量數(shù)組,元素為第一次匹配的起點(diǎn)位置(如果沒有匹配成功則顯示-1)同時還附有匹配長度(匹配顯示匹配長度,否則顯示-1),如果想計(jì)算bytes的長度,請使用 useBytes = TRUE
gregexpr和regexec 均返回一個list列表,經(jīng)測試內(nèi)容和regexpr保持一致,但不知其具體區(qū)別,望告知
Example_1
假設(shè)我們有一個字符向量,包括了三個字符串。我們的目標(biāo)是從中抽取電郵地址。R語言中很多字符函數(shù)都能識別正則表達(dá)式,而最重要的函數(shù)就是gregexpr()。該函數(shù)的第一個參數(shù)是正則表達(dá)式,前后需要用引號,對元字符進(jìn)行轉(zhuǎn)義時要用\。第二個參數(shù)是等待處理的文本。那么用如下三行代碼,我們從word字符向量中得到一個列表,其中第一項(xiàng)元素中的5表示電郵地址從第5個字符位置開始,24表示電郵地址長度為24。
> word <- c('abc noboby@stat.berkeley.edu','text with no email','first me@mything.com also you@yourspace.com')
> pattern <- '[-A-Za-z0-9_.%]+@[-A-Za-z0-9_.%]+\\.[A-Za-z]+'
> pattern
[1] "[-A-Za-z0-9_.%]+@[-A-Za-z0-9_.%]+\\.[A-Za-z]+"
> (gregout <- gregexpr(pattern,word))
[[1]]
[1] 5
attr(,"match.length")
[1] 24
attr(,"useBytes")
[1] TRUE
[[2]]
[1] -1
attr(,"match.length")
[1] -1
attr(,"useBytes")
[1] TRUE
[[3]]
[1] 7 27
attr(,"match.length")
[1] 14 17
attr(,"useBytes")
[1] TRUE
>
下一步我們需要將電郵地址抽取出來,此時配合substr函數(shù),即可根據(jù)需要字符串的位置來提取子集。
> substr(word[1],gregout[[1]],gregout[[1]]+attr(gregout[[1]],'match.length')-1)
[1] "noboby@stat.berkeley.edu"
>
更方便的使用方式是根據(jù)上述方法建立一個自定義函數(shù)getcontent,參數(shù)s表示待處理的文本,參數(shù)g表示的是通過gregexpr函數(shù)處理后的結(jié)果。這個函數(shù)我們在后面還會用到。
> getcontent <- function(s,g){
+ substring(s,g,g+attr(g,'match.length')-1)
+ }
> getcontent(word[1],gregout[[1]])
[1] "noboby@stat.berkeley.edu"
> getcontent(word[2],gregout[[2]])
[1] ""
> getcontent(word[3],gregout[[3]])
[1] "me@mything.com" "you@yourspace.com"
>
Example_2
regexpr、gregexpr和regexec 這三個函數(shù)返回的結(jié)果包含了匹配的具體位置和字符串長度信息,可以用于字符串的提取操作。用函數(shù)獲得位置信息后再進(jìn)行字符串提取的操作可以自己試試看。
> text <- c("Hellow, Adam!", "Hi, Adam!", "How are you, Adam.")
> regexpr("Adam", text)
[1] 9 5 14
attr(,"match.length")
[1] 4 4 4
attr(,"useBytes")
[1] TRUE
> gregexpr("Adam", text)
[[1]]
[1] 9
attr(,"match.length")
[1] 4
attr(,"useBytes")
[1] TRUE
[[2]]
[1] 5
attr(,"match.length")
[1] 4
attr(,"useBytes")
[1] TRUE
[[3]]
[1] 14
attr(,"match.length")
[1] 4
attr(,"useBytes")
[1] TRUE
> regexec("Adam", text)
[[1]]
[1] 9
attr(,"match.length")
[1] 4 [[2]]
[1] 5
attr(,"match.length")
[1] 4 [[3]]
[1] 14
attr(,"match.length")
[1] 4
Example_3
這里出現(xiàn)的replacement參數(shù),在x中搜索pattern,并以文本replacement將其替換;其他的各個參數(shù)和grep的作用相同。
> p="Who wins the prize?"
> sub("Who",replacement="James",sub("[\\?]","!",p,perl=T))
[1] "James wins the prize!"
gsub和sub函數(shù)的不同之處在于sub函數(shù)只替換其匹配文本中第一次出現(xiàn)的匹配,而gsub為globe sub全局匹配替換,即替換匹配到的所有匹配值。
> txt <- "a test of capitalizing"
> gsub("(\\w)(\\w*)", "\\U\\1\\L\\2", txt, perl=TRUE)
[1] "A Test Of Capitalizing"
> gsub("\\b(\\w)", "\\U\\1", txt, perl = T)
[1] "A Test Of Capitalizing"
perl正則表達(dá)式中,\為轉(zhuǎn)義符,\w為元字符,匹配數(shù)字型的字符;* 為匹配0個或多個x,\U:大寫字符,直到字符串末尾或碰到\E,\L:小寫字符,直到字符串末尾或碰到\E,\b:匹配以英文字母,數(shù)字為邊界的字符串, \1,\2分別為匹配第一組括號和第二組括號。使用以上的gsub替換,可以將文本每個單詞的首字母大寫。
regexpr,gregexpr,regexec,它們可以返回和text相同長度的向量,包括不匹配的值
> m=c("e","the","end","of","line")
> regexpr("e",m)
[1] 1 3 1 -1 4
attr(,"match.length")
[1] 1 1 1 -1 1
attr(,"useBytes")
[1] TRUE
Example_4
從regexpr()的返回結(jié)果看,返回結(jié)果是個整數(shù)型向量,但是它還具有兩個額外的屬性(attributes),分別是匹配字段的長度和是否按字節(jié)進(jìn)行匹配;regexpr()的返回結(jié)果為-1和1,其中-1表示沒有匹配上,1表示text中第2個元素中的第一個字符被匹配上,且匹配字符的長度為2(屬性值中提供);gregexpr()的返回結(jié)果中包含了全部的匹配結(jié)果的位置信息,而regexpr()只返回了向量text里每個元素中第一個匹配的位置信息,gregexpr()的返回結(jié)果類型是list類型對象;regexec()的返回結(jié)果基本與regexpr()類似,只返回了第一個匹配的位置信息,但其結(jié)果是一個list類型的對象,并且列表里面的元素少了一個屬性值,即attr(,“useBytes”)。
#### grep 和 grepl text <- c("We are the world", "we are the children")
grep("We", text) #向量text中的哪些元素匹配了單詞'We' ## [1] 1
grep("We", text, invert = T) #向量text中的哪些元素沒有匹配單詞'We' ## [1] 2
grep("we", text, ignore.case = T) #匹配時忽略大小寫 ## [1] 1 2
grepl("are", text) #向量text中的每個元素是否匹配了單詞'We',即只返回TRUE或FALSE ## [1] TRUE TRUE #### regexpr、gregexpr和regexec text <- c("We are the world", "we are the children")
regexpr("e", text)
## [1] 2 2 ## attr(,"match.length") ## [1] 1 1 ## attr(,"useBytes") ## [1] TRUE class(regexpr("e", text))
## [1] "integer"
gregexpr("e", text)
## [[1]] ## [1] 2 6 10 ## attr(,"match.length") ## [1] 1 1 1 ## attr(,"useBytes") ## [1] TRUE ## ## [[2]] ## [1] 2 6 10 18 ## attr(,"match.length") ## [1] 1 1 1 1 ## attr(,"useBytes") ## [1] TRUE class(gregexpr("e", text))
## [1] "list"
regexec("e", text)
## [[1]] ## [1] 2 ## attr(,"match.length") ## [1] 1 ## ## [[2]] ## [1] 2 ## attr(,"match.length") ## [1] 1 class(regexec("e", text))
## [1] "list"
除了上面的字符串的查詢,有時還會用到完全匹配,這是會用到match(),其命令形式如下: match(x, table, nomatch= NAinteger, incomparables)只有參數(shù)x的內(nèi)容被完全匹配,函數(shù)才會返回參數(shù)x所在table參數(shù)中的下標(biāo),否則的話會返回nomatch參數(shù)中定義的值(默認(rèn)是NA)。
text <- c("We are the world", "we are the children", "we")
match("we", text)
## [1] 3
match(2, c(3, 4, 2, 8))
## [1] 3
match("xx", c("abc", "xxx", "xx", "xx")) #只會返回第一個完全匹配的元素的下標(biāo) ## [1] 3
match(2, c(3, 4, 2, 8, 2))
## [1] 3
match("xx", c("abc", "xxx")) # 沒有完全匹配的,因此返回NA ## [1] NA
此外還有一個charmatch(),其命令形式類似于match,但從下面的例子來看其行為有些古怪。同樣該函數(shù)也會返回其匹配字符串所在table中的下標(biāo),該函數(shù)在進(jìn)行匹配時,會從table里字符串的最左面(即第一個字符)開始匹配,如果起始位置沒有匹配則返回NA;如果同時部分匹配和完全匹配,則會優(yōu)先選擇完全匹配;如果同時有多個完全匹配或者多個部分匹配時,則會返回0;如果以上三個都沒有,則返回NA。另外還有一個pmatch(),其功能同charmatch()一樣,僅僅寫法不同。
charmatch("xx", c("abc", "xxa"))
## [1] 2
charmatch("xx", c("abc", "axx")) # 從最左面開始匹配 ## [1] NA charmatch("xx", c("xxa", "xxb")) # 不唯一 ## [1] 0 charmatch("xx", c("xxa", "xxb", "xx")) # 優(yōu)先選擇完全匹配,盡管有兩個部分匹配 ## [1] 3 charmatch(2, c(3, 4, 2, 8))
## [1] 3 charmatch(2, c(3, 4, 2, 8, 2))
## [1] 0
不知道這樣一個奇怪的函數(shù)在那里能夠用到,真是有點(diǎn)期待!
Example_5
正則匹配是一個非常常用的字符搜索手段,在數(shù)據(jù)挖掘中起著非常重要的作用。所以雖然它是一種常規(guī)手段,但我還是另起一段來專門講述這個概念。
在R當(dāng)中,可以使用三種正則:
擴(kuò)展正則
基本正則
Perl風(fēng)格正則
正則的使用主要涉汲以下7個函數(shù):grep, grepl, sub, gsub, regexpr, gregrexpr, regexec。而象strsplit, apropos以及browseEnv都是基于這7個函數(shù)基礎(chǔ)之上的。
我們先從正則講起。
假設(shè)我們現(xiàn)在需要從一堆字符當(dāng)中找到一個符合一定規(guī)則的字符串,比如說從一個表格中找到所有人的email地址,或者說找到一段文字中所有的URL地址,你會如何做呢?嗯,回答肯定是正則了。正則就是做這個用的。我們知道一個email地址通常都是這樣的(最簡單情行),xxxxxx@ppp.ddd,其中,xxxxxx可能是任意字母,數(shù)字,以及下劃線,點(diǎn)等組成,而ppp.ddd就是一個域名地址。它們之間以@相隔。在正則下是這樣表示,^[A-Za-z0-9\._%+-]+@[A-Za-z0-9\.-]+\.[A-Za-z]{2,4}$
> pattern="^[A-Za-z0-9\\._%+-]+@[A-Za-z0-9\\.-]+\\.[A-Za-z]{2,4}$"
> str<-c("abc","someone@qiuworld.com","efg","anotherone@gmail.com","thirdone@yahoo.cn")
> #grepl會返回一個邏輯值,l就代表logical, g就代表global
> grepl(pattern,str)
[1] FALSE TRUE FALSE TRUE TRUE
> #grep會返回匹配的id
> grep(pattern,str)
[1] 2 4 5
> #regexpr會返回一個數(shù)字,1表示匹配,-1表示不匹配,還會返回兩個屬性,匹配的長度以及是否使用useBytes。useBytes一般很少會使用到false,因?yàn)槲覀儾惶幚韺捵址?div style="height:15px;">
> sub("\\w+@\\w+\\.[a-zA-Z]{2,4}","sub_function","abc@qiuworld.com,efd@qiuworld.com")
> gsub("\\w+@\\w+\\.[a-zA-Z]{2,4}","gsub_function","abc@qiuworld.com,efd@qiuworld.com")
正則的使用我們已經(jīng)看到了,但是讓人看不明白就是字符. \ | ( ) [ { ^ $ * + - ? 這些符號都是什么意思???
. # 除了換行以外的任意字符
$ # 一行字符串的結(jié)束,它并不代表最后一個字符(因?yàn)閾Q行符并不會被包含進(jìn)匹配當(dāng)中),只代表一行字符串到這里結(jié)束。
.* # 任意一個無換行的字符串, # . 匹配任何一個非換行字符 # * 將匹配一個或者多個之前的字符.
^$ # 一個空行. # 在正則中有方括號[],代表可以匹配其中任何一個字符。而^在[]中就有了新的意義,代表“非”, -代表了“之間”
[a-z]+ # 一個或者多個小寫英文字母 # |代表或者 小括號(...)可以把或者的部分括起來。注意小括號可能還有別的用途,但是在R當(dāng)中先不使用。
\W # \w的反義. # 等同于[^a-zA-Z0-9_]
[:punct:] # 標(biāo)點(diǎn)符號,包括:^ ! " # $ % & ' ( ) * + - . / : ; < = > ? @ [ ] \ _ { } ` ~
[:space:] # 空格,包括tab, newline, vertical tab, form feed, carriage return, and space
$ # 一行字符串的結(jié)束,它并不代表最后一個字符(因?yàn)閾Q行符并不會被包含進(jìn)匹配當(dāng)中),只代表一行字符串到這里結(jié)束。
有人會問了,為什么轉(zhuǎn)義字符都要寫兩次???因?yàn)镽本身也把 \ 當(dāng)成轉(zhuǎn)義字符,所以在寫pattern的時候,就需要使用\\ 來表示轉(zhuǎn)義字符。還有一種辦法就是設(shè)置fixed為TRUE。那么參數(shù)中perl是什么意思呢?其實(shí)就是指是否使用PCRE的算法,我們來看實(shí)例: