Coalesce two string columns with alternating missing values to one
Asked Answered
F

8

15

I have a data frame with two columns "a" and "b" with alternating missing values (NA)

a      b
dog    <NA>
mouse  <NA>
<NA>   cat
bird   <NA>

I want to "merge" / combine them to a new column c that looks like this, i.e. the non-NA element in each row is selected:

c
dog
mouse
cat
bird

I tried merge and join, but neither worked as I wanted. Maybe because I do not have an id with which to merge? For integers I would just circumvent this and add both columns, but how in my case?

Fonz answered 8/1, 2015 at 22:2 Comment(1)
Are those real NA values or fake?Parashah
R
12

You may try pmax

df$c <- pmax(df$a, df$b)
df
#       a    b     c
# 1   dog <NA>   dog
# 2 mouse <NA> mouse
# 3  <NA>  cat   cat
# 4  bird <NA>  bird

...or ifelse:

df$c <- ifelse(is.na(df$a), df$b, df$a)

For more general solutions in cases with more than two columns, you find several ways to implement coalesce in R here.

Richy answered 8/1, 2015 at 22:25 Comment(1)
the best solution for me was the second option using ifelse. ThanksCancer
E
12

dplyr has exactly what you are looking for, function coalesce()

library(dplyr)

a<-c("dog","mouse",NA,"bird")
b<-c(NA,NA,"cat",NA)

coalesce(a,b)

[1] "dog"   "mouse" "cat"   "bird"
Excited answered 28/8, 2018 at 15:35 Comment(0)
W
9

I wrote a coalesce() function for this type of task which works much like the SQL coalesce function. You would use it like

dd<-read.table(text="a      b
dog    NA
mouse  NA
NA   cat
bird   NA", header=T)

dd$c <- with(dd, coalesce(a,b))
dd
#       a    b     c
# 1   dog <NA>   dog
# 2 mouse <NA> mouse
# 3  <NA>  cat   cat
# 4  bird <NA>  bird
Willaims answered 8/1, 2015 at 22:15 Comment(0)
P
5

Here's my attempt (modified by @MrFlick)

df$c <- apply(df, 1, function(x) na.omit(x)[1])
df
#       a    b     c
# 1   dog <NA>   dog
# 2 mouse <NA> mouse
# 3  <NA>  cat   cat
# 4  bird <NA>  bird
Parashah answered 8/1, 2015 at 22:17 Comment(3)
Wouldn't apply(df, 1, function(x) na.omit(x)[1]) work just as well here, and be a bit simpler?Willaims
I would also use df[which(!is.na(df), arr.ind=TRUE)]Patrizia
@akrun, that is very nice vectoerized approach. I would post it as your own answerParashah
P
5

Another option is to use which with arr.ind=TRUE

indx <- which(!is.na(df), arr.ind=TRUE)
df$c <-  df[indx][order(indx[,1])]
df
#    a    b     c
#1   dog <NA>   dog
#2 mouse <NA> mouse
#3  <NA>  cat   cat
#4  bird <NA>  bird

Or

df$c <- df[cbind(1:nrow(df),max.col(!is.na(df)))]
Patrizia answered 9/1, 2015 at 11:56 Comment(0)
H
2

You could use a simple apply :

df$c <- apply(df,1,function(x)  x[!is.na(x)]  ) 

> df
      a    b     c
1   dog <NA>   dog
2 mouse <NA> mouse
3  <NA>  cat   cat
4  bird <NA>  bird
Hawkbill answered 8/1, 2015 at 22:17 Comment(0)
C
2

Using if else logic:

a<-c("dog","mouse",NA,"bird")
b<-c(NA,NA,"cat",NA)

test.df <-data.frame(a,b, stringsAsFactors = FALSE)
test.df$c <- ifelse(is.na(test.df$a), test.df$b, test.df$a)

test.df

      a    b     c
1   dog <NA>   dog
2 mouse <NA> mouse
3  <NA>  cat   cat
4  bird <NA>  bird
Crustaceous answered 16/7, 2020 at 21:52 Comment(0)
L
0

Use tidyr::unite to be safe in case of a row containing two values:

df <- df |>
 unite(c,
       c(a, b),
       remove = FALSE,
       na.rm = TRUE)
Leveller answered 3/7, 2023 at 13:39 Comment(0)

© 2022 - 2025 — McMap. All rights reserved.