I am interested in setting new methods for generic functions. For example, let's say I have a new class (e.g. coolClass
). I could write a wrapper to calculate the log10
of that class and easily set the method with the following code:
setMethod("Math", c(x="coolClass"),
function(x)
{
op = .Generic[[1]]
switch(op,
`log10` = log10_for_coolClass(x),
stop("Undefined operation")
)
}
)
However, I am unable to figure out how to set a method to pass multiple arguments. For example, the generic log
method. Running getGeneric("log")
shows the following:
> getGeneric("log")
standardGeneric for "log" defined from package "base"
belonging to group(s): Math
function (x, ...)
standardGeneric("log", .Primitive("log"))
<bytecode: 0x2f9a958>
<environment: 0x2f937f0>
Methods may be defined for arguments: x
Use showMethods("log") for currently available ones.
Seeing this, I would expect I could write the following to be able to pass the optional base
argument.
setMethod("Math", signature(x="coolClass",...),
function(x, ...)
{
op = .Generic[[1]]
switch(op,
`log` = log_for_coolClass(x, ...),
stop("Undefined operation")
)
}
)
But I get the following error:
Error in matchSignature(signature, fdef, where) :
'...' used in an incorrect context
Tried without ...
in the signature
I get a different error:
Error in rematchDefinition(definition, fdef, mnames, fnames, signature) :
methods can add arguments to the generic ‘Math’ only if '...' is an argument to the generic
Which seems strange to me given that getGeneric
log shows the ...
in the method.
Any ideas? Is there a way to capture additional arguments? I am trying to become more comfortable with S4 methods but I am confused about how to pass optional arguments. If this is not possible, I would appreciate it if someone could explain how the log
function does work, for example, given that the function is part of the Math
group but accepts multiple arguments.
Update
Curiously, as noted below, I can use setMethod
directly on log
with the following:
setMethod("log", signature(x="big.matrix"),
function(x, base=exp(1))
{
log_for_coolClass(x, base=base)
}
)
However, this doesn't quite pacify my curiosity. For example, it seems strange to be so repetitive in code if I creating many new methods within the Math
group. Ideally it would look something like this:
setMethod("Math", c(x="coolClass"),
function(x, base=exp(1))
{
op = .Generic[[1]]
switch(op,
`log10` = log10_for_coolClass(x),
`log` = log_for_coolClass(x, base=base),
stop("Undefined operation")
)
}
)
Here's a class
For a function like
log
we havewhere the signature includes
...
. So we writeand have success
Next up: log10
No
...
arguments in the generic, so we're basically stuck -- write our own generic and implement methods that do include...
, or write a log10,A-method that does not have...
argument.The same principle with respect to
...
applies to group generics -- the Math group generic doesn't have dotsso our method (implementing all Math operations, not just log; note that we don't typically reference .Generic explicitly) on objects of our class might be
and now we have methods for all the Math functions, e.g.,
also, our variant of log, accepting ..., is still available, so we can both define behavior of our class for all Math generics, and provide specialized implementations for specific generics.
There is probably a bug in the implementation of these group generics that should be addressed. If in a new session we defined our class and
Math
generic, but not thelog
function with...
argument, we can still invokewhere the second argument is ignored but should really cause an error.