安裝需要的packages

packages = c("dplyr", "tidytext", "jiebaR", "gutenbergr", "stringr", "wordcloud2", "ggplot2", "tidyr", "scales")
existing = as.character(installed.packages()[,1])
for(pkg in packages[!(packages %in% existing)]) install.packages(pkg)
require(dplyr)
require(tidytext)
require(jiebaR)
require(gutenbergr)
library(stringr)
library(wordcloud2)
library(ggplot2)
library(tidyr)
library(scales)
# 下載 "三國演義" 書籍,並且將text欄位為空的行給清除,以及將重複的語句清除
three <- gutenberg_download(23950) %>% filter(text!="") %>% distinct(gutenberg_id, text)
doc = paste0(three$text,collapse = "") #將text欄位的全部文字合併
docVector = unlist(strsplit(doc,"[。.]"), use.names=FALSE) #以全形或半形句號斷句
three = data.frame(gutenberg_id = "23950" , text = docVector, stringsAsFactors = FALSE) #gutenberg_id換成自己的書本id

#View(three)
# 斷完句後
head(three, 20)
##    gutenberg_id
## 1         23950
## 2         23950
## 3         23950
## 4         23950
## 5         23950
## 6         23950
## 7         23950
## 8         23950
## 9         23950
## 10        23950
## 11        23950
## 12        23950
## 13        23950
## 14        23950
## 15        23950
## 16        23950
## 17        23950
## 18        23950
## 19        23950
## 20        23950
##                                                                                    text
## 1  第一回:宴桃園豪傑三結義,斬黃巾英雄首立功  詞曰:  滾滾長江東逝水,浪花淘盡英雄
## 2                                                是非成敗轉頭空:青山依舊在,幾度夕陽紅
## 3                                                          白髮漁樵江渚上,慣看秋月春風
## 4                                                一壺濁酒喜相逢:古今多少事,都付笑談中
## 5                            話說天下大勢,分久必合,合久必分:周末七國分爭,并入於秦
## 6                                                    及秦滅之後,楚、漢分爭,又并入於漢
## 7                                                      漢朝自高祖斬白蛇而起義,一統天下
## 8                                                    後來光武中興,傳至獻帝,遂分為三國
## 9                                                        推其致亂之由,殆始於桓、靈二帝
## 10                                                               桓帝禁錮善類,崇信宦官
## 11                                   及桓帝崩,靈帝即位,大將軍竇武、太傅陳蕃,共相輔佐
## 12                             時有宦官曹節等弄權,竇武、陳蕃謀誅之,作事不密,反為所害
## 13                                                                         中涓自此愈橫
## 14                                                       建寧二年四月望日,帝御溫德殿
## 15                       方陞座,殿角狂風驟起,只見一條大青蛇,從梁上飛將下來,蟠於椅上
## 16                                                     帝驚倒,左右急救入宮,百官俱奔避
## 17                                                                       須臾,蛇不見了
## 18                                   忽然大雷大雨,加以冰雹,落到半夜方止,壞卻房屋無數
## 19                       建寧四年二月,洛陽地震;又海水泛溢,沿海居民,盡被大浪捲入海中
## 20                                                                   光和元年,雌雞化雄

觀察資料我們可以發現,三國演義中每章的開始會有“第X回:”為標題。
差別在於有的章節單純一“第X回”表示

# 根據上方整理出來的規則,我們可以使用正規表示式,將句子區分章節
three <- three %>% 
  mutate(chapter = cumsum(str_detect(three$text, regex("第.*回:"))))
# 使用三國演義專有名詞字典
jieba_tokenizer <- worker(user="three_kingdoms.traditional.dict", stop_word = "stop_words.txt")
# 設定斷詞function
three_tokenizer <- function(t) {
  lapply(t, function(x) {
    tokens <- segment(x, jieba_tokenizer)
    return(tokens)
  })
}
tokens <- three %>% unnest_tokens(word, text, token=three_tokenizer)
str(tokens)
## 'data.frame':    236502 obs. of  3 variables:
##  $ gutenberg_id: chr  "23950" "23950" "23950" "23950" ...
##  $ chapter     : int  1 1 1 1 1 1 1 1 1 1 ...
##  $ word        : chr  "第一回" "宴桃園豪傑三結義" "斬黃巾英雄首立功" "詞曰" ...
head(tokens, 20)
##      gutenberg_id chapter             word
## 1           23950       1           第一回
## 1.1         23950       1 宴桃園豪傑三結義
## 1.2         23950       1 斬黃巾英雄首立功
## 1.3         23950       1             詞曰
## 1.4         23950       1             滾滾
## 1.5         23950       1               長
## 1.6         23950       1             江東
## 1.7         23950       1             逝水
## 1.8         23950       1             浪花
## 1.9         23950       1             淘盡
## 1.10        23950       1             英雄
## 2           23950       1         是非成敗
## 2.1         23950       1             轉頭
## 2.2         23950       1               空
## 2.3         23950       1             青山
## 2.4         23950       1             依舊
## 2.5         23950       1           夕陽紅
## 3           23950       1             白髮
## 3.1         23950       1             漁樵
## 3.2         23950       1           江渚上

三國演義文字雲

# 計算詞彙的出現次數,如果詞彙只有一個字則不列入計算
tokens_count <- tokens %>% 
  filter(nchar(.$word)>1) %>%
  group_by(word) %>% 
  summarise(sum = n()) %>% 
  filter(sum>20) %>%
  arrange(desc(sum))

# 印出最常見的20個詞彙
head(tokens_count, 20)
## # A tibble: 20 x 2
##    word    sum
##    <chr> <int>
##  1 玄德   1779
##  2 孔明   1644
##  3 曹操    922
##  4 將軍    725
##  5 卻說    644
##  6 丞相    533
##  7 關公    501
##  8 二人    460
##  9 雲長    430
## 10 不可    425
## 11 荊州    408
## 12 張飛    364
## 13 引兵    359
## 14 呂布    355
## 15 商議    341
## 16 軍士    321
## 17 魏延    321
## 18 主公    319
## 19 大喜    310
## 20 孫權    309
tokens_count %>% wordcloud2()

《三國演義》描寫的是從東漢末年到西晉初年近百年間的歷史。在對三國態度上,尊劉反曹鄙吳是民間的主要傾向,以劉備集團作為描寫的中心。

各章節長度,以語句數來計算

plot <- 
  bind_rows(
    three %>% 
      group_by(chapter) %>% 
      summarise(count = n(), type="sentences"),
    tokens %>% 
      group_by(chapter) %>% 
      summarise(count = n(), type="words")) %>% 
  ggplot(aes(x = chapter, y=count, fill="type", color=factor(type))) +
  geom_line() + 
  ggtitle("各章節的句子總數") + 
  xlab("章節") + 
  ylab("句子數量") #+ 
  #theme(text = element_text(family = "Heiti TC Light"))
plot

true <-  gutenberg_download(25606) %>% filter(text!="") %>% distinct(gutenberg_id, text)
#View(book)
doc <- paste0(true$text,collapse = "") #將text欄位的全部文字合併
docVector = unlist(strsplit(doc,"[。.]"), use.names=FALSE) #以全形或半形句號斷句
true <- data.frame(gutenberg_id = "25606" , text = docVector,stringsAsFactors = F) #gutenberg_id換成自己的書本id

計算出現次數最多的3個詞彙其在各章節的總數~畫圖

top20 <- head(tokens_count, 20)
top3 <- head(top20,3)

top1_tokens <- subset(tokens, str_detect(tokens$word, top3$word[1]))

top2_tokens <- subset(tokens, str_detect(tokens$word, top3$word[2]))
#View(top2_tokens)

top3_tokens <- subset(tokens, str_detect(tokens$word, top3$word[3]))

binds <- bind_rows(
  top1_tokens %>% 
    group_by(chapter) %>% 
    summarise(count = n(), name=top3$word[1]),
  top2_tokens %>% 
    group_by(chapter) %>% 
    summarise(count = n(), name=top3$word[2]),
  top3_tokens %>% 
    group_by(chapter) %>% 
    summarise(count = n(), name=top3$word[3]))

top3_plot <- 
  binds %>% 
  group_by(name)%>%
  ggplot(aes(x = chapter, y=count, fill="type", color=factor(name))) +
  geom_line() + 
  ggtitle("最常出現的3個詞彙其在各章節的數量") + 
  xlab("章節") + 
  ylab("數量")
top3_plot


從上圖中可以看出孔明在35~50 以及 80 ~ 100回中出現較多次數,因為都是很著名的故事: 三顧茅廬、巧佈八陣圖、七擒孟獲等等,那後面章節孔明之所以很常出現可能跟玄德死後他答應要輔佐阿斗有關,而我們推測曹操跟玄德可能都在80幾回去世。

bro1_tokens <- subset(tokens, str_detect(tokens$word, "玄德"))

bro2_tokens <- subset(tokens, str_detect(tokens$word, "關羽"))

bro3_tokens <- subset(tokens, str_detect(tokens$word, "張飛"))

binds_bro <- bind_rows(
  bro1_tokens %>% 
    group_by(chapter) %>% 
    summarise(count = n(), name="玄德"),
  bro2_tokens %>% 
    group_by(chapter) %>% 
    summarise(count = n(), name="關羽"),
  bro3_tokens %>% 
    group_by(chapter) %>% 
    summarise(count = n(), name="張飛"))

bro3_plot <- 
  binds_bro %>% 
  group_by(name)%>%
  ggplot(aes(x = chapter, y=count, fill="type", color=factor(name))) +
  geom_line()+
  geom_point() + 
  ggtitle("桃園三結義在各章節的數量") + 
  xlab("章節") + 
  ylab("數量")
bro3_plot


接下來列出三兄弟在各章節的出現次數,可以看到三人主要集中在前80回,有幾個重要的時間點: 第一回的桃園三結義所以他們都有出現;第65回馬超在葭萌關大戰張飛所以此時張飛出現很多次;張飛在關羽死後因為傷心而喝酒,借醉鞭打部下所以引起部下不滿而被暗算,所以80回後就很少出現張飛了。另外,我們推測,由於劉備是政治家、掌權者,所以作者是以劉備的故事為主線,而較常寫到他

# 使用三國志專有名詞字典
jieba_tokenizer <- worker(user="true_three.traditional.dict", stop_word = "stop_words.txt")

# 設定斷詞function
book_tokenizer <- function(t) {
  lapply(t, function(x) {
    tokens <- segment(x, jieba_tokenizer)
    return(tokens)
  })
}
true_tokens <- true %>% unnest_tokens(word, text, token=book_tokenizer)
str(true_tokens)
## 'data.frame':    164633 obs. of  2 variables:
##  $ gutenberg_id: chr  "25606" "25606" "25606" "25606" ...
##  $ word        : chr  "魏書" "武帝紀" "第一" "太祖" ...
head(true_tokens, 50)
##      gutenberg_id               word
## 1           25606               魏書
## 1.1         25606             武帝紀
## 1.2         25606               第一
## 1.3         25606               太祖
## 1.4         25606             武皇帝
## 1.5         25606         沛國譙人也
## 1.6         25606               姓曹
## 1.7         25606               諱操
## 1.8         25606             字孟德
## 1.9         25606       漢相國參之後
## 2           25606                 曹
## 2.1         25606                 瞞
## 2.2         25606                 曰
## 2.3         25606       太祖一名吉利
## 2.4         25606           小字阿瞞
## 3           25606         王沈魏書曰
## 3.1         25606       其先出於黃帝
## 4           25606           當高陽世
## 4.1         25606       陸終之子曰安
## 4.2         25606           是為曹姓
## 5           25606         周武王克殷
## 5.1         25606         存先世之後
## 5.2         25606         封曹俠於邾
## 6           25606           春秋之世
## 6.1         25606           與於盟會
## 6.2         25606           逮至戰國
## 6.3         25606           為楚所滅
## 7           25606           子孫分流
## 7.1         25606           或家於沛
## 8           25606         漢高祖之起
## 8.1         25606   曹參以功封平陽侯
## 8.2         25606           世襲爵土
## 8.3         25606               絕而
## 8.4         25606                 複
## 8.5         25606                 紹
## 8.6         25606   至今適嗣國於容城
## 9           25606               桓帝
## 9.1         25606                 世
## 9.2         25606 曹騰為中常侍大長秋
## 9.3         25606           封費亭侯
## 10          25606     司馬彪續漢書曰
## 10.1        25606             騰父節
## 10.2        25606             字元偉
## 10.3        25606         素以仁厚稱
## 11          25606       鄰人有亡豕者
## 11.1        25606         與節豕相類
## 11.2        25606           詣門認之
## 11.3        25606           節不與爭
## 11.4        25606   後所亡豕自還其家
## 11.5        25606         豕主人大慚

文字雲

# 計算詞彙的出現次數,如果詞彙只有一個字則不列入計算
true_tokens_count <- true_tokens %>% 
  filter(nchar(.$word)>1) %>%
  group_by(word) %>% 
  summarise(sum = n()) %>% 
  filter(sum>10) %>%
  arrange(desc(sum))

# 印出最常見的20個詞彙
head(true_tokens_count, 50)
## # A tibble: 50 x 2
##    word     sum
##    <chr>  <int>
##  1 將軍     845
##  2 太祖     609
##  3 太守     415
##  4 天下     382
##  5 陛下     204
##  6 不可     203
##  7 大將軍   202
##  8 天子     192
##  9 春秋     161
## 10 以為     155
## # ... with 40 more rows
#true_tokens_count %>% wordcloud2()

三國志 三國志作者為西晉陳濤,裡面包含《魏志》三十卷,《蜀志》十五卷,《吳志》二十卷,由此可見魏國在三國志的比例最高。陳壽對曹操的稱謂太祖、大將軍、王,始終稱劉備為先主,而對於吳主孫權,無甚感情,因而通篇以名字稱權。將軍和太守為官職。

水滸傳與三國演義詞彙進行比較

動機:網路上流傳著三國演義作者:羅貫中與水滸傳作者:施耐庵是師徒關係,因此想比較兩本書的詞彙

# 下載 "水滸傳" 書籍,並且將text欄位為空的行給清除,以及將重複的語句清除
water <- gutenberg_download(23863) %>% filter(text!="") %>% distinct(gutenberg_id, text)
doc = paste0(water$text,collapse = "") #將text欄位的全部文字合併
docVector = unlist(strsplit(doc,"[。.]"), use.names=FALSE) #以全形或半形句號斷句
water = data.frame(gutenberg_id = "23863" , text = docVector, stringsAsFactors = FALSE) #gutenberg_id換成自己的書本id

View(water)
# 使用水滸傳專有名詞字典
jieba_tokenizer <- worker(user="water_margin.traditional.dict", stop_word = "stop_words.txt")
# 設定斷詞function
water_tokenizer <- function(t) {
  lapply(t, function(x) {
    tokens <- segment(x, jieba_tokenizer)
    return(tokens)
  })
}
# 根據上方整理出來的規則,我們可以使用正規表示式,將句子區分章節
water <- water %>% 
  mutate(chapter = cumsum(str_detect(water$text, regex("^第.{1,3}回 ?"))))
water_tokens <- water %>% unnest_tokens(word, text, token=water_tokenizer)
str(water_tokens)
## 'data.frame':    195781 obs. of  3 variables:
##  $ gutenberg_id: chr  "23863" "23863" "23863" "23863" ...
##  $ chapter     : int  0 0 0 0 0 0 0 0 0 0 ...
##  $ word        : chr  "楔子" "張天師" "祈" "禳" ...
head(water_tokens, 20)
##      gutenberg_id chapter   word
## 1           23863       0   楔子
## 1.1         23863       0 張天師
## 1.2         23863       0     祈
## 1.3         23863       0     禳
## 1.4         23863       0   瘟疫
## 1.5         23863       0 洪太尉
## 1.6         23863       0   誤走
## 1.7         23863       0   妖魔
## 1.8         23863       0   紛紛
## 1.9         23863       0   五代
## 1.10        23863       0     亂
## 1.11        23863       0   離間
## 1.12        23863       0   雲開
## 1.13        23863       0     復
## 1.14        23863       0   見天
## 1.15        23863       0   草木
## 1.16        23863       0   百年
## 1.17        23863       0     新
## 1.18        23863       0   雨露
## 1.19        23863       0   車書

水滸傳文字雲

# 計算詞彙的出現次數,如果詞彙只有一個字則不列入計算
water_tokens_count <- water_tokens %>% 
  filter(nchar(.$word)>1) %>%
  group_by(word) %>% 
  summarise(sum = n()) %>% 
  filter(sum>20) %>%
  arrange(desc(sum))

# 印出最常見的20個詞彙
head(water_tokens_count, 20)
## # A tibble: 20 x 2
##    word     sum
##    <chr>  <int>
##  1 兩個    1260
##  2 宋江    1246
##  3 一個     983
##  4 武松     947
##  5 只見     665
##  6 李逵     642
##  7 哥哥     616
##  8 林沖     593
##  9 說道     538
## 10 頭領     478
## 11 小人     449
## 12 <U+8846>人     425
## 13 兄弟     421
## 14 婦人     400
## 15 吳用     371
## 16 今日     366
## 17 好漢     348
## 18 便是     336
## 19 問道     317
## 20 梁山泊   315
#water_tokens_count %>% wordcloud2()

水滸傳 計算前 三國演義與水滸傳的詞彙在全文中出現比率的差異由文字雲可發現水滸傳的量詞:「一個」、「兩個」的詞頻很高,武松與宋江因為是水滸傳裡著名的腳色,詞頻也很高

計算三國演義與水滸傳 的詞彙在全文中出現比率的差異

動機:因為是師徒關係,因此想比較兩本書詞彙出現比率是否呈現線性關係。會分別比較三國演義全文與水滸傳以及比較三國演義後40回與水滸傳的詞彙出現比例原因是網路上流傳著三國演義後40回的作者可能是施耐庵,因此特別再將三國演義後40回拿出來和水滸傳比較。

計算三國演義全文與水滸傳 的詞彙在全文中出現比率的差異

bind_tokens <- rbind(tokens, water_tokens)
frequency <- bind_tokens %>% select(-chapter) %>% 
  mutate(name = ifelse(gutenberg_id==23950, "three", "water")) %>%
  filter(nchar(.$word)>1) %>%
  mutate(word = str_extract(word, "[^0-9a-z']+")) %>%
  mutate(word = str_extract(word, "^[^一二三四五六七八九十]+")) %>%
  count(name, word)%>%
  group_by(name) %>%
  mutate(proportion = n / sum(n)) %>% 
  select(-n) %>% 
  spread(name, proportion) 

匯出圖表

plot1 <- ggplot(frequency, aes(x = `three`, y = `water`, color = abs(`three` - `water`))) +
  geom_abline(color = "gray40", lty = 2) +
  geom_jitter(alpha = 0.1, size = 2.5, width = 0.3, height = 0.3) +
  geom_text(aes(label = word), check_overlap = TRUE, vjust = 1.5) +
  scale_x_log10(labels = percent_format()) +
  scale_y_log10(labels = percent_format()) +
  scale_color_gradient(limits = c(0, 0.001), low = "darkslategray4", high = "gray75") +
  theme(legend.position="none") +
  labs(y = "water", x = "three")

計算前三國演義後40回與水滸傳的詞彙在全文中出現比率的差異

last40_three_tokens <- filter(tokens, chapter > 80)
bind_last40_tokens <- rbind(last40_three_tokens, water_tokens)
last40_frequency <- bind_last40_tokens %>% select(-chapter) %>% 
  mutate(name = ifelse(gutenberg_id==23950, "three", "water")) %>%
  filter(nchar(.$word)>1) %>%
  mutate(word = str_extract(word, "[^0-9a-z']+")) %>%
  mutate(word = str_extract(word, "^[^一二三四五六七八九十]+")) %>%
  count(name, word)%>%
  group_by(name) %>%
  mutate(proportion = n / sum(n)) %>% 
  select(-n) %>% 
  spread(name, proportion) 
plot2 <- ggplot(last40_frequency, aes(x = `three`, y = `water`, color = abs(`three` - `water`))) +
  geom_abline(color = "gray40", lty = 2) +
  geom_jitter(alpha = 0.1, size = 2.5, width = 0.3, height = 0.3) +
  geom_text(aes(label = word), check_overlap = TRUE, vjust = 1.5) +
  scale_x_log10(labels = percent_format()) +
  scale_y_log10(labels = percent_format()) +
  scale_color_gradient(limits = c(0, 0.001), low = "darkslategray4", high = "gray75") +
  theme(legend.position="none") +
  labs(y = "water", x = "three_last40")
par(mfrow = c(1,1))
plot1
## Warning: Removed 50318 rows containing missing values (geom_point).
## Warning: Removed 50319 rows containing missing values (geom_text).


圖一可以看到有些詞彙出現的比率相近,像是人馬、今日等,而比較特別的是孔明在水滸傳也有出現,他是水滸傳的一個腳色,但他的一生,跟諸葛亮毫無關係,除此之外也有呂布的詞彙,是因為有位水滸傳的腳色想要效仿三國的呂布,因此才會出現。

plot2
## Warning: Removed 34719 rows containing missing values (geom_point).
## Warning: Removed 34720 rows containing missing values (geom_text).


圖二與圖一相比,散佈圖並沒有更趨於線性