S3 та S4, схоже, є офіційними (тобто вбудованими) підходами до програмування ОО. Я почав використовувати комбінацію S3 з функціями, вбудованими в функцію / метод конструктора. Моєю метою було мати синтаксис типу $ method (), щоб у мене були напівзакриті поля. Я кажу напівприватним, тому що насправді їх неможливо приховати (наскільки мені відомо). Ось простий приклад, який насправді нічого не робить:
EmailClass <- function(name, email) {
nc = list(
name = name,
email = email,
get = function(x) nc[[x]],
set = function(x, value) nc[[x]] <<- value,
props = list(),
history = list(),
getHistory = function() return(nc$history),
getNumMessagesSent = function() return(length(nc$history))
)
nc$sendMail = function(to) {
cat(paste("Sending mail to", to, 'from', nc$email))
h <- nc$history
h[[(length(h)+1)]] <- list(to=to, timestamp=Sys.time())
assign('history', h, envir=nc)
}
nc$addProp = function(name, value) {
p <- nc$props
p[[name]] <- value
assign('props', p, envir=nc)
}
nc <- list2env(nc)
class(nc) <- "EmailClass"
return(nc)
}
print.EmailClass <- function(x) {
if(class(x) != "EmailClass") stop();
cat(paste(x$get("name"), "'s email address is ", x$get("email"), sep=''))
}
І кілька тестових кодів:
test <- EmailClass(name="Jason", "jason@bryer.org")
test$addProp('hello', 'world')
test$props
test
class(test)
str(test)
test$get("name")
test$get("email")
test$set("name", "Heather")
test$get("name")
test
test$sendMail("jbryer@excelsior.edu")
test$getHistory()
test$sendMail("test@domain.edu")
test$getNumMessagesSent()
test2 <- EmailClass("Nobody", "dontemailme@nowhere.com")
test2
test2$props
test2$getHistory()
test2$sendMail('nobody@exclesior.edu')
Ось посилання на допис у блозі, який я писав про цей підхід: http://bryer.org/2012/object-oriented-programming-in-r. Я вітаю коментарі, критику та пропозиції щодо цього підходу, оскільки я не переконаний я, якщо це найкращий підхід. Однак проблема, яку я намагався вирішити, чудово спрацювала. Зокрема, для пакету makeR ( http://jbryer.github.com/makeR ) я не хотів, щоб користувачі безпосередньо змінювали поля даних, тому що мені потрібно було забезпечити синхронізацію XML-файлу, що представляє стан мого об’єкта. Це працювало чудово, доки користувачі дотримуються правил, які я виклав у документації.