如何实现S4方法的子集替换

时间:2015-05-04 11:36:06

标签: r s4

我正在编写一个S4类,其中内部数据存储在数据库中,该类主要是用于访问和修改数据库中信息的守门员。该类将使用getInfoAgetInfoA<-等方法来提取和设置某些信息。

我的问题与以下案件有关:

myObject <- new('myClass', db='path/to/database')
getInfoA(myObject)[1:5] <- letters[1:5]

这里的setter在赋值之前是子集的。通常,当数据存储在标准R结构中时,这会自动解决,但是当数据存储在别处时如何正常处理? R中有一个[<-原语,但我不清楚调度是如何进行的,以及在何处以及如何拦截它......

1 个答案:

答案 0 :(得分:2)

可悲的是,我没有很好的解释,但它开箱即用。 也许R专家可以澄清这一点。

主要原因可能是R永远不会替换任何内容,只会创建一个新副本 对象(有一些例外,如原始运算符,例如[[<- 在某些情况下可以替换它。)

myClass <- setClass("myClass", slots=c(letters="character"))

setGeneric("getLetters", function(x)standardGeneric("getLetters"))
setGeneric("getLetters<-", function(x, value)standardGeneric("getLetters<-"))

setMethod("getLetters", "myClass", function(x) {
  x@letters
})

setReplaceMethod("getLetters", c("myClass", "character"), function(x, value) {
  message("value: ", paste0(value, collapse=", "))
  x@letters <- value
  x
})

a <- myClass(letters=LETTERS[1:10])
tracemem(a)
# [1] "<0x3716b40>"
getLetters(a)
#  [1] "A" "B" "C" "D" "E" "F" "G" "H" "I" "J"
getLetters(a)[1:5] <- letters[1:5]
# tracemem[0x3716b40 -> 0x39439c8]:
# tracemem[0x39439c8 -> 0x3293f70]:
# value: a, b, c, d, e, F, G, H, I, J
# tracemem[0x3293f70 -> 0x34aae60]: getLetters<- getLetters<-

因此,当您致电getLetters(a)[1:5] <- letters[1:5]时,基本上似乎发生了以下情况:

value <- getLetters(a)
value <- c(letters[1:5], value[6:10])
a <- `getLetters<-`(a, value=value)