Bas
Bas

Reputation: 1076

Replace value in one data table based on NA in another data table

I've got genotyping data from several overlapping NPs/individuals which I am attempting to compare. As you can see in the data structure below, e[1,2] and e[2,3] have NA's. Now I want to replace d[1,2](1) and d[2,3](1) by NA values.

d <- structure(list(`100099681` = c(0L, 2L, 0L), `101666591` = c(1L, 1L, 0L), `102247652` = c(1L, 1L, 1L), `102284616` = c(0L, 1L, 0L), `103582612` = c(0L, 1L, 1L), `104344528` = c(2L, 1L, 0L),     `105729734` = c(1L, 0L, 1L), `109897137` = c(0L, 0L, 2L),     `112768301` = c(0L, 1L, 1L), `114724443` = c(1L, 1L, 1L),     `114826164` = c(1L, 0L, 1L), `115358770` = c(0L, 2L, 0L),     `115399788` = c(1L, 1L, 0L), `118669033` = c(0L, 1L, 1L),     `118875482` = c(2L, 1L, 0L), `119366362` = c(0L, 2L, 0L),     `119627971` = c(0L, 1L, 1L), `120295351` = c(0L, 2L, 0L),     `120998030` = c(0L, 0L, 2L)), .Names = c("100099681", "101666591", "102247652", "102284616", "103582612", "104344528", "105729734", "109897137", "112768301", "114724443", "114826164", "115358770", "115399788", "118669033", "118875482", "119366362", "119627971", "120295351", "120998030"), row.names = c("7:100038150_C", "7:100079759_T", "7:100256942_A"), class = "data.frame") 
> d
#             100099681 101666591 102247652 102284616 103582612 104344528 105729734 109897137 112768301 114724443 114826164 115358770 115399788 118669033 118875482 119366362 119627971 120295351 120998030
#7:100038150_C         0         1         1         0         0         2         1         0         0         1         1         0         1         0         2         0         0         0        0
#7:100079759_T         2         1         1         1         1         1         0         0         1         1         0         2         1         1         1         2         1         2        0
#7:100256942_A         0         0         1         0         1         0         1         2         1         1         1         0         0         1         0         0         1         0        2

e<- structure(list(`100099681` = c(1L, 1L, 0L), `101666591` = c(NA, 1L, 1L), `102247652` = c(0L, NA, 0L), `102284616` = c(1L, 1L, 0L), `103582612` = c(1L, 0L, 1L), `104344528` = c(1L, 0L, 1L),     `105729734` = c(0L, 0L, 1L), `109897137` = c(1L, 1L, 0L),     `112768301` = c(0L, 1L, 1L), `114724443` = c(0L, 2L, 0L),     `114826164` = c(0L, 0L, 2L), `115358770` = c(0L, 0L, 2L),     `115399788` = c(0L, 2L, 0L), `118669033` = c(0L, 0L, 2L),     `118875482` = c(0L, 1L, 1L), `119366362` = c(2L, 1L, 0L),     `119627971` = c(0L, 1L, 1L), `120295351` = c(0L, 2L, 0L),     `120998030` = c(0L, 2L, 1L)), .Names = c("100099681", "101666591", "102247652", "102284616", "103582612", "104344528", "105729734", "109897137", "112768301", "114724443", "114826164", "115358770", "115399788", "118669033", "118875482", "119366362", "119627971", "120295351", "120998030"), row.names = c("7:100038150_C", "7:100079759_T", "7:100256942_A"), class = "data.frame")  
> e
#             100099681 101666591 102247652 102284616 103582612 104344528 105729734 109897137 112768301 114724443 114826164 115358770 115399788 118669033 118875482 119366362 119627971 120295351   120998030
#7:100038150_C         1        NA         0         1         1         1         0         1         0         0         0         0         0         0         0         2         0         0          0
#7:100079759_T         1         1        NA         1         0         0         0         1         1         2         0         0         2         0         1         1         1         2          2
#7:100256942_A         0         1         0         0         1         1         1         0         1         0         2         2         0         2         1         0         1         0          1

Thus my expected output would be

> expected_d
#             100099681 101666591 102247652 102284616 103582612 104344528 105729734 109897137 112768301 114724443 114826164 115358770 115399788 118669033 118875482 119366362 119627971 120295351 120998030
#7:100038150_C         0         NA         1         0         0         2         1         0         0         1         1         0         1         0         2         0         0         0        0
#7:100079759_T         2         1         NA         1         1         1         0         0         1         1         0         2         1         1         1         2         1         2        0
#7:100256942_A         0         0          1         0         1         0         1         2         1         1         1         0         0         1         0         0         1         0        2

I've gotten this far;

g <- which(is.na(e), arr.ind=TRUE)
> g
#              row col
#7:100038150_C   1   2
#7:100079759_T   2   3

Then trying to use an apply function to replace the location by "TEST" (or na for that matter)

apply(g, 1, function(x){   
    e[x[1], x[2]] <- "TEST" }
)
#>  apply(g, 1, function(x){   e[x[1], x[2]] <- "TEST" })
#7:100038150_C 7:100079759_T
#       "TEST"        "TEST"    

I will be running this bit of code over several million rows/columns so speed will be an issue. Thank you in advance:)

Upvotes: 1

Views: 53

Answers (2)

Sotos
Sotos

Reputation: 51582

Another way based on your approach,

d[which(is.na(e), arr.ind = T)] <- NA

Upvotes: 2

akrun
akrun

Reputation: 886938

We can try doing

 NA^(is.na(e))*d

If memory is an issue

d[] <- Map(function(x,y) NA^(is.na(y))* x, d, e)

Upvotes: 4

Related Questions