来自 Google 的 R 语言编码风格指南

R 语言是一门主要用于统计计算和绘图的高级编程语言。这份 R 语言编码风格指南旨在让我们的 R 代码更容易阅读、分享和检查。以下规则系与 Google 的 R 用户群体协同设计而成。


概要:R编码风格约定

一、表示和命名

  • 文件命名:以.R(大写)结尾
  • 标识符命名:variable.name(或variableName),FunctionName,kConstantName

二、语法

  • 单行长度:不超过80个字符
  • 缩进:两个空格,不使用制表符
  • 空格
  • 花括号:前括号不折行写,后括号独占一行
  • else语句:else两侧必须与括号连接
  • 赋值符号:使用<−,而非=
  • 分号:勿用分号

三、代码组织

  • 总体布局和顺序
  • 注释准则:所有注释以#开始,后接一个空格;行内注释需要在#前加两个空格
  • 函数的定义和调用
  • 函数文档
  • 示例函数
  • TODO 书写风格:TODO(用户名)

概要:R语言使用规则

四、语言

  • attach:避免使用
  • 函数:错误(error)应当使用stop()抛出
  • 对象和方法:尽可能避免使用 S4 对象和方法;永远不要混用 S3 和 S4

五、例外

六、结语

七、参考文献


一、表示和命名

  • 文件命名

    文件名应以.R(大写)结尾,文件名本身要有意义。
    正例:predict_ad_revenue.R
    反例:foo.R

  • 标识符命名

    在标识符中不要使用下划线_或连字符-。标识符应根据如下惯例命名:

    1. 变量命名——应使用点.分隔所有的小写字母或单词(例如最佳命名格式variable.name),variableName亦可;
      正例:avg.clicks
      可用:avgClicks
      反例:avg_Clicks
    2. 函数命名——应为动词或动词性短语,首字母大写,不用点分隔(例如FunctionName);
      正例:CalculateAvgClicks
      反例:calculate_avg_clicks,calculateAvgClicks
      例外:当创建一个含类(class)属性的对象时,函数名(也是constructor)和类名(class)应当匹配(例如lm)。
    3. 常数命名——规则同函数,但需以字母k开头(例如kConstantName)。

二、语法

  • 单行长度

    单行最大长度为 80 个字符。

  • 缩进

    使用两个空格来缩进代码。切忌使用制表符或将二者混合使用。
    例外:当括号内发生折行时,所折行与括号内的第一个字符对齐。

  • 空格

    在所有二进制操作符(=,+,-,<−等等)的两侧加上空格。
    不可在逗号前加空格,而逗号后必须加空格。
    例外:在函数调用中传递参数时=两边的空格可加可不加。

# 正例
tabPrior <- table(df[df$daysFromOpt < 0, "campaignid"])
total <- sum(x[, 1])
total <- sum(x[1, ])
# 反例
tabPrior <- table(df[df$daysFromOpt<0, "campaignid"])  # '<'两侧需要增加空格
tabPrior <- table(df[df$daysFromOpt < 0,"campaignid"])  # 逗号后需要增加一个空格
tabPrior<- table(df[df$daysFromOpt < 0, "campaignid"])  # <-前需要增加一个空格
tabPrior<-table(df[df$daysFromOpt < 0, "campaignid"])  # '<-'两侧需要增加空格
total <- sum(x[,1])  # 逗号后需要增加一个空格
total <- sum(x[ ,1])  # 逗号后需要增加一个空格,而非逗号之前

  在前括号前加一个空格,函数调用时除外。

# 正例
if (debug)
# 反例
if(debug)

  如果能够改善等号或箭头(<−)的对齐效果,也可以在同一行内使用多个空格。

plot(x    = xCoord,
     y    = dataMat[, makeColName(metric, ptiles[1], "roiOpt")],
     ylim = ylim,
     xlab = "dates",
     ylab = metric,
     main = (paste(metric, " for 3 samples ", sep="")))

  不要在圆括号或方括号的两侧加入空格。
  例外:逗号后总须加空格。

# 正例
if (debug)
x[1, ]
# 反例
if ( debug )  # debug两侧不要加空格
x[1,]  # 逗号后需要增加一个空格 
  • 花括号

    前括号切忌独占一行;而后括号应独占一行。
    当代码块只含单个语句时,可以省略花括号。
    但在处理这类单个语句时,必须前后一致:要么全部使用花括号,或者全部不用花括号。

# 正例
if (is.null(ylim)) {
  ylim <- c(0, 0.06)
}

  或用以下形式(二者不可混用):

if (is.null(ylim))
  ylim <- c(0, 0.06)

  必须另起一行书写代码块的主体部分。

# 反例
if (is.null(ylim)) ylim <- c(0, 0.06) 
if (is.null(ylim)) {ylim <- c(0, 0.06)}
  • else两侧必须与括号连接

    else两侧必须与花括号在同一行。

# 正例
if (condition) {
  one or more lines
} else {
  one or more lines
}
# 反例
if (condition) {
  one or more lines
}
else {
  one or more lines
}
# 反例
if (condition)
  one line
else
  one line
  • 赋值

    使用<−进行赋值,不用=赋值。

# 正例
x <- 5
# 反例
x = 5
  • 分号

    不要以分号结束一行,也不要利用分号在同一行放多于一个命令。
    没有必要使用分号,为了与其他Google编码风格指南保持一致,此处同样略去。


三、代码组织

  • 总体布局和顺序

    如果所有人都以相同顺序安排代码内容,我们就可以更加轻松快速地阅读并理解他人的脚本。

    1. 版权声明注释
    2. 作者信息注释
    3. 文件描述注释,包括程序的用途,输入和输出
    4. source()和 library()语句
    5. 函数定义
    6. 待执行的语句(若有的话),例如print,plot

    单元测试应在另一个名为originalfilename_test.R的独立文件中进行。

  • 注释准则

    给代码加注释,整行注释应在#后接一个空格开始。
    行内短注释应在代码后接两个空格,#,再接一个空格。

# 生成一个频率直方图,以描述campaigns的预算百分比
    hist(df$pctSpent,
    breaks = "scott",  # 定义柱形数量的参数
    main   = "Histogram: fraction budget spent by campaignid",
    xlab   = "Fraction of budget spent",
    ylab   = "Frequency (count of campaignids)")
  • 函数的定义和调用

    函数定义应首先列出无默认值的参数,然后再列出有默认值的参数。
    函数定义和函数调用中,允许每行写多个参数;折行只允许在赋值语句外进行。

# 正例
PredictCTR <- function(query, property, numDays,
                       showPlot = TRUE)
# 反例
PredictCTR <- function(query, property, numDays, showPlot =
                       TRUE)

  理想情况下,单元测试应该充当函数调用的样例(对于包中的程序来说)。

  • 函数文档

    函数的定义行下方应该紧接一个注释区。注释应当包含如下内容:

    1. 用一句话描述此函数;
    2. 此函数的参数列表,用Args:表示,及对每个参数的描述(包括数据类型);
    3. 说明返回值,用Returns:表示。

    注释部分应当对函数做充分解释,这样调用者无须阅读函数中的任何代码即可使用此函数。

  • 示例函数

CalculateSampleCovariance <- function(x, y, verbose = TRUE) {
  # 计算两个向量之间的样本协方差
  #
  # Args:
  #   x: 两个向量的其中一个,其样本协方差待计算
  #   y: 两个向量中的另外一个,x和y的长度必须一致,且长度大于1
  #      没有缺失值
  #   verbose: 若为TRUE,则输出样本协方差;否则不输出。默认值为TRUE
  #
  # Returns:
  #   x和y的样本协方差
  n <- length(x)
  # Error handling
  if (n <= 1 || n != length(y)) {
    stop("Arguments x and y have invalid lengths: ",
         length(x), " and ", length(y), ".")
  }
  if (TRUE %in% is.na(x) || TRUE %in% is.na(y)) {
    stop(" Arguments x and y must not have missing values.")
  }
  covariance <- var(x, y)
  if (verbose)
    cat("Covariance = ", round(covariance, 4), ".\n", sep = "")
  return(covariance)
}
  • TODO 书写风格

    编码时通篇使用一种一致的风格来书写 TODO。
    TODO(用户名):所要采取行动的明确描述


四、语言

  • Attach

    使用attach造成错误的不胜枚举,避免使用之。

  • 函数

    错误(error)应当使用stop()抛出。

  • 对象和方法

    S 语言中有两套面向对象系统:S3 和 S4,在 R 中这两套均可使用。
    S3 方法的可交互性更强,更加灵活;反之,S4 方法更加正式和严格。
    对这两套系统的说明,参见Thomas Lumley的文章Programmer's Niche: A Simple Class, in S3 and S4(发表于R News 4/1,2004,33-36页:https://cran.r-project.org/doc/Rnews/Rnews_2004-1.pdf
    这里推荐使用S3对象和方法,除非您有很强烈的理由去使用S4对象和方法。
    使用S4对象的一个主要理由是,在C++代码中直接使用对象。
    使用一个 S4 泛型/方法的主要理由是对双参数的分发。
    避免混用 S3 和 S4:S4 方法会忽略 S3 中的继承,反之亦然。


五、例外

除非有不去这样做的正当理由,否则应当遵循以上描述的编码惯例。例外包括遗留代码的维护和对第三方代码的修改。


六、结语

遵守常识,前后一致。
如果您在编辑现有代码,花几分钟看看代码的上下文并弄清它的风格。
如果其他人在 if 语句周围使用了空格,那您也应该这样做。
如果他们的注释是用星号组成的小盒子围起来的,那您也要这样写。
遵循编码风格准则的意义在于,人们相当于有了一个编程的通用词汇表。于是人们可以专注于您在说什么,而不是您是怎么说的。
我们在这里提供全局的编码风格规则以便人们了解这些词汇,但局部风格也很重要。
如果您加入文件中的代码看起来和周围的已有代码截然不同,那么代码阅读者的阅读节奏就会被破坏,尽量避免这样做。
OK,关于如何写代码已经写得够多了,代码本身要有趣的多。编码愉快!


七、参考文献

R语言编码惯例
为emacs用户而生,在您的emacs中运行R并产生一个emacs模式


参考资料:
Google’s R Style Guide
来自 Google 的 R 语言编码风格指南

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值