.
in their name and are not necessarily the methods of the function print. Hence, amore elegant ways is to use the purpose build function
methods()
.
methods()
methods(methods) ## no methods found methods(mean) ## [1] mean.Date mean.default mean.difftime ## [4] mean.POSIXct mean.POSIXlt ## see ‘?methods’ for accessing help and source code
Do not use the dot “.” in function names because it makes them look like S3 functional methods. This might lead to confusion with the convention that the methods are named as <<generic function>>.<<class name>>
. Especially, if there is more than one dot in the name. For example, print.data.frame()
is not univocal: is it a dataframe method for the generic function print or is it the frame method for the generic function print.data? Another example is the existence of the function t.test()
to run t-tests as well as t.dataframe()
, that is the S3 method for the generic function t()
to transpose a data frame.
t.test()
t.data.frame()
t()
To access the source code of the class-specific methods, one can use the function getS3method()
.
getS3method()
getS3method(“print”,“table”) ## function (x, digits = getOption(“digits”), quote = FALSE, na.print = “”, ## zero.print = “0”, justify = “none”, …) ## { ## d <- dim(x) ## if (any(d == 0)) { ## cat(“< table of extent”, paste(d, collapse = “ x “), ## “>\n”) ## return(invisible(x)) ## } ## xx <- format(unclass(x), digits = digits, justify = justify) ## if (any(ina <- is.na(x))) ## xx[ina] <- na.print ## if (zero.print != “0” && any(i0 <- !ina & x == 0)) ## xx[i0] <- zero.print ## if (is.numeric(x) || is.complex(x)) ## print(xx, quote = quote, right = TRUE, …) ## else print(xx, quote = quote, …) ## invisible(x) ## } ## <bytecode: 0x5634250f12e8> ## <environment: namespace:base>
The other way around it is possible to list all generic functions that have a specific method for a given class.
You can also list all generics that have a method for a given class:
methods(class = “data.frame”) ## [1] [ [[ [[<- ## [4] [<- $ $<- ## [7] aggregate anyDuplicated as.data.frame ## [10] as.list as.matrix by ## [13] cbind coerce dim ## [16] dimnames dimnames<- droplevels ## [19] duplicated edit format ## [22] formula head initialize ## [25] is.na Math merge ## [28] na.exclude na.omit Ops ## [31] plot print prompt ## [34] rbind row.names row.names<- ## [37] rowsum show slotsFromS3 ## [40] split split<- stack ## [43] str subset summary ## [46] Summary t tail ## [49] transform unique unstack ## [52] within ## see ‘?methods’ for accessing help and source code
6.2.1 Creating S3 Objects
S3 is a minimalistic and informal OO system; there is not even a formal definition of a class. In S3, you never create a class definition and start from the instance itself. Actually, to create an S3 object, it is sufficient to set its class attribute.
S3
my_curr_acc <- list(“name” = “Philippe”, “balance” <- 100) class(my_curr_acc) <- “account” # set the class attribute otype(my_curr_acc) # it is an S3 object ## [1] “S3” class(my_curr_acc) # the class type is defined above ## [1] “account” # Note that the class attribute is not visible in the structure: str(my_curr_acc) ## List of 2 ## $ name: chr “Philippe” ## $ : num 100 ## - attr(*, “class”)= chr “account”
It is also possible to create a class and set the attribute simultaneously with the function structure
.
structure()
my_object <- structure(list(), class = “boringClass”)
To create methods for S3 generic function, all we have to do is follow the syntax: <<generic function>>.<<class name>>
. R will then make sure that if an object of “class name” calls the generic function that then the generic function will dispatch the action to this specific function.
generic function
# print.account # Print an object of type ‘account’ # Arguments: # x -- an object of type account print.account <- function(x){ print(paste(“account holder”,x[[1]],sep=”: “)) print(paste(“balance “ ,x[[2]],sep=”: “)) } print(my_curr_acc) ## [1] “account holder: Philippe” ## [1] “balance : 100”
S3 objects are always build on other more elementary types. The function inherits (x, “classname”)
allows the user to determine if the class x inherits from the class “classname.”
You probably remember that R returned “internal” “generic”
as the class for some functions; so the class can be a vector. That means that the behaviour of that object can depend on different class-specific methods. The classes have to be listed from from most to least specific, so that the behaviour can follow this cascade and will always execute the most specific behaviour if it is present.
For example, the class of the glm()
object is c(“glm”, “lm”)
. This means that the most specific class is the generalised linear model, but that some behaviour they might inherit from linear models. When a generic function will be called, it will first try to find a glm-specific method. If that fails, it will look for the lm-method.
It is possible to provide a constructor function for an S3 class. This constructor function can, for example be used to check if we use the right data-type for its attributes.
constructor
# account # Constructor function for an object of type account # Arguments: # x -- character (the name of the account holder) # y -- numeric (the initial balance of the account # Returns: # Error message in console in case of failure. account <- function(x,y) { if (!is.numeric(y)) stop(“Balance must be numeric!”) if (!is.atomic(x)) stop(“Name must be atomic!!”) if (!is.character(x)) stop(“Name must be a string!”) structure(list(“name” = x, “balance” = y), class = “account”) } # create a new instance for Paul: paul_account <- account(“Paul”, 200) # print the object with print.account(): paul_account ## [1] “account holder: Paul” ## [1] “balance : 200”
The advantage of using the creator function for an instance of an object is obvious: it will perform some checks and will avoid problems later on. Unlike in message-passing OO implementations, the S3 implementation allows to bypass the creator function or worse it allows you to change the class all too easy. Consider the following example.
creator function
class(paul_account)