S3とS4は、オブジェクト指向プログラミングの公式(つまり組み込み)アプローチのようです。S3とコンストラクター関数/メソッドに埋め込まれた関数の組み合わせを使い始めました。私の目標は、object $ 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-directiond-programming-in-r確信が持てないため、このアプローチに対するコメント、批判、提案を歓迎します。これが最善のアプローチである場合は自分自身。しかし、私が解決しようとしていた問題については、うまくいきました。具体的には、makeRパッケージ(http://jbryer.github.com/makeR)の場合、オブジェクトの状態を表すXMLファイルの同期を維持する必要があるため、ユーザーがデータフィールドを直接変更することを望まなかった。これは、ユーザーがドキュメントで概説したルールを順守している限り、完全に機能しました。