-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathtidyverse_stringr.Rmd
910 lines (574 loc) · 17 KB
/
tidyverse_stringr.Rmd
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
# 正则表达式 {#tidyverse-stringr}
```{r stringr-1, message = FALSE, warning = FALSE}
library(tidyverse)
library(stringr)
```
## 问题
这是一份关于地址信息的数据
```{r stringr-2, echo=FALSE, message=FALSE, warning=FALSE}
d <- tibble::tribble(
~No, ~address,
1L, "Sichuan Univ, Coll Chem",
2L, "Sichuan Univ, Coll Elect Engn",
3L, "Sichuan Univ, Dept Phys",
4L, "Sichuan Univ, Coll Life Sci",
6L, "Sichuan Univ, Food Engn",
7L, "Sichuan Univ, Coll Phys",
8L, "Sichuan Univ, Sch Business",
9L, "Wuhan Univ, Mat Sci"
)
d
```
**问题**:如何提取`Sichuan Univ`后面的学院?这需要用到正则表达式的知识。
## 什么是正则表达式
我们在word文档或者excel中,经常使用查找和替换, 然而有些情况,word是解决不了的,比如
- 条件搜索
- 统计文中,前面有 “data”, “computer” or “statistical” 的 “analysis”,这个单词的个数
- 找出文中重复的单词,比如“we love love you”
- 拼写检查
- 电话号码(邮件,密码等)是否正确格式
- 日期书写的规范与统一
- 提取信息
- 提取文本特定位置的数据
- 文本挖掘
- 非结构化的提取成结构化
这个时候就需要用到正则表达式(Regular Expression),这一强大、便捷、高效的文本处理工具。那么,什么是正则表达式呢?简单点说,正则表达式是处理字符串的。复杂点说,正则表达式描述了一种字符串匹配的模式(pattern),通常被用来检索、替换那些符合某个模式(规则)的文本。这种固定的格式的文本,生活中常见的有电话号码、网络地址、邮件地址和日期格式等等。
正则表达式并不是R语言特有的,事实上,几乎所有程序语言都支持正则表达式 (e.g. Perl, Python, Java, Ruby, etc).
R 语言中很多函数都需要使用正则表达式,然而正则表达式不太好学。幸运的是,大神Hadley Wickham开发的stringr包让正则表达式简单易懂,因此今天我们就介绍这个包。本章的内容与《R for data science》第10章基本一致。本章目的教大家写**简单的**正则表示式就行了。
## 字符串基础
### 字符串长度
想获取字符串的长度,可以使用`str_length()`函数
```{r stringr-3}
str_length("R for data science")
```
字符串向量,也适用
```{r stringr-4}
str_length(c("a", "R for data science", NA))
```
数据框里配合dplyr函数,同样很方便
```{r stringr-5}
data.frame(
x = c("a", "R for data science", NA)
) %>%
mutate(y = str_length(x))
```
### 字符串组合
把字符串拼接在一起,使用 `str_c()` 函数
```{r stringr-6}
str_c("x", "y")
```
把字符串拼接在一起,可以设置中间的间隔
```{r stringr-7}
str_c("x", "y", sep = ", ")
```
```{r stringr-8}
str_c(c("x", "y", "z"), sep = ", ")
```
是不是和你想象的不一样,那就`?str_c`,或者试试这个
```{r stringr-9}
str_c(c("x", "y", "z"), c("x", "y", "z"), sep = ", ")
```
用在数据框里
```{r stringr-10}
data.frame(
x = c("I", "love", "you"),
y = c("you", "like", "me")
) %>%
mutate(z = str_c(x, y, sep = "|"))
```
使用collapse选项,是先组合,然后再转换成单个字符串,大家对比下
```{r stringr-11}
str_c(c("x", "y", "z"), c("a", "b", "c"), sep = "|")
```
```{r stringr-12}
str_c(c("x", "y", "z"), c("a", "b", "c"), collapse = "|")
```
### 字符串取子集
截取字符串的一部分,需要指定截取的开始位置和结束位置
```{r stringr-13}
x <- c("Apple", "Banana", "Pear")
str_sub(x, 1, 3)
```
开始位置和结束位置如果是负整数,就表示位置是从后往前数,比如下面这段代码,截取倒数第3个至倒数第1个位置上的字符串
```{r stringr-14}
x <- c("Apple", "Banana", "Pear")
str_sub(x, -3, -1)
```
也可以进行赋值,如果该位置上有字符,就用新的字符替换旧的字符
```{r stringr-15}
x <- c("Apple", "Banana", "Pear")
x
```
```{r stringr-16}
str_sub(x, 1, 1)
```
```{r stringr-17}
str_sub(x, 1, 1) <- "Q"
x
```
## 使用正则表达式进行模式匹配
正则表示式慢慢会呈现了
### 基础匹配
`str_view()` 是查看string是否匹配pattern,如果匹配,就高亮显示
```{r stringr-18}
x <- c("apple", "banana", "pear")
str_view(string = x, pattern = "an")
```
有时候,我们希望在字符`a`前后都有字符(即,a处在两字符中间,如rap, bad, sad, wave,spear等等)
```{r stringr-19}
x <- c("apple", "banana", "pear")
str_view(x, ".a.")
```
这里的`.` 代表任意字符。如果向表达.本身呢?
```{r stringr-20}
c("s.d") %>%
str_view(".")
```
```{r stringr-21}
c("s.d") %>%
str_view("\\.")
```
### 锚点
希望`a`是字符串的开始
```{r stringr-22}
x <- c("apple", "banana", "pear")
str_view(x, "^a")
```
希望`a`是一字符串的末尾
```{r stringr-23}
x <- c("apple", "banana", "pear")
str_view(x, "a$")
```
```{r stringr-24}
x <- c("apple pie", "apple", "apple cake")
str_view(x, "^apple$")
```
### 字符类与字符选项
前面提到,`.`匹配任意字符,事实上还有很多这种**特殊含义**的字符:
* `\d`: matches any digit.
* `\s`: matches any whitespace (e.g. space, tab, newline).
* `[abc]`: matches a, b, or c.
* `[^abc]`: matches anything except a, b, or c.
```{r stringr-25}
str_view(c("grey", "gray"), "gr[ea]y")
```
### 重复
控制匹配次数:
* `?`: 0 or 1
* `+`: 1 or more
* `*`: 0 or more
```{r stringr-26}
x <- "Roman numerals: MDCCCLXXXVIII"
str_view(x, "CC?")
str_view(x, "X+")
```
控制匹配次数:
* `{n}`: exactly n
* `{n,}`: n or more
* `{,m}`: at most m
* `{n,m}`: between n and m
```{r stringr-27}
x <- "Roman numerals: MDCCCLXXXVIII"
str_view(x, "C{2}")
str_view(x, "C{2,}")
str_view(x, "C{2,3}")
```
- 默认的情况,`*`, `+` 匹配都是**贪婪**的,也就是它会尽可能的匹配更多
- 如果想让它不贪婪,而是变得懒惰起来,可以在`*`, `+` 加个`?`
```{r stringr-28}
x <- "Roman numerals: MDCCCLXXXVIII"
str_view(x, "CLX+")
str_view(x, "CLX+?")
```
小结一下呢
```{r stringr-29, out.width = '75%', echo = FALSE}
knitr::include_graphics("images/regex_repeat.jpg")
```
### 分组与回溯引用
```{r stringr-30}
ft <- fruit %>% head(10)
ft
```
我们想看看这些单词里,有哪些字母是重复两次的,比如`aa`, `pp`. 如果用上面学的方法
```{r stringr-31}
str_view(ft, ".{2}", match = TRUE)
```
发现不是和我们的预想不一样呢。
所以需要用到新技术 **分组与回溯引用**,
```{r stringr-32}
str_view(ft, "(.)\\1", match = TRUE)
```
- `.` 是匹配任何字符
- `(.)` 将匹配项括起来,它就用了一个名字,叫`\\1`; 如果有两个括号,就叫`\\1`和`\\2`
- `\\1` 表示回溯引用,表示引用`\\1`对于的`(.)`
所以`(.)\\1`的意思就是,匹配到了字符,后面还希望有个**同样的字符**
如果是匹配`abab`, `wcwc`
```{r stringr-33}
str_view(ft, "(..)\\1", match = TRUE)
```
如果是匹配`abba`, `wccw`呢?
```{r stringr-34}
str_view(ft, "(.)(.)\\2\\1", match = TRUE)
```
是不是很神奇?
### 大小写敏感
希望找出线上平台是google和meet的记录,显然Google和google是一个意思,都应该被筛选出
```{r}
df <- tibble::tribble(
~tch_id, ~online_platform,
105, "Google",
106, "meet",
107, "Zoom",
108, "zoom",
109, "Google Meet",
112, "Microsoft Teams",
113, NA
)
df
```
方法1,使用`stringr::regex()`
```{r}
df %>%
filter(
str_detect(online_platform, regex("google|meet", ignore_case = TRUE))
)
```
方法2,使用`(?i)`
```{r}
df %>%
filter(
str_detect(online_platform, "(?i)google|meet")
)
```
## 解决实际问题
### 确定一个字符向量是否匹配一种模式
实际问题中,想判断是否匹配?可以用到`str_detect()`函数
```{r stringr-35}
x <- c("apple", "banana", "pear")
str_detect(x, "e")
```
数据框中也是一样
```{r stringr-36, echo=FALSE}
d <- tibble(x = c("apple", "banana", "pear"))
d
```
```{r stringr-37}
d %>% mutate(has_e = str_detect(x, "e"))
```
用于筛选也很方便
```{r stringr-38}
d %>% dplyr::filter(str_detect(x, "e"))
```
`stringr::words`包含了牛津字典里常用单词
```{r stringr-39}
stringr::words %>% head()
```
我们统计下以`t`开头的单词,有多少个?
```{r stringr-40}
# How many common words start with t?
sum(str_detect(words, "^t"))
```
我们又一次看到了**强制转换**.
以元音结尾的单词,占比多少?
```{r stringr-41}
# proportion of common words end with a vowel?
mean(str_detect(words, "[aeiou]$"))
```
放在数据框里看看, 看看以`x`结尾的单词是哪些?
```{r stringr-42}
tibble(
word = words
) %>%
dplyr::filter(str_detect(word, "x$"))
```
`str_detect()` 有一个功能类似的函数`str_count()`,区别在于,后者不是简单地返回是或否,而是返回字符串中匹配的数量
```{r stringr-43}
x <- c("apple", "banana", "pear")
str_count(x, "a")
```
```{r stringr-44}
tibble(
word = words
) %>%
mutate(
vowels = str_count(word, "[aeiou]"),
consonants = str_count(word, "[^aeiou]")
)
```
### 确定匹配的位置
大家放心,正则表达式不会重叠匹配。比如用`"aba"`去匹配`"abababa"`,肉眼感觉是三次,但正则表达式告诉我们是两次,因为不会重叠匹配
```{r stringr-45}
str_count("abababa", "aba")
```
```{r stringr-46}
str_view_all("abababa", "aba")
```
这里插播一个正则表达式"\b",它用于匹配单词字符与非单词字符(比如逗号、空格、句点)之间的位置。比如这里返回字符串中`s`的个数
```{r}
c("she sells seashells") %>% str_count(pattern = "s")
```
这里的`s`后面必须跟一个非字符的符号,比如逗号、空格、句点等,因此是2个
```{r}
c("she sells seashells") %>% str_count(pattern = "s\\b")
```
### 提取匹配的内容
```{r stringr-47}
colours <- c(
"red", "orange", "yellow",
"green", "blue", "purple"
)
colour_match <- str_c(colours, collapse = "|")
colour_match
```
colour_match 这里是一个字符串,放在pattern参数位置上也是正则表达式了,
这里注意以下两者的区别
```{r stringr-48, eval=FALSE}
str_view("abcd", "ab|cd")
str_view("abc", "a[bc]d")
```
```{r stringr-49}
more <- "It is hard to erase blue or red ink."
str_extract(more, pattern = colour_match)
```
```{r stringr-50}
str_extract_all(more, pattern = colour_match)
```
```{r stringr-51}
more <- sentences[str_count(sentences, colour_match) > 1]
more
```
取出sentences中,含有有两种和两种颜色以上的句子。不过,不喜欢这种写法,看着费劲,还是用tidyverse的方法
```{r stringr-52}
tibble(sentence = sentences) %>%
filter(str_count(sentences, colour_match) > 1)
```
`str_extract()`提取匹配, 谁先匹配就提取谁
```{r stringr-53}
tibble(x = more) %>%
mutate(color = str_extract(x, colour_match))
```
`str_extract_all()`提取全部匹配项
```{r stringr-54}
tibble(x = more) %>%
mutate(color = str_extract_all(x, colour_match))
```
```{r stringr-55}
tibble(x = more) %>%
mutate(color = str_extract_all(x, colour_match)) %>%
unnest(color)
```
### 替换匹配内容
只替换匹配的第一项
```{r stringr-56}
x <- c("apple", "pear", "banana")
str_replace(x, "[aeiou]", "-")
```
替换全部匹配项
```{r stringr-57}
str_replace_all(x, "[aeiou]", "-")
```
### 拆分字符串
这个和`str_c()`是相反的操作
```{r stringr-58}
lines <- "I love my country"
lines
```
```{r stringr-59}
str_split(lines, " ")
```
```{r stringr-60}
fields <- c("Name: Hadley", "Country: NZ", "Age: 35")
fields %>% str_split(": ", n = 2, simplify = TRUE)
```
## 进阶部分
带有条件的匹配
### look ahead
想匹配Windows,同时希望Windows右侧是`"95", "98", "NT", "2000"`中的一个
```{r stringr-61}
win <- c("Windows2000", "Windows", "Windows3.1")
str_view(win, "Windows(?=95|98|NT|2000)")
```
```{r stringr-62}
win <- c("Windows2000", "Windows", "Windows3.1")
str_view(win, "Windows(?!95|98|NT|2000)")
```
Windows后面的 `()` 是匹配条件,事实上,有四种情形:
- `(?=pattern)` 要求此位置的后面必须匹配表达式pattern
- `(?!pattern)` 要求此位置的后面不能匹配表达式pattern
- `(?<=pattern)` 要求此位置的前面必须匹配表达式pattern
- `(?<!pattern)` 要求此位置的前面不能匹配表达式pattern
```{block stringr-63, type="danger"}
注意:对于正则表达式引擎来说,它是从文本头部向尾部(从左到右)开始解析的,因此对于文本尾部方向,称为“前”,因为这个时候,正则引擎还没走到那块;而对文本头部方向,则称为“后”,因为正则引擎已经走过了那一块地方。
```
### look behind
```{r stringr-64}
win <- c("2000Windows", "Windows", "3.1Windows")
str_view(win, "(?<=95|98|NT|2000)Windows")
```
```{r stringr-65}
win <- c("2000Windows", "Windows", "3.1Windows")
str_view(win, "(?<!95|98|NT|2000)Windows")
```
## 案例分析
### 案例1
我们希望能提取第二列中的数值,构成新的一列
```{r stringr-66}
dt <- tibble(
x = 1:4,
y = c("wk 3", "week-1", "7", "w#9")
)
dt
```
```{r stringr-67, eval=FALSE,include=FALSE}
dt %>%
mutate(z = parse_number(y))
```
```{r stringr-68, eval = FALSE, include=FALSE}
# parse_number() parse_integer() parse_double()
# parse_factor() parse_logical() parse_character()
# parse_datetime() parse_time() parse_date()
```
```{r stringr-69}
dt %>%
mutate(
z = str_extract(y, "[0-9]")
)
```
### 案例2
提取第二列中的大写字母
```{r stringr-70}
df <- data.frame(
x = seq_along(1:7),
y = c("2016123456", "20150513", "AB2016123456", "J2017000987", "B2017000987C", "aksdf", "2014")
)
df
```
```{r stringr-71}
df %>%
mutate(
item = str_extract_all(y, "[A-Z]")
) %>%
tidyr::unnest(item)
```
### 案例3
要求:中英文分开
```{r stringr-72, include=FALSE}
tb <- tibble(x = c("I我", "love爱", "you你"))
tb
```
```{r stringr-73}
tb %>%
tidyr::extract(
# x, c("en", "cn"), "([:alpha:]+)([^:alpha:]+)",
x, c("en", "cn"), "([a-zA-Z]+)([^a-zA-Z]+)",
remove = FALSE
)
```
### 案例4
要求:提取起始数字
```{r stringr-74}
df <- tibble(x = c("1-12week", "1-10week", "5-12week"))
df
```
```{r stringr-75}
df %>% extract(
x,
# c("start", "end", "cn"), "([:digit:]+)-([:digit:]+)([^:alpha:]+)",
c("start", "end", "cn"), "(\\d+)-(\\d+)(\\D+)",
remove = FALSE
)
```
### 案例5
要求:提取大写字母后的数字
```{r stringr-76}
df <- tibble(
x = c("12W34", "AB2C46", "B217C", "akTs6df", "21WD4")
)
```
```{r stringr-77}
df %>%
mutate(item = str_extract_all(x, "(?<=[A-Z])[0-9]")) %>%
tidyr::unnest(item)
```
思考题,
- 如何提取大写字母后的连续数字,比如B217C后面的217
- 如何提取提取数字前的大写字母?
- 为什么第一个正则表达式返回结果为""
```{r stringr-78}
x <- "Roman numerals: MDCCCLXXXVIII"
str_match_all(x, "C?")
str_match_all(x, "CC?")
```
"?"的意思是匹配0次或者1次
### 案例6
提取数字并求和
```{r stringr-79}
df <- tibble(
x = c("1234", "B246", "217C", "2357f", "21WD4")
)
df
```
```{r stringr-80}
df %>%
mutate(num = str_match_all(x, "\\d")) %>%
unnest(num) %>%
mutate_at(vars(num), as.numeric) %>%
group_by(x) %>%
summarise(sum = sum(num))
```
### 案例7
```{r stringr-81}
text <- "Quantum entanglement is a physical phenomenon that occurs when pairs or groups of particles are generated, interact, or share spatial proximity in ways such that the quantum state of each particle cannot be described independently of the state of the others, even when the particles are separated by a large distance."
pairs <-
tibble::tribble(
~item, ~code,
"Quantum entanglement", "A01",
"physical phenomenon", "A02",
"quantum state", "A03",
"quantum mechanics", "A04"
) %>%
tibble::deframe()
text %>% str_replace_all(pairs)
```
## 回答提问
回到上课前的提问:如何提取`Sichuan Univ`后面的学院?
```{r stringr-82, echo=FALSE, message=FALSE, warning=FALSE}
d <- tibble::tribble(
~No, ~address,
1L, "Sichuan Univ, Coll Chem",
2L, "Sichuan Univ, Coll Elect Engn",
3L, "Sichuan Univ, Dept Phys",
4L, "Sichuan Univ, Coll Life Sci",
6L, "Sichuan Univ, Food Engn",
7L, "Sichuan Univ, Coll Phys",
8L, "Sichuan Univ, Sch Business",
9L, "Wuhan Univ, Mat Sci"
)
d
```
```{r stringr-83}
d %>%
dplyr::mutate(
coll = str_extract(address, "(?<=Sichuan Univ,).*")
) %>%
tidyr::unnest(coll, keep_empty = TRUE)
```
当然还有其他的解决办法
```{r stringr-84, eval=FALSE}
d %>% mutate(
coll = str_remove_all(address, ".*,")
)
```
```{r stringr-85, eval=FALSE}
d %>% tidyr::separate(
address,
into = c("univ", "coll"), sep = ",", remove = FALSE
)
```
```{r stringr-86, eval=FALSE}
d %>%
tidyr::extract(
address, c("univ", "coll"), "(Sichuan Univ), (.+)",
remove = FALSE
)
```