25

How do I aggregate data frame by group in column group and collapse text in column text?

Sample data:

df <- read.table(header=T, text="
group text
a a1
a a2
a a3
b b1
b b2
c c1
c c2
c c3
")

Required output (data frame):

group text
a     a1a2a3
b     b1b2
c     c1c2c3

Now I have:

sapply(unique(df$group), function(x) {
  paste0(df[df$group==x,"text"], collapse='')
})

This works to some extent as it returns text properly collapsed by group, but as a vector:

[1] "a1a2a3" "b1b2"   "c1c2c3"

I need a data frame with group column as a result.

rawr
  • 19,873
  • 4
  • 42
  • 74
Tomas Greif
  • 20,087
  • 21
  • 101
  • 151

2 Answers2

34

Simply use aggregate :

aggregate(df$text, list(df$group), paste, collapse="")
##   Group.1      x
## 1       a a1a2a3
## 2       b   b1b2
## 3       c c1c2c3

Or with plyr

library(plyr)
ddply(df, .(group), summarize, text=paste(text, collapse=""))
##   group   text
## 1     a a1a2a3
## 2     b   b1b2
## 3     c c1c2c3

ddply is faster than aggregate if you have a large dataset.

EDIT : With the suggestion from @SeDur :

aggregate(text ~ group, data = df, FUN = paste, collapse = "")
##   group   text
## 1     a a1a2a3
## 2     b   b1b2
## 3     c c1c2c3

For the same result with earlier method you have to do :

aggregate(x=list(text=df$text), by=list(group=df$group), paste, collapse="")

EDIT2 : With data.table :

library("data.table")
dt <- as.data.table(df)
dt[, list(text = paste(text, collapse="")), by = group]
##    group   text
## 1:     a a1a2a3
## 2:     b   b1b2
## 3:     c c1c2c3
Victorp
  • 12,926
  • 1
  • 46
  • 54
  • 3
    using the formula form of `aggregate`gives prettier name : aggregate( text ~ group, data = df, FUN = paste, collapse = "") – SeDur Mar 31 '14 at 10:09
  • @rawr that's in the first edit – Victorp Jan 17 '16 at 16:34
  • The non-formula `aggregate` doesn't need to be as torturous either - `aggregate(df["text"], df["group"], paste, collapse="")` will do it just fine. – thelatemail Mar 03 '16 at 05:57
27

You can use dplyr package for this

library(dplyr)

df %>%
  group_by(group) %>%
  summarise(text=paste(text,collapse=''))
David Arenburg
  • 89,637
  • 17
  • 130
  • 188
Chitrasen
  • 1,626
  • 18
  • 15